CF 161B Discounts(贪心)

时间:2022-08-31 12:33:18

题目链接: 传送门

Discounts

time limit per test:3 second     memory limit per test:256 megabytes

Description

One day Polycarpus stopped by a supermarket on his way home. It turns out that the supermarket is having a special offer for stools. The offer is as follows: if a customer's shopping cart contains at least one stool, the customer gets a 50% discount on the cheapest item in the cart (that is, it becomes two times cheaper). If there are several items with the same minimum price, the discount is available for only one of them!
Polycarpus has k carts, and he wants to buy up all stools and pencils from the supermarket. Help him distribute the stools and the pencils among the shopping carts, so that the items' total price (including the discounts) is the least possible.
Polycarpus must use all k carts to purchase the items, no shopping cart can remain empty. Each shopping cart can contain an arbitrary number of stools and/or pencils.

Input

The first input line contains two integers n and k (1 ≤ k ≤ n ≤ 103) — the number of items in the supermarket and the number of carts, correspondingly. Next n lines describe the items as "ci ti" (without the quotes), where ci (1 ≤ ci ≤ 109) is an integer denoting the price of the i-th item, ti (1 ≤ ti ≤ 2) is an integer representing the type of item i (1 for a stool and 2 for a pencil). The numbers in the lines are separated by single spaces.

Output

In the first line print a single real number with exactly one decimal place — the minimum total price of the items, including the discounts.
In the following k lines print the descriptions of the items in the carts. In the i-th line print the description of the i-th cart as "t b1 b2 ... bt" (without the quotes), where t is the number of items in the i-th cart, and the sequence b1, b2, ..., bt (1 ≤ bj ≤ n) gives the indices of items to put in this cart in the optimal distribution. All indices of items in all carts should be pairwise different, each item must belong to exactly one cart. You can print the items in carts and the carts themselves in any order. The items are numbered from 1 to n in the order in which they are specified in the input.
If there are multiple optimal distributions, you are allowed to print any of them.

Sample Input

3 2
2 1
3 2
3 1

4 3
4 1
1 2
2 2
3 2

Sample Output

5.5
2 1 2
1 3

8.0
1 1
2 4 2
1 3

解题思路:

题目大意:商场打折,如果购买的商品包含凳子,则购物车中最便宜的商品能打五折,如果有多件价格一样低,仅只一件能打折,同时K辆购物车必须均有物品,问最少花费。
一开始无脑的分情况,分到最后写得自己都乱了,才开始仔细回想这个购物过程。其实不管怎么买,一定要让凳子最优打折。比如,凳子的价格比铅笔的高,那么凳子肯定要自己一辆购物车,因为再进来一件商品,购物车中最低价就变小,能打折的优惠就少了,另外,凳子的价格比铅笔低,那么无论是自己放还是和铅笔一起放,折扣也在凳子这。最后需要分类的也变得简单,就是分凳子的总数比K大还是小就行了。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const double INF = 0x3f3f3f3f;

struct Node
{
    double val;
    int id;
};

bool cmp(Node x,Node y)
{
    return x.val > y.val;
}

int main()
{
    int N,K;
    while (~scanf("%d%d",&N,&K))
    {
        double tmp,sum = 0,smin = INF,smax = 0,pmin = INF,pmax = 0,sum1 = 0,sum2 = 0;
        int opt;
        Node stool[1005],pencil[1005];
        memset(stool,0,sizeof(stool));
        memset(pencil,0,sizeof(pencil));
        int s = 0, p = 0;
        for (int i = 1; i <= N; i++)
        {
            scanf("%lf %d",&tmp,&opt);
            if (opt == 1)
            {
                stool[s].id = i;
                stool[s++].val = tmp;
                smin = min(smin,tmp);
                smax = max(smax,tmp);
                sum1 += tmp;
            }
            else
            {
                pencil[p].id = i;
                pencil[p++].val = tmp;
                pmax = max(pmax,tmp);
                pmin = min(pmin,tmp);
                sum2 += tmp;
            }
        }
        if (s >= K)
        {
            sort(stool,stool+s,cmp);
            for (int i = 0; i < K - 1; i++)
            {
                sum += stool[i].val;
            }
            sum *= 0.5;
            for (int i = K - 1; i < s; i++)
            {
                sum += stool[i].val;
            }
            sum += sum2;
            double minn  = min(pmin,smin);
            sum -= minn*0.5;
            printf("%.1lf\n",sum);
            for (int i = 0; i < K - 1; i++)
            {
                printf("1 %d\n",stool[i].id);
            }
            printf("%d",p+s-K+1);
            for (int i = K - 1; i < s; i++)
            {
                printf(" %d",stool[i].id);
            }
            for (int i = 0; i < p; i++)
            {
                printf(" %d",pencil[i].id);
            }
            printf("\n");
        }
        else
        {
            sum1 *= 0.5;
            sum += sum1 + sum2;
            printf("%.1lf\n",sum);
            for (int i = 0; i < s; i++)
            {
                printf("1 %d\n",stool[i].id);
            }
            for (int i = 0 ; i < K - s - 1; i++)
            {
                printf("1 %d\n",pencil[i].id);
            }
            printf("%d",p-(K-s-1));
            for (int i = K - s - 1; i < p; i++)
            {
                printf(" %d",pencil[i].id);
            }
            printf("\n");
        }
    }
    return 0;
}

