zoukankan      html  css  js  c++  java
  • 寒假集训日志(六)——DP

    (一)数位DP

    1057. Amount of Degrees

    Time limit: 1.0 second
    Memory limit: 64 MB
    Create a code to determine the amount of integers, lying in the set [X;Y] and being a sum of exactly K different integer degrees of B.
    Example. Let X=15, Y=20, K=2, B=2. By this example 3 numbers are the sum of exactly two integer degrees of number 2:
    17 = 24+20,
    18 = 24+21,
    20 = 24+22.

    Input

    The first line of input contains integers X and Y, separated with a space (1 ≤ X ≤ Y ≤ 231−1). The next two lines contain integers K and B (1 ≤ K ≤ 20; 2 ≤ B ≤ 10).

    Output

    Output should contain a single integer — the amount of integers, lying between X and Y, being a sum of exactly K different integer degrees of B.

    Sample

    inputoutput
    15 20
    2
    2
    
    3
    

    一道据说比较经典的题目,基本弄懂,ans++ 那里实在没搞懂,不过也没那么多时间给我犹豫。。

    主要借助一个状态转移方程 f[i][j] = f[i-1][j-1] + f[i-1][j]

    f[i][j] 表示二进制中前i位中有j个1的个数,其他进制的数在转换过来之后都可一借用f[i][j]来解决。可以借助这个图理解一下这个方程。

    http://img2.shangxueba.com/img/uploadfile/20150814/16/1EFCEE80E20B6A790478581BDE062868.png不过ans++那里实在每看懂。

    #include<iostream>
    #include<cstdio>
    #include<cmath>
    #include<algorithm>
    #include<cstring>
    #include<iomanip>
    #include<fstream>
    #include<vector>
    typedef long long LL;
    using namespace std;
    
    const int maxN = 50;
    int f[maxN][maxN];
    
    void getMap (){
        f[0][ 0] = 1;
        for( int i = 1 ; i< maxN ;++i){
            f[i][0] = 1;
            for( int j=  1 ; j<=i ;++j)
                f[i][j] = f[i-1][j] + f[i-1][j-1] ;
        }
    }
    
    int solve ( int x, int k ,  int c) {
        vector< int > arr;
        while( x) {
            arr.push_back ( x% c);
            x /= c;
        }
        int ans = 0, cnt = 0  ;
        for( int i = arr.size() -1 ; i>= 0 ; --i){
            if( arr[i] == 1){
                ans += f[i][ k - cnt];
                cnt++ ;
                if( cnt == k )  break;
            }
            else if( arr[i] > 1){
                ans += f[i+1][k - cnt];
                break;
            }
        }
        if( cnt == k )  ans ++;
        return ans ;
    }
    
    int main()
    {
        for( int i = 0 ; i< maxN ; ++i)
            for( int j = 0 ; j< maxN; ++j)
                    f[i][j] =  0 ;
        int x, y ;
        int k, b;
        getMap();
        cin>>x>>y>>k>>b;
        cout<<solve(y,k,b) - solve(x-1 , k , b)<<endl;
    return 0;
    }
    A - 数位DP
    Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u
    Submit Status

    Description

    The counter-terrorists found a time bomb in the dust. But this time the terrorists improve on the time bomb. The number sequence of the time bomb counts from 1 to N. If the current number sequence includes the sub-sequence "49", the power of the blast would add one point.
    Now the counter-terrorist knows the number N. They want to know the final points of the power. Can you help them?
     

    Input

    The first line of input consists of an integer T (1 <= T <= 10000), indicating the number of test cases. For each test case, there will be an integer N (1 <= N <= 2^63-1) as the description.

    The input terminates by end of file marker.
     

    Output

    For each test case, output an integer indicating the final points of the power.
     

    Sample Input

    3 1 50 500
     

    Sample Output

    0 1 15

    Hint

    From 1 to 500, the numbers that include the sub-sequence "49" are "49","149","249","349",
    "449","490","491","492","493","494","495","496","497","498","499", so the answer is 15.
    //也是一道很经典的题目,但是有个坑一定要注意一下,就是以防止出现49结尾的情况
    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<queue>
    #include<vector>
    #include<cstring>
    
    using namespace std;
    typedef long long LL;
    
    const int maxN = 22;
    LL dp[maxN][3];
    
    void init(){
        for( int i = 0 ; i<=maxN ;++i)
            for( int j = 0 ;j<= maxN; ++j)
                dp[i][j] = 0;
        dp[0][0] = 1;
        for(int i = 1;i<=22;i++)
        {
            dp[i][0] = dp[i-1][0]*10-dp[i-1][1];
            dp[i][1] = dp[i-1][0];
            dp[i][2] = dp[i-1][2]*10+dp[i-1][1];
        }
    }
    
    LL solve( LL x){
        int arr[maxN] ;
        memset( arr, 0 , sizeof(arr));
        int len = 0;
        while( x){
            arr[ ++ len]  = x%10;
            x /= 10;
        }
        arr[ len + 1] = 0;
        LL ans = 0, flag = 0;
        for( int i = len ; i>=1 ; --i){
            if( flag) ans += (dp[i-1][0] + dp[i-1][2])* arr[i];
            else if(arr[i] >=5)
                ans += dp[i-1][1] + dp[i-1][2]* arr[i];
            else ans += dp[i-1][2]*arr[i];
            if( arr[i+1] == 4 && arr[i] == 9 )  flag = 1;
        }
        return ans ;
    }
    
    int main(){
        int T ;
        cin>>T;
        init();
        while(T--){
            LL n;
            scanf("%lld", &n);
            LL ans ;
            ans = solve(n+1);//因为flag有延迟,所以需要加1
            printf("%lld
    ", ans);
        }
        return 0;
    }

    (二)树形DP

  • 相关阅读:
    Javascript + Dom知识点总结
    C#编码规范
    windows7下安装apache+PHP5.3
    Sublime Text 使用方法
    如何使用 Azure PowerShell 在 Azure Marketplace 中查找 Windows VM 映像
    Azure 实例元数据服务
    Azure 元数据服务:适用于 Windows VM 的计划事件(预览)
    处理 Windows 虚拟机的计划内维护通知
    Azure 虚拟机代理概述
    下载 VM 模板
  • 原文地址:https://www.cnblogs.com/topW2W/p/5163815.html
Copyright © 2011-2022 走看看