POJ1013 称硬币
题目链接:
POJ 1013 http://poj.org/problem?id=1013
百炼 假币问题 http://bailian.openjudge.cn/practice/2692/
题目大意
有12枚硬币。其中有11枚真币和1枚假币。假币和真币重量不同,但不知道假币比真币轻还是重。现在,用一架天平称了这些币三次,告诉你称的结果,请你找出假币并且确定假币是轻是重(数据保证一定能找出来)。
输入
第一行是测试数据组数。
每组数据有三行,每行表示一次称量的结果。银币标号为A-L。每次称量的结果用三个以空格隔开的字符串表示:天平左边放置的硬币、天平右边放置的硬币、平衡状态。其中平衡状态用``up'', ``down'', 或 ``even''表示, 分别为右端高、右端低和平衡。天平左右的硬币数总是相等的。
输出
输出哪一个标号的银币是假币,并说明它比真币轻还是重 。
输入样例
1
ABCD EFGH even
ABCI EFJK up
ABIJ EFGH even
输出样例
K is the counterfeit coin and it is light.
解题思路
第一种代码
每一枚硬币有三种状态:较重的假币、较轻的假币、真硬币。
由于只有一枚假币,且总共只有12枚硬币,因此可以把所有硬币的所有状态都枚举一遍,对每一种假设的情况进行验证:若是某种假设符合输入的三个条件,那么该假设就是成立的。
#include<stdio.h>
int status[]={};//表示12枚硬币的真假状态:0为真,-1为假币且假币较轻,1为假币且假币较重
char left[][],right[][],result[][];
bool Balanced();//判断某种假设是否合理
int main()
{
int i,num;
scanf("%d",&num);
while(num--)
{
for(i=;i<;i++)
scanf("%s %s %s",left[i],right[i],result[i]);
for(i=;i<;i++) status[i]=;//开始的时候,假设全部硬币都是真硬币。
for(i=;i<;i++)
{
status[i]=;//假设第i枚硬币是假币且假币较重
if(Balanced())//检验该种假设是否合理
break; //若合理则已经得到答案,结束循环 status[i]=-;//假设第i枚硬币是假币且假币较轻
if(Balanced())//检验该种假设是否合理
break; //若合理则已经得到答案,结束循环 status[i]=;//上述两种假设都不合理的话,第i枚硬币为真硬币。
} printf("%c is the counterfeit coin and it is %s.\n",i+'A',status[i]==?"heavy":"light");
/**/
} return ;
} bool Balanced()//判断某种假设是否合理:合理则返回true,否则返回false.
{
int i,k,leftW,rightW;
for(i=;i<;i++)//依次检验三条规则
{
leftW=rightW=;
for(k=;left[i][k]!='\0';k++)
{
leftW+=status[left[i][k]-'A'];
rightW+=status[right[i][k]-'A'];
}
if(leftW>rightW&&result[i][]!='u') return false;
if(leftW==rightW&&result[i][]!='e') return false;
if(leftW<rightW&&result[i][]!='d') return false;
}
return true;
}
来自刘家瑛《算法基础与在线实践》
第二个代码
来源:北大郭炜老师。道理都一样,只是上面那段代码理解起来似乎比较好理解一点。
对于每一枚硬币先假设它是轻的,看这样是否符合称量结果。如果符合,问题即解决。如果不符合,就假设它是重的,看是否符合称量结果。
把所有硬币都试一遍,一定能找到特殊硬币。
#include <iostream>
#include <cstring>
using namespace std;
char Left[][]; //天平左边硬币
char Right[][]; //天平右边硬币
char result[][]; //称量结果
bool IsFake(char c,bool light);//light为真表示假设假币为轻,否则表示假设假币为重
int main()
{
int t;
cin >> t;
while(t--)
{
for(int i = ;i < ; ++i) cin >> Left[i] >> Right[i] >> result[i];
for(char c='A'; c<='L';c++)
{
if( IsFake(c,true) )//假设c这个硬币为假硬币而且它比真硬币轻
{
cout << c << " is the counterfeit coin and it is light.\n";
break;
}
else if( IsFake(c,false) )//假设c这个硬币为假硬币而且它比真硬币重
{
cout << c << " is the counterfeit coin and it is heavy.\n";
break;
}
}
}
return ;
}
bool IsFake(char c,bool light)//light 为真表示假设假币为轻,否则表示假设假币为重
{
for(int i = ;i < ; ++i)
{
char * pLeft,*pRight; //指向天平两边的字符串
if(light)
{
pLeft = Left[i];
pRight = Right[i];
}
else
{
pLeft = Right[i];
pRight = Left[i];
} switch(result[i][])
{
case 'u':
if ( strchr(pRight,c) == NULL) return false;
break;
case 'e':
if( strchr(pLeft,c) || strchr(pRight,c)) return false;
break;
case 'd':
if ( strchr(pLeft,c) == NULL) return false;
break;
}
}
return true;
}
POJ1013 称硬币的更多相关文章
- POJ1013称硬币【枚举】
Counterfeit Dollar Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 52474 Accepted: 16 ...
- 枚举-称硬币POJ1013
#include <iostream> #include<string.h> using namespace std; char Lleft[][]; char Lright[ ...
- 算法题----称硬币: 2n(并不要求n是2的幂次方)个硬币,有两个硬币重量为m+1, m-1, 其余都是m 分治 O(lgn)找出假币
Description: 有2n个硬币和一个天平,其中有一个质量是m+1, 另一个硬币质量为m-1, 其余的硬币质量都是m. 要求:O(lgn)时间找出两枚假币 注意: n不一定是2的幂次方 算法1: ...
- C++基础算法学习——猜假币
有12枚硬币.其中有11枚真币和1枚假币.假币和真币重量不同,但不知道假币比真币轻还是重.现在,用一架天平称了这些币三次,告诉你称的结果,请你找出假币并且确定假币是轻是重(数据保证一定能找出来).例题 ...
- ACM训练计划建议(写给本校acmer,欢迎围观和指正)
ACM训练计划建议 From:freecode# Date:2015/5/20 前言: 老师要我们整理一份训练计划给下一届的学弟学妹们,整理出来了,费了不少笔墨,就也将它放到博客园上供大家参考. 菜 ...
- 【POJ水题完成表】
题目 完成情况 poj1000:A+B problem 完成 poj1002:电话上按键对应着数字.现在给n个电话,求排序.相同的归一类 完成 poj1003:求最小的n让1+1/2+1/3+...+ ...
- ACM训练计划建议(转)
ACM训练计划建议 From:freecode# Date:2015/5/20 前言: 老师要我们整理一份训练计划给下一届的学弟学妹们,整理出来了,费了不少笔墨,就也将它放到博客园上供大家参考. 菜 ...
- (BruteForce)暴力破解经典题目总结
在算法竞赛中,很多问题是来不及用数学公式推导出来的.或者说根本就找不到数学规律,这时我们就需要使用枚举来暴力破解. 不过枚举也是需要脑子的,一味的暴力只能超时.因此我这里选择了几道mooc上经典的题目 ...
- codes often WA
枚举: 1.完美立方 #include<iostream> #include <cstdio> using namespace std; int main() { int N; ...
随机推荐
- verilog语法实例学习(10)
常用的时序电路介绍 T触发器和JK触发器 在D触发器输入端添加一些简单的逻辑电路,可以生成另一种类型的存储元件.比如下图所示的T触发器.该电路有一个上升沿触发的触发器和两个与门,一个或门,以及一个反相 ...
- CSS-返回顶部代码
现在的网站基本上都是长页面,多的有四五屏,少的话也有两三屏,页面太长有的时候为了提升用户体验,会在页面右边出现一个回到顶部的按钮,这样能快速回到顶部,以免在滑动页面出现视觉屏幕,回到顶部一般有四种方式 ...
- 第一个手写Win32窗口程序
第一个手写Win32窗口程序 一 Windows编程基础 1 Win32应用程序的基本类型 1.1 控制台程序 不需要完善的Windows窗口,可以使用DOS窗口 的方式显示. 1.2 Win32窗口 ...
- light oj 1007 Mathematically Hard (欧拉函数)
题目地址:light oj 1007 第一发欧拉函数. 欧拉函数重要性质: 设a为N的质因数.若(N % a == 0 && (N / a) % a == 0) 则有E(N)=E(N ...
- Kafka:ZK+Kafka+Spark Streaming集群环境搭建(十)安装hadoop2.9.0搭建HA
如何搭建配置centos虚拟机请参考<Kafka:ZK+Kafka+Spark Streaming集群环境搭建(一)VMW安装四台CentOS,并实现本机与它们能交互,虚拟机内部实现可以上网.& ...
- Java-Shiro(四):Shiro
https://blog.csdn.net/visket2008/article/details/78539334 不错的视屏教程,很实用:https://www.bilibili.com/video ...
- PHP 使用PHPExcel删除Excel单元格指定列
需求是这样的: 有一个系统仅公司内部和外部经销商使用,在一个导出功能中公司内部员工跟外部经销商导出的列是不一样的(某些数据是不能提供给经销商的) 因为导出的数据都是一样的(某些列外数据外部没有)因此并 ...
- Docker-machine创建虚机时停在虚机启动的提示上,并且创建的虚机显示Ip Not found
Docker-machine创建虚机时停在虚机启动的提示上,并且创建的虚机用docker-machine ls 列出来的时候显示Ip Not found, 是什么原因那? [答案] 看这个帖子: ht ...
- [Algorithm] Reverse a linked list
It helps to understands how recursive calls works. function Node(val) { return { val, next: null }; ...
- [Functional Programming] Arrow Functor with contramap
What is Arrow Functor? Arrow is a Profunctor that lifts a function of type a -> b and allows for ...