• BNUOJ 52325 Increasing or Decreasing 数位dp


    题目链接:

    https://acm.bnu.edu.cn/v3/contest_show.php?cid=8506#problem/I

    I. Increasing or Decreasing

    Case Time Limit: 1000ms
    Memory Limit: 524288KB

    题意

    求[l,r]上,数位满足非递增获非递减的数的个数。

    题解

    1、dp[i][j][k]表示低i位的第i位为j的,k1时表示满足非递减,k2时满足非递增,k==0时表示每一位都相等。

    #include<map>
    #include<set>
    #include<cmath>
    #include<queue>
    #include<stack>
    #include<ctime>
    #include<vector>
    #include<cstdio>
    #include<string>
    #include<bitset>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #include<functional>
    using namespace std;
    #define X first
    #define Y second
    #define mkp make_pair
    #define lson (o<<1)
    #define rson ((o<<1)|1)
    #define mid (l+(r-l)/2)
    #define sz() size()
    #define pb(v) push_back(v)
    #define all(o) (o).begin(),(o).end()
    #define clr(a,v) memset(a,v,sizeof(a))
    #define bug(a) cout<<#a<<" = "<<a<<endl
    #define rep(i,a,b) for(int i=a;i<(b);i++)
    #define scf scanf
    #define prf printf
    
    typedef long long LL;
    typedef vector<int> VI;
    typedef pair<int,int> PII;
    typedef vector<pair<int,int> > VPII;
    
    const int INF=0x3f3f3f3f;
    const LL INFL=0x3f3f3f3f3f3f3f3fLL;
    const double eps=1e-8;
    const double PI = acos(-1.0);
    
    //start----------------------------------------------------------------------
    
    const int maxn=22;
    
    int arr[maxn],tot;
    ///type根据dp具体维数调整
    LL dp[maxn][11][4];
    ///ismax标记表示前驱是否是边界值
    ///ser标记前驱是否是前导零
    LL dfs(int len,int dig,int type,bool ismax,bool iszer) {
        if (len == 0) {
            ///递归边界,这说明前驱都合法了
            return 1LL;
        }
        if (!ismax&&dp[len][dig][type]>=0) return dp[len][dig][type];
        LL res = 0;
        int ed = ismax ? arr[len] : 9;
    
        ///这里插入递推公式
        for (int i = 0; i <= ed; i++) {
            if(i==0&&iszer){
                ///处理前导零
                res+=dfs(len-1,10,0,ismax&&i==ed,1);
            }else{
                if(type==0){
                    if(i==dig||dig==10) res+=dfs(len-1,i,0,ismax&&i==ed,0);
                    else if(i>dig) res+=dfs(len-1,i,1,ismax&&i==ed,0);
                    else if(i<dig) res+=dfs(len-1,i,2,ismax&&i==ed,0);
                }else if(type==1){
                    if(i>=dig){
                        res+=dfs(len-1,i,type,ismax&&i==ed,0);
                    }
                }else if(type==2){
                    if(i<=dig){
                        res+=dfs(len-1,i,type,ismax&&i==ed,0);
                    }
                }
            }
        }
        return ismax ? res : dp[len][dig][type] = res;
    }
    
    LL solve(LL x) {
        tot = 0;
        while (x) { arr[++tot] = x % 10; x /= 10; }
        LL ret=0;
        return dfs(tot,10,0,true,true);
    }
    
    void init() {
        clr(dp,-1);
    }
    
    int main() {
        init();
        int n; scf("%d",&n);
        while(n--){
            LL l,r;
            scf("%lld%lld",&l,&r);
            prf("%lld
    ",solve(r)-solve(l-1));
        }
        return 0;
    }
    
    //end-----------------------------------------------------------------------
    

    2、状态和上面差不多,不过是固定k,分别求出来,既ans=非递增+非递减-每个数位都相同。

    #include<map>
    #include<set>
    #include<cmath>
    #include<queue>
    #include<stack>
    #include<ctime>
    #include<vector>
    #include<cstdio>
    #include<string>
    #include<bitset>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #include<functional>
    using namespace std;
    #define X first
    #define Y second
    #define mkp make_pair
    #define lson (o<<1)
    #define rson ((o<<1)|1)
    #define mid (l+(r-l)/2)
    #define sz() size()
    #define pb(v) push_back(v)
    #define all(o) (o).begin(),(o).end()
    #define clr(a,v) memset(a,v,sizeof(a))
    #define bug(a) cout<<#a<<" = "<<a<<endl
    #define rep(i,a,b) for(int i=a;i<(b);i++)
    #define scf scanf
    #define prf printf
    
    typedef long long LL;
    typedef vector<int> VI;
    typedef pair<int,int> PII;
    typedef vector<pair<int,int> > VPII;
    
    const int INF=0x3f3f3f3f;
    const LL INFL=0x3f3f3f3f3f3f3f3fLL;
    const double eps=1e-8;
    const double PI = acos(-1.0);
    
    //start----------------------------------------------------------------------
    
    const int maxn=22;
    
    int arr[maxn],tot;
    ///type根据dp具体维数调整
    LL dp[maxn][11][4];
    ///ismax标记表示前驱是否是边界值
    ///ser标记前驱是否是前导零
    LL dfs(int len,int dig,int type,bool ismax,bool iszer) {
        if (len == 0) {
            ///递归边界,这说明前驱都合法了
            return 1LL;
        }
        if (!ismax&&dp[len][dig][type]>=0) return dp[len][dig][type];
        LL res = 0;
        int ed = ismax ? arr[len] : 9;
    
        ///这里插入递推公式
        for (int i = 0; i <= ed; i++) {
            if(i==0&&iszer){
                ///处理前导零
                res+=dfs(len-1,10,type,ismax&&i==ed,true);
            }else{
                if(type==1){
                    if(i>=dig||dig==10) res+=dfs(len-1,i,type,ismax&&i==ed,false);
                }else if(type==2){
                    if(i<=dig||dig==10) res+=dfs(len-1,i,type,ismax&&i==ed,false);
                }else if(type==0){
                    if(i==dig||dig==10) res+=dfs(len-1,i,type,ismax&&i==ed,false);
                }
            }
        }
        return ismax ? res : dp[len][dig][type] = res;
    }
    
    LL solve(LL x) {
        tot = 0;
        while (x) { arr[++tot] = x % 10; x /= 10; }
        LL ret=0;
        ret+=dfs(tot,10,1,true,true);
        ret+=dfs(tot,10,2,true,true);
        ret-=dfs(tot,10,0,true,true);
        return ret;
    }
    
    void init() {
        clr(dp,-1);
    }
    
    int main() {
        init();
        int n; scf("%d",&n);
        while(n--){
            LL l,r;
            scf("%lld%lld",&l,&r);
            prf("%lld
    ",solve(r)-solve(l-1));
        }
        return 0;
    }
    
    //end-----------------------------------------------------------------------
  • 相关阅读:
    char , varchar和Nvarchar区别
    练习2-11 计算分段函数[2] (10 分)
    练习2-10 计算分段函数[1] (10 分)
    练习2-9 整数四则运算 (10 分)
    练习2-8 计算摄氏温度 (10 分)
    练习2-6 计算物体自由下落的距离 (5 分)
    练习2-4 温度转换 (5 分)
    练习2-3 输出倒三角图案 (5 分)
    MySQL之Xtrabackup备份与恢复
    MySQL中show语法
  • 原文地址:https://www.cnblogs.com/fenice/p/5935215.html
Copyright © 2020-2023  润新知