日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

分治应用--万里挑一 找假硬币

發布時間:2024/7/5 编程问答 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 分治应用--万里挑一 找假硬币 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

文章目錄

    • 1. 問題描述
    • 2. 解題思路
    • 3. 代碼實現

1. 問題描述

n 個硬幣中有1枚是假幣,真假幣唯一的區別是假幣重量輕,如何快速找出假幣

2. 解題思路

  • 暴力做法,一個一個的稱重,O(n)復雜度
  • 分治思路
  • 將硬幣等分成兩份,若為奇數,多出一枚,放在天平兩邊
  • 輕的一邊包含假幣,若相等,則假幣是多出的那一枚
  • 對輕的一邊繼續上述操作,直到找出假幣
    復雜度O(log n)
  • 3. 代碼實現

    /*** @description: n 個硬幣中有1枚是假幣,假幣重量輕,如何快速找出假幣* @author: michael ming* @date: 2019/7/6 20:37* @modified by: */ #include <iostream> #include <ctime> #include <random> using namespace std; int findcoin(int *weight, int left, int right, int &weightimes) {if(left+1 == right)//只有2枚硬幣{weightimes++;//稱重比較一次if(weight[left] < weight[right])return left;//返回重量小的位置elsereturn right;}int i, mid, weightsumL, weightsumR;weightsumL = weightsumR = 0;mid = left + (right-left)/2;if((right-left+1)%2 == 0)//偶數枚銀幣{weightimes++;for(i = left; i <= mid; ++i)weightsumL += weight[i];//計算左邊重量(計算機沒有天平,只能一個個加)for(i = mid+1; i <= right; ++i)weightsumR += weight[i];//右邊重量if(weightsumL > weightsumR)//左邊重,假幣在右邊return findcoin(weight,mid+1,right,weightimes);else if(weightsumL < weightsumR)//假幣在左邊return findcoin(weight,left,mid,weightimes);else//假幣不在兩邊(偶數枚銀幣);//什么都不做,不必再找了}else//奇數枚硬幣{weightimes++;for(i = left; i <= mid-1; ++i)weightsumL += weight[i];//計算左邊重量for(i = mid+1; i <= right; ++i)weightsumR += weight[i];//右邊重量if(weightsumL > weightsumR)//左邊重,假幣在右邊return findcoin(weight,mid+1,right,weightimes);else if(weightsumL < weightsumR)//假幣在左邊return findcoin(weight,left,mid-1,weightimes);else//兩邊相等(奇數枚硬幣),剩余的那個是假幣return mid;} } int main() {srand(unsigned(time(0)));int num, i, weightimes = 0;cout << "請輸入硬幣總個數:";cin >> num;const int coinNum = num;int *weight = new int [coinNum];for(i = 0; i < coinNum; ++i){weight[i] = 10;}i = rand()%num;weight[i] = 9; //隨機生成假幣for(i = 0; i < coinNum; ++i)//打印硬幣信息{cout << i + 1 << " 硬幣重量: " << weight[i] << endl;}cout << "假硬幣是第" << findcoin(weight,0,coinNum-1,weightimes)+1 << "個。" << endl;cout << "共稱了" << weightimes << "次,找到假幣。";delete[]weight;return 0; }

    輸入 2500枚、5001枚,100萬枚,最多需要 log2n 向上取整次就能找到。



    總結

    以上是生活随笔為你收集整理的分治应用--万里挑一 找假硬币的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。