找假币算法

题目的大致意思:有一枚假币混在一些真币中,已知假币比真币轻一些。总共n枚硬币中(包含一枚假币),给一个天平,请用最少的次数找出这枚假币。


在网上看到的分析,直接贴过来了。https://www.nowcoder.com/profile/897162/codeBookDetail?submissionId=12554611

先枚举一些例子,找出其中规律:
对于 1个硬币,称量 0次
对于 2个硬币,称量 1次
对于 3个硬币,称量 1次
对于 4个硬币,称量 2次,先分成(2,2,0),第一次称量前两份(2,2),如果重量不一样,再次求出判断另外2个硬币需要称量的次数。
对于 5个硬币,称量 2次,先分成(2,2,1),第一次称量前两份(2,2),如果重量不一样,再次判断另外1个硬币需要称量的次数。
对于 6个硬币,称量 2次,先分成(2,2,2),第一次称量前两份(2,2),如果重量不一样,再次判断求出另外2个硬币需要称量的次数。
对于 7个硬币,称量 2次,先分成(3,3,1),第一次称量前两份(3,3),如果重量不一样,再次判断求出另外3个硬币需要称量的次数。


通过上面分析可以看出,对于要称量的硬币,每次称量前分成3份,要求前两份的个数不小于第三份。如果前两份重量是一样,那么假币在第三份中,这样就除去了2/3的硬币。

如果前两份重量不一样,那么假币在重量轻的一份中,这样也除去了2/3的硬币。
这样以来,称量一次除去了将近2/3的硬币,一直重复上面的分法,就可以很快求出称量次数。


然后找规律总结出来的算法就是满足 pow(3,cnt)>=n 即可。其中cnt是要求的次数,n是总共硬币数。

得到如下算法:

#include<stdio.h>
#include<string.h>
#include<math.h>
int main()
{
    int n;
    while(scanf("%d",&n)!=EOF,n)
    {
        printf("%.0f\n",ceil(log(n*1.0)/log(3.0)));
    }
    return 0;
}
ceil是天花板的意思,就是向上取整。后面的就是求对数的值,相当于用了一个对数换底公式。

顺带提一下,还有个floor函数,顾名思义就是向下取整的功能。


写普通点就是下面这个算法。

#include<stdio.h>
#include<math.h>
int main() {
    int n;
    while(scanf("%d",&n)!=EOF){
        if(n==0) break;
        int cnt=0;
        while(pow(3,cnt)<n) {
            cnt++;
        }
        printf("%d\n",cnt);
    }
    return 0;
}
不过我总是怕pow函数的精度有问题,改成下面的方法感觉稳妥一些。但是下面的代码提交上去竟然是错的。。。上面用pow函数的竟然是对的。。。不知是不是网站的问题,等我再想想。

#include<stdio.h>
int main() {
    int n;
    while(scanf("%d",&n)!=EOF){
        if(n==0) break;
        int z=1,cnt=0;
        while(z<n) {
            z=z*3;
            cnt++;
        }
        printf("%d\n",cnt);
    }
    return 0;
}




  • 8
    点赞
  • 47
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
分治法求假币的非递归算法可以使用迭代的方式来实现。具体步骤如下: 1. 定义一个栈,用于存储待处理的硬币序列。 2. 将原始硬币序列压入栈中。 3. 循环处理栈中的硬币序列,直到栈为空。 4. 取出栈顶的硬币序列,判断其长度是否大于1。 5. 如果长度大于1,则将该序列拆分成三等份,分别计每份的重量。 6. 根据三份重量的大小关系,确定假币所在的范围。 7. 将假币所在范围的硬币序列压入栈中。 8. 重复步骤3-7,直到假币或者栈为空。 以下是C++代码实现: ```c++ #include <iostream> #include <stack> using namespace std; int findFakeCoin(int n, int* coins) { stack<pair<int, int>> s; s.push(make_pair(0, n - 1)); // 将整个硬币序列入栈 while (!s.empty()) { int left = s.top().first; int right = s.top().second; s.pop(); if (left == right) return left; // 只剩一个硬币,即为假币 int len = right - left + 1; int third = len / 3; int mid1 = left + third - 1; int mid2 = right - third + 1; int w1 = mid1 - left + 1; // 左侧硬币重量 int w2 = mid2 - mid1; // 中间硬币重量 int w3 = right - mid2 + 1; // 右侧硬币重量 if (w1 > w2) { // 假币在左侧 s.push(make_pair(left, mid1)); } else if (w2 > w3) { // 假币在中间 s.push(make_pair(mid1 + 1, mid2 - 1)); } else { // 假币在右侧 s.push(make_pair(mid2, right)); } } return -1; // 未假币 } int main() { int coins[] = { 1, 1, 1, 1, 1, 1, 1, 1, 0, 1 }; int n = sizeof(coins) / sizeof(int); int fakeCoin = findFakeCoin(n, coins); cout << "The fake coin is at position " << fakeCoin << endl; return 0; } ```
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值