传送门

2144 砝码称重 2

 时间限制: 1 s
 空间限制: 16000 KB
 题目等级 : 钻石 Diamond
 
题目描述 Description

有n个砝码,现在要称一个质量为m的物体,请问最少需要挑出几个砝码来称?

注意一个砝码最多只能挑一次

输入描述 Input Description

第一行两个整数n和m,接下来n行每行一个整数表示每个砝码的重量。

输出描述 Output Description

输出选择的砝码的总数k,你的程序必须使得k尽量的小。

样例输入 Sample Input

3 10
5
9
1

样例输出 Sample Output

数据范围及提示 Data Size & Hint

1<=n<=30,1<=m<=2^31,1<=每个砝码的质量<=2^30

【思路】
正解:搜索
神正解:双向搜索
吐槽:你都知道质量是m了你称个鬼啊╭(╯^╰)╮
【code】
我的智障搜索
  1. #include<iostream>
  2. #include<cstdio>
  3. using namespace std;
  4. int w[],use[];
  5. long long ans,m;
  6. int n,res=;
  7. void dfs(int x)
  8. {
  9. if(ans>=res){
  10. return;
  11. }
  12. if(!m)
  13. {
  14. if(ans<res)
  15. res=ans;
  16. return ;
  17. }
  18. for(int i=;i<=n;i++)
  19. {
  20. if(!use[i]&&w[i]<=m)
  21. {
  22. use[i]=;
  23. m-=w[i];
  24. ans++;
  25. dfs(x+);
  26. m+=w[i];
  27. use[i]=;
  28. ans--;
  29. }
  30. }
  31. return;
  32. }
  33. int main()
  34. {
  35. scanf("%d%lld",&n,&m);
  36. for(int i=;i<=n;i++)
  37. scanf("%d",&w[i]);
  38. dfs();
  39. printf("%d\n",res);
  40. return ;
  41. }

比较尴尬.....

用后缀和优化 剪枝

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<algorithm>
  4. using namespace std;
  5. long long w[],last[];
  6. int n,m;
  7. int ans;
  8. inline int read()
  9. {
  10. int f=,x=;char ch=getchar();
  11. while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
  12. while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
  13. return f*x;
  14. }
  15. inline bool cmp(int a,int b)
  16. {
  17. return a>b;
  18. }
  19. inline void dfs(int now,int use,long long ww)
  20. {
  21. if(use>=ans)return;
  22. if(ww==m){ans=min(ans,use);}
  23. for(int i=now+;i<=n;i++)
  24. {
  25. if(ww+last[i]<m)return;
  26. if(ww+w[i]>m)continue;
  27. dfs(i,use+,ww+w[i]);
  28. }
  29. return;
  30. }
  31. int main()
  32. {
  33. n=read();m=read();
  34. for(int i=;i<=n;i++)
  35. w[i]=read();
  36. sort(w+,w+n+,cmp);
  37. for(int i=n;i>=;i--)
  38. last[i]=last[i+]+w[i];
  39. ans=n;
  40. dfs(,,);
  41. printf("%d\n",ans);
  42. return ;
  43. }

发现加上inline也没多大用。(前后对比)

比较神的双向搜索,其实我做这个题就是为了学这个方法。

就是把原来的数分成两部分进行dfs。

并用map<int,int>这些质量需要多少砝码。

mm-sum能在第一次dfs中能找到。(sum表示第二次dfs中目前总质量)

js+m[mm-sum];来更新最优解。

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<algorithm>
  4. #include<map>
  5. using namespace std;
  6. int ans(),w[],n;
  7. long long mm;
  8. map<int, int>m;
  9. void dfs(int js,int last,int sum,bool k)
  10. {
  11. int r=n;
  12. if(k){m[sum]=js;r/=;}
  13. else
  14. {
  15. if(m.find(mm - sum)!=m.end())
  16. ans=min(ans,js+m[mm-sum]);
  17. }
  18. for(int i=last;i<r;i++)
  19. dfs(js+,i+,sum+w[i],k);
  20. }
  21. int main()
  22. {
  23. scanf("%d%lld",&n,&mm);
  24. for(int i=;i<n;i++)
  25. scanf("%d",&w[i]);
  26. dfs(,,,true);
  27. dfs(,n/,,false);
  28. printf("%d\n",ans);
  29. return ;
  30. }

