zoukankan      html  css  js  c++  java
  • P5020 货币系统

    P5020 货币系统

    题解

    仔细分析。。。

    这道题其实就是求所给数组中有多少个数字不能被该数组中的数字自由组合表示出来

    比如样例1

     3,10 不能被该集合里的数字表示出来,所以他们组成目标集合

    6=3+3   19=10+6+3

    那么问题来了,只知道思路不会写(于是我们翻开题解)

    solution 1

    考虑到对于任意一个数字 x ,如果它能被集合里面的数字表示出来,并且表示它的数字中包含数字 a[i] ,那么 x-a[i] 一定也可以被集合里的数字表示出来

    这就涉及到了集合的并集

    can[i] 表示数字 i 能不能被表示出来(为什么用can而不是f而不是vis???因为惨)

    can[i] = can[i] ∪ can[ i-a[k] ]

    注意初始化 can[0]=1 

    代码

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cmath>
    #include<string>
    #include<cstring>
    #include<queue>
    #include<cstdlib>
    
    using namespace std;
    
    typedef long long ll;
    
    inline int read(){
        int ans=0;
        char last=' ',ch=getchar();
        while(ch<'0'||ch>'9') last=ch,ch=getchar();
        while(ch>='0'&&ch<='9') ans=ans*10+ch-'0',ch=getchar();
        if(last=='-') ans=-ans;
        return  ans;
    }
    
    const int maxn=25005;
    int T,n,ans;
    int can[maxn];
    int a[105];
    
    int main()
    {
        T=read();
        while(T--){
            memset(a,0,sizeof(a));    
            memset(can,0,sizeof(can));
            can[0]=1;
            n=read();ans=n;
            for(int i=1;i<=n;i++) a[i]=read();
            sort(a+1,a+n+1);
            for(int i=1;i<=n;i++){
                if(can[a[i]]) ans--;
                else{
                    for(int j=a[i];j<=maxn;j++)
                        can[j]=can[j]|can[j-a[i]];
                }
            }
            printf("%d
    ",ans);
        }
        
        return 0;
    }

    solution 2

    可以完全背包一下

    f[i] 表示数字 i 最多可以被几个数字表示出来

    显然上一个做法的结论成立:考虑到对于任意一个数字 x ,如果它能被集合里面的数字表示出来,并且表示它的数字中包含数字 a[i] ,那么 x-a[i] 一定也可以被集合里的数字表示出来

    f[i] = max( f[i] , f[i-a[k]]+1 )

    那么只能被一个数字表示出来的,显然就是答案啦

    注意初始化负无穷 这里用到了指针写法 f i l l 

    代码

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cmath>
    #include<string>
    #include<cstring>
    #include<queue>
    #include<cstdlib>
    
    using namespace std;
    
    typedef long long ll;
    
    inline int read(){
        int ans=0;
        char last=' ',ch=getchar();
        while(ch<'0'||ch>'9') last=ch,ch=getchar();
        while(ch>='0'&&ch<='9') ans=ans*10+ch-'0',ch=getchar();
        if(last=='-') ans=-ans;
        return  ans;
    }
    
    const int maxn=25005;
    int f[maxn],a[105];
    int T,n,ans=0;
    
    int main()
    {
        T=read();
        while(T--){
            memset(a,0,sizeof(a));
            fill(f + 1, f + maxn + 1, -2147480000);
    //        memset(f,-0x7f,sizeof(f));
            f[0]=0;
            ans=0;
            
            n=read();
            for(int i=1;i<=n;i++) a[i]=read();
            
            sort(a+1,a+n+1);
            
            for(int i=1;i<=n;i++)
              for(int j=a[i];j<=maxn;j++)
                 f[j]=max(f[j],f[j-a[i]]+1);
                 
            for(int i=1;i<=n;i++) 
                if(f[a[i]]==1) 
                    ans++;
                    
            printf("%d
    ",ans);
        }
        return 0;
    }
  • 相关阅读:
    REDUCING THE SEARCH SPACE FOR HYPERPARAMETER OPTIMIZATION USING GROUP SPARSITY阅读笔记
    缓存穿透、缓存击穿、缓存雪崩区别和解决方案
    Jenkins中构建时提示:Couldn't find any revision to build. Verify the repository and branch config
    Docker中使用Dockerfile定制化jar启动时:at sun.awt.FontConfiguration.getVersion(FontConfiguration.java:1264)
    Docker中部署mysql后SpringBoot连接时提示表不存在(修改表名忽略大小写)
    js使用y-seal实现印章功能
    手写js原生方法总结(简版)
    P5666
    CF653G
    P4649
  • 原文地址:https://www.cnblogs.com/xiaoyezi-wink/p/11827419.html
Copyright © 2011-2022 走看看