五一劳动节快乐加餐(动态规划)-----P1094 [NOIP2007 普及组] 纪念品分组

题意:

 

元旦快到了,校学生会让乐乐负责新年晚会的纪念品发放工作。为使得参加晚会的同学所获得 的纪念品价值相对均衡,他要把购来的纪念品根据价格进行分组,但每组最多只能包括两件纪念品, 并且每组纪念品的价格之和不能超过一个给定的整数。为了保证在尽量短的时间内发完所有纪念品,乐乐希望分组的数目最少。

你的任务是写一个程序,找出所有分组方案中分组数最少的一种,输出最少的分组数目。

思路:

第一眼看完应该想的都是尽可能的每组分到两个物品,那么可以先排序然后用最大的去放一个最契合它的(就是能放下的最大的),就是加上之后刚好可以等于w?可能是最优的?

然后否决了,首先你咋找一个能放下的最大的?因为每一次用完之后这物品就没了,打标机啥的不符合单调性啊,不会实现o(╥﹏╥)o

等一下,看一眼范围??价值最高才200?那标记二分个屁,直接放map里计数了,然后枚举每个分数去寻找下一个,多么完美的思路啊~

#include<bits/stdc++.h>

using namespace std;

const int maxn=3e4+100;
map<int,int >mo;
int main()
{
    int w,n,i,j;
    cin>>w;
    cin>>n;
    for(j=0;j<n;j++)
    {
        cin>>i;
        mo[i]++;
    }
    int ans=0;
    for(i=200;i>=5;)
    {
        if(mo[i]>0)
        {
            int flag=0;
            for(j=200;j>=0;j--)
            {
                if((mo[j]>0&&i+j<=w&&j!=i)||(mo[j]>=2&&j+j<=w)){
                    mo[j]--;
                    mo[i]--;
                    ans++;
                    flag=1;
//                    cout<<i<<" "<<j<<endl;
                    break;
                }
            }
            if(flag==0){
                 ans++;
                 mo[i]--;
//                 cout<<i<<endl;
            }
        }
        if(mo[i]==0) {
            i--;
        }
    }
    cout<<ans<<endl;
}

然后就寄了!!!

难道这思路有问题??实现都找了一会bug在第一个if判断里。。哦。。我是傻逼,if判断写错了第二种情况应该是i和j相等的时候所以还是i+i<=w..

看了一眼自己两年前的做法,感觉代码一眼抄啊,我以前怎么会写那样子的代码(码风奇特

居然是个双指针贪心

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值