感觉要没书读了。。
DAY48
卡码网57爬楼梯
复习一遍:背包容量在外层循环,物品在内层循环,则为排序数;
物品在外层循环,背包容量在内层循环,则为组合数。
- #include<iostream>
- #include<vector>
- using namespace std;
- int main(){
- int n,m;
- scanf("%d%d",&n,&m);
- vector<int> dp(n+1);
- dp[0]=1;
- vector<int> nums(m,1);
- int count=1;
- for(int &n:nums)n=count++;
- //先背包,再物品
- for(int i=0;i<=n;i++){
- for(int j=0;j<nums.size();j++){
- if(i>=nums[j])
- dp[i]+=dp[i-nums[j]];
- }
- }
- cout<<dp[n]<<endl;
- return 0;
- }
322零钱兑换
- class Solution {
- public:
- static bool mycmp(int &a,int &b){
- return a>b;
- }
- int coinChange(vector<int>& coins, int amount) {
- sort(coins.begin(),coins.end(),mycmp);
- if(amount==0) return 0;
- vector<int> dp(amount+1,INT_MAX);
- dp[0]=0;
- for(int i=0;i<coins.size();i++){
- for(int j=coins[i];j<=amount;j++){
- //要用本层已被覆盖的值,那么要跳过没覆盖的值 if
- if(dp[j-coins[i]]!=INT_MAX){
- dp[j]=min(dp[j],dp[j-coins[i]]+1);
- }
- }
- }
- if(dp[amount]==INT_MAX) return -1;
- return dp[amount];
- }
- };
860柠檬水找零
- class Solution {
- public:
- bool lemonadeChange(vector<int>& bills) {
- int five=0,ten=0;
- for(int i=0;i<bills.size();i++){
- if(bills[i]==5){
- five++;
- }
- else if(bills[i]==10){
- if(five<=0) return false;
- ten++;
- five--;
- }
- else{
- if(ten>0&&five>0) five--,ten--;
- else if(five>2) five-=3;
- else return false;
- }
- }
- return true;
- }
- };
518零钱兑换ii
- class Solution {
- public:
- int change(int amount, vector<int>& coins) {
- vector<int> dp(amount+1);
- dp[0]=1;
- for(int i=0;i<coins.size();i++){
- for(int j=0;j<=amount;j++){
- if(j>=coins[i])
- dp[j]+=dp[j-coins[i]];
- }
- }
- return dp[amount];
- }
- };
279完全平方数
- class Solution {
- public:
- int numSquares(int n) {
- int uper=sqrt(n);
- vector<int> dp(n+1,INT_MAX);
- vector<int>nums(uper+1,1);
- int p=1;
- for(auto &num:nums) num=p*p,p++;
- dp[0]=0;
- for(int i=0;i<nums.size();i++){
- for(int j=nums[i];j<=n;j++){
- if(dp[j-nums[i]]!=INT_MAX)
- dp[j]=min(dp[j],dp[j-nums[i]]+1);
- }
- }
- return dp[n];
- }
- };
节省空间的写法:
- class Solution {
- public:
- int numSquares(int n) {
- int uper=sqrt(n);
- vector<int> dp(n+1,INT_MAX);
- dp[0]=0;
- //遍历背包,再物品
- for(int i=0;i<=n;i++){
- for(int j=1;j*j<=i;j++){
- dp[i]=min(dp[i],dp[i-j*j]+1);
- }
- }
- return dp[n];
- }
- };
遍历顺序不一样时,需要注意dp里面的符号。