CF 161B Discounts(贪心)的更多相关文章

  1. Codeforces 731B Coupons and Discounts&lpar;贪心&rpar;

    题目链接 Coupons and Discounts 逐步贪心即可. 若当前位为奇数则当前位的下一位减一,否则不动. #include <bits/stdc++.h> using name ...

  2. CF 949D Curfew——贪心&lpar;思路&excl;&excl;&excl;&rpar;

    题目:http://codeforces.com/contest/949/problem/D 有二分答案的思路. 如果二分了一个答案,首先可知越靠中间的应该大约越容易满足,因为方便把别的房间的人聚集过 ...

  3. Codeforces 161 B&period; Discounts &lpar;贪心&rpar;

    题目链接:http://codeforces.com/contest/161/problem/B 题意: 有n个商品和k辆购物车,给出每个商品的价钱c和类别t(1表示凳子,2表示铅笔),如果一辆购物车 ...

  4. CF Covered Path &lpar;贪心&rpar;

    Covered Path time limit per test 1 second memory limit per test 256 megabytes input standard input o ...

  5. CF 389 E 贪心(第一次遇到这么水的E)

    http://codeforces.com/contest/389/problem/E 这道题目刚开始想的特别麻烦...但是没想到竟然是贪心 我们只需要知道偶数的时候可以对称取的,然后奇数的时候没次取 ...

  6. CF 463A &amp&semi;&amp&semi; 463B 贪心 &amp&semi;&amp&semi; 463C 霍夫曼树 &amp&semi;&amp&semi; 463D 树形dp &amp&semi;&amp&semi; 463E 线段树

    http://codeforces.com/contest/462 A:Appleman and Easy Task 要求是否全部的字符都挨着偶数个'o' #include <cstdio&gt ...

  7. cf 之lis&plus;贪心&plus;思维&plus;并查集

    https://codeforces.com/contest/1257/problem/E 题意:有三个集合集合里面的数字可以随意变换位置,不同集合的数字,如从第一个A集合取一个数字到B集合那操作数+ ...

  8. Codeforces Round &num;160 &lpar;Div&period; 2&rpar;

    A. Roma and Lucky Numbers 暴力计算. B. Roma and Changing Signs 每次取最小值改变正负,优先队列维护. C. Maxim and Discounts ...

  9. 【清真dp】cf1144G&period; Two Merged Sequences

    成就:赛后在cf使用错误的贪心通过一题 成就:在cf上赛后提交hack数据 成就:在cf上赛后hack自己 题目大意 有一长度$n \le 2\times 10^5$的序列,要求判断是否能够划分为一个 ...

随机推荐

  1. QDirModel

    #include "dialog.h" #include "ui_dialog.h" #include<QInputDialog> Dialog:: ...

  2. POJ 3597 Polygon Division &lpar;DP&rpar;

    题目链接 题意:把一个正多边形分成数个三角形或者四边形,问有多少种方案. 题解: 如果分出的全为三角形的话,那就是正多边形三角剖分问题.它的结果就是Catalan数.现在也可以划分出四边形的话,可以采 ...

  3. centos6&period;5下磁盘分区及挂载

    1..查看磁盘空间 2.磁盘分区 3.格式化分区 4.挂载/卸载

  4. MVVM模式应用 之在ViewModel中使用NavigationService

    在ViewModel.cs页面中是不能使用NavigationService,那该怎么实现跳转呢? 其实在ViewModel中实现页面的跳转也很简单,下面的代码: using Microsoft.Ph ...

  5. poj1014:母函数&plus;优化

    题目大意: 有1~6六种宝石,价格分别为1~6 ..给定每种宝石的个数,问能否平分给两个人 分析: 一看显然是个多重背包问题,也可以用母函数做 不过母函数的复杂度是n*v*k,第一次tle了.. 后来 ...

  6. 【转】高斯-克吕格投影与UTM投影异同

    高斯-克吕格(Gauss-Kruger)投影与UTM投影(Universal Transverse Mercator,通用横轴墨卡托投影)都是横轴墨卡托投影的变种,目前一些国外的软件或国外进口仪器的配 ...

  7. C&num; 跳转新的标签页

    ///这个是拿别人的,找到好多这个方法,溜了,不知道谁是原创 protected void btnPrint_Click(object sender, EventArgs e)        {    ...

  8. 一致性哈希算法----nginx负载均衡器配置之一

    一直性Hash算法在很多场景下都有应用,尤其是在分布式缓存系统中,经常用其来进行缓存的访问的负载均衡,比如:redis等<k,v>非关系数据库作为缓存系统.我们首先来看一下采用取模方式进行 ...

  9. bzoj4720 &sol; P1850 换教室(Floyd&plus;期望dp)

    P1850 换教室 先用Floyd把最短路处理一遍,接下来就是重头戏了 用 f [ i ][ j ][ 0/1 ] 表示在第 i 个时间段,发出了 j 次申请(注意不一定成功),并且在这个时间段是否( ...

  10. ScreenOper

    /// <summary> /// 屏幕操作类 /// Add by 2017-07-25 /// 1.屏幕生成Image 方法 /// 2.Image按百分比压缩 方法 /// 3.Im ...