codevs 2144 砝码称重2的更多相关文章

  1. Codevs 2144 砝码称重 2

    2144 砝码称重 2  时间限制: 1 s  空间限制: 16000 KB  题目等级 : 钻石 Diamond 题解       题目描述 Description 有n个砝码,现在要称一个质量为m ...

  2. Codevs No.2144 砝码称重2

    2016-05-31 22:01:16 题目链接: 砝码称重2 (Codevs No.2144) 题目大意: 给定N个砝码,求称出M的重量所需砝码最小个数 解法: 贪心 使砝码数量最小,当然是每个砝码 ...

  3. NOI题库--砝码称重V2(多重背包2^n拆分)

    以前只会写多重背包的原版,渣的不行,为了做此题不得不学习了一下,发现其实也不难,只要理解了方法就好多了(PS:其实和倍增挺像的) 8756:砝码称重V2 总时间限制: 1000ms 内存限制: 655 ...

  4. 安徽省2016“京胜杯”程序设计大赛_A_砝码称重

    砝码称重 Time Limit: 1000 MS Memory Limit: 65536 KB Total Submissions: 61 Accepted: 37 Description 小明非常喜 ...

  5. P2347 砝码称重-DP方案数-bitset

    P2347 砝码称重 DP做法 : 转化为 01背包. 进行方案数 更新.最后统计种类. #include<bits/stdc++.h> using namespace std; #def ...

  6. 51nod 1449 砝码称重 (进制思想)

    1449 砝码称重 题目来源: CodeForces 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题  收藏  关注 现在有好多种砝码,他们的重量是 w0,w1,w ...

  7. 51nod 1837 砝码称重【数学,规律】

    题目链接:51nod 1837 砝码称重 小 Q 有 n 个砝码,它们的质量分别为 1 克. 2 克.……. n 克. 他给 i 克的砝码标上了编号 i (i = 1, 2, ..., n),但是编号 ...

  8. P2347 砝码称重

    P2347 砝码称重 题目描述 设有1g.2g.3g.5g.10g.20g的砝码各若干枚(其总重<=1000), 输入输出格式 输入格式: 输入方式:a1 a2 a3 a4 a5 a6 (表示1 ...

  9. 洛谷P1441 砝码称重

    P1441 砝码称重 题目描述 现有n个砝码,重量分别为a1,a2,a3,……,an,在去掉m个砝码后,问最多能称量出多少不同的重量(不包括0). 输入输出格式 输入格式: 输入文件weight.in ...

随机推荐

  1. rootkit基础

    应用程序总是离不开系统内核所提供的服务,比如它要使用内存的时候,只要跟操作系统申请就行了,而不用自己操心哪里有空闲的内存空间等问题,实际上,这些问题是由操作系统的内核来代劳的.站在黑客的角度讲,如果能 ...

  2. PHP实现上次登录功能

    通过一个sql语句把上次的登录时间给本次登录时间,再把当前时间记录下来 update userinfo  set lasttime=userinfo.logintime,logintime= CURR ...

  3. rtems 4.11 启动流程(arm, beagle)

    请参照官方的 bsp_howto 文档,对arm来说,首先执行的文件是start.S start.S c/src/lib/libbsp/arm/shared/start/start.S 1.从 _st ...

  4. 本地filezilla&amp;servervsftp搭配使用

    环境:本地ubuntu系统&serverubuntu系统 本地安装filezilla  apt-get install filezilla '安装filezilla filezilla '执行 ...

  5. 多文档自己主动文摘:Multi-Document Summarization,MDS

  6. c++ 通过数据流方式实现文件拷贝

    #include "stdafx.h"#include <iostream>#include<fstream>using namespace std;voi ...

  7. python 基础 7.8 json--下

      一. 文件和json 之间的转换 1. json.dump()   #/usr/bin/python #coding=utf-8 #@Time   :2017/11/13 0:12 #@Authe ...

  8. 内存MCE错误导致暴力扩充messages日志 以及chattr记录

    由于放假,好久没登过服务器,今天登上服务器查看日志意外发现:/var/log/messages文件竟然被撑到20多个G!!!赶紧查看是什么情况,首先,20多个G的文件根本无法查看,因此,我想到了spl ...

  9. Elasticsearch集群UNASSIGNED

    Elasticsearch集群UNASSIGNED http://shineforever.blog.51cto.com/1429204/1859734 http://www.searchtech.p ...

  10. Hadoop基础学习(一)分析、编写并执行WordCount词频统计程序

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/jiq408694711/article/details/34181439 前面已经在我的Ubuntu ...