C++之练习题4

4.赛利有12枚银币。其中有11枚真币和1枚假币。假币看起来和真币没有区别,但是重量不同。 但赛利不知道假币比真币轻还是重。于是他向 朋友借了一架天平。朋友希望赛利称三次就能找出假币并且确定假币是轻是重。例如:如果赛利用天平称两枚硬币,发现天平平衡,说明两 枚都是真的。如果赛利用一枚真币与另一枚银 币比较,发现它比真币轻或重,说明它是假 币。经过精心安排每次的称量,赛利保证在称 三次后确定假币。

输入
输入有三行,每行表示一次称量的结果。赛利事先 将银币标号为A-L。每次称量的结果用三个以空格隔开的字符串表示:天平左边放置的硬币天平右边放置的硬币平衡状态。其中平衡状态用``up'', ``down'', 或``even''表示, 分别为右端高、右端低和平衡。天平左右的硬币数总是相等的。
输出
输出哪一个标号的银币是假币,并说明它比真币轻 还是重。

ABCD EFGH even
ABCI EFJK up
ABIJ EFGH even
K is the counterfeit coin and it is light

总体构想– 逐一试探法:
        对于每一枚硬币,先假设它是轻的,看这样是否符合称量结果。如 果符合,问题即解决。如果不符合,就假设它是重的,看是否符合称量结果,把所有硬币都试一遍,一定能找到特殊硬币。

可能情况:
x-假币的标号(12种可能),w-假币比真币轻还是重(2种可能)
猜测(x,w)有24种可能
根据假设,只有唯一的猜测与3组称量结果不矛盾
若猜测(x,w)满足如下条件,则此猜测就是所求答案
在称量结果为”even”的天平两边都没有出现x;
若w表示假币比真币轻,则在称量结果为”up”的天平右边一定出现x,在称量结果为”down”的天平左边一定出现x
若w表示假币比真币重,则在称量结果为”up”的天平左边一定出现x,在称量结果为”down”的天平右边一定出现x

逐一枚举硬币的代码
for(char c=‘A’; c<=‘L’; c++)
if( isLight(c) ){
cout << c <<
" is the counterfeit coin and it is light.\n";
break;
}
else if( isHeavy(c) ){
cout << c <<
" is the counterfeit coin and it is heavy.\n";
break;
}

bool isLight(char x)
{ // 判断硬币x是否为轻的代码
int i;
for(i=0; i<3; i++) // 判断是否与三次称量结果矛盾
switch( result[i][0] ) {
case 'u': if( ! inRight(i,x) ) return false;
break;
case 'e': if(inRight(i,x) || inLeft(i,x)) return false;
break;
case 'd': if(! inLeft(i,x)) return false;
break;
}
return true;
}

bool isHeavy(char x)
{ //判断硬币x是否为重的代码
int i;
for(i=0; i<3; i++) // 判断是否与三次称量结果矛盾
switch( result[i][0] ) {
case 'u': if( ! inLeft(i,x) ) return false;
break;
case 'e': if(inRight(i,x) || inLeft(i,x)) return false;
break;
case 'd': if(! inRight(i,x)) return false;
break;
}
return true;
}

bool inLeft(int i, char x){
// 判断硬币x 是否在第i次称量左侧
return strchr(left[i], x);
}
bool inRight(int i, char x){
// 判断硬币x 是否在第i次称量右侧
return strchr(right[i],x);
}




  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值