860. 柠檬水找零

860. 柠檬水找零

在柠檬水摊上,每一杯柠檬水的售价为 5 美元。

顾客排队购买你的产品,(按账单 bills 支付的顺序)一次购买一杯。

每位顾客只买一杯柠檬水,然后向你付 5 美元、10 美元或 20 美元。你必须给每个顾客正确找零,也就是说净交易是每位顾客向你支付 5 美元。

注意,一开始你手头没有任何零钱。

如果你能给每位顾客正确找零,返回 true ,否则返回 false 。

示例 1:

输入:[5,5,5,10,20]
输出:true
解释:
前 3 位顾客那里,我们按顺序收取 3 张 5 美元的钞票。
第 4 位顾客那里,我们收取一张 10 美元的钞票,并返还 5 美元。
第 5 位顾客那里,我们找还一张 10 美元的钞票和一张 5 美元的钞票。
由于所有客户都得到了正确的找零,所以我们输出 true。
示例 2:

输入:[5,5,10]
输出:true
示例 3:

输入:[10,10]
输出:false
示例 4:

输入:[5,5,10,10,20]
输出:false
解释:
前 2 位顾客那里,我们按顺序收取 2 张 5 美元的钞票。
对于接下来的 2 位顾客,我们收取一张 10 美元的钞票,然后返还 5 美元。
对于最后一位顾客,我们无法退回 15 美元,因为我们现在只有两张 10 美元的钞票。
由于不是每位顾客都得到了正确的找零,所以答案是 false。

提示:

0 <= bills.length <= 10000
bills[i] 不是 5 就是 10 或是 20 

 1 解法1:16 ms    9.6 MB
 2 class Solution {
 3 public:
 4     bool lemonadeChange(vector<int>& bills) {
 5         //思路:
 6         //对账单顺序遍历,若是5元,直接收取,不用找零
 7         //若不是5元,那么需要对其找零,若能找开,true 否则false
 8         //还是需要记录5元的10元的,20元的面值 个数,根据个数,贪心找
 9         int sum=0;//初始化为0,没有任何钱
10         int data[3]={0};//0代表5 元,1 代表10元 ,2代表20元
11         for(int i=0;i<bills.size();i++){//遍历账单
12             if(bills[i]==5){//是5元,零钱+5
13                 sum+=5;
14                 data[0]++;
15             }
16             else {
17                 bills[i]==10?data[1]++:data[2]++;
18                 int rest=bills[i]-5;//需要找的钱
19                 int mon=0;
20                 for(int j=1;j>=0;j--){//我们只找 20-5 ,和10-5的
21                     if(rest==0) break;
22                     j==1?mon=10:mon=5;
23                     while(data[j]>0&&rest-mon>=0){//若是大于这个面值,可以用这个找钱,且有这个零钱
24                         data[j]--;//零钱个数减一个
25                         rest-=mon;//要找的钱 减去这个面值 (贪心)
26                     }
27                 }
28                 if(rest>0) return false;//没找完,返回false
29             }
30         }
31         return true;//找得开
32     }
33 };
 1 解法2: 16 ms    9.6 MB
 2 
 3 class Solution {
 4 public:
 5     bool lemonadeChange(vector<int>& bills) {
 6         //思路:
 7         //对账单顺序遍历,若是5元,直接收取,不用找零
 8         //若不是5元,那么需要对其找零,若能找开,true 否则false
 9         //还是需要记录5元的10元的,20元的面值 个数,根据个数,贪心找
10         //初始化为0,没有任何钱
11         map<int,int,greater<int> > mp;//从大到小排序map 按键
12         mp[10]=0,mp[5]=0,mp[20]=0;
13         for(int i=0;i<bills.size();i++){//遍历账单
14             if(bills[i]==5){//5元零钱个数+1
15                 mp[5]++;
16             }
17             else {
18                 mp[bills[i]]++;//10 或20元个数加1
19                 int rest=bills[i]-5;//需要找的钱(开始贪心)
20                 for(auto& iter:mp){//从大到小遍历 20 -10- 5
21                    while(iter.second>0&&rest-iter.first>=0){//若是大于这个面值,可以用这个找钱,且有这个零钱
22                         iter.second--;//零钱个数减一个
23                         rest-=iter.first;//要找的钱 减去这个面值
24                     }
25                 }
26                 if(rest>0) return false;//没找完,返回false
27             }
28         }
29         return true;//找得开
30     }
31 };
原文地址:https://www.cnblogs.com/NirobertEinteson/p/11967903.html