那么后者是大于前者的

时间:2022-04-04 04:20:37

    标题问题大意:求一段区间中的windy数个数。

    注释:如果一个数任意相邻两位的差的绝对值都不小于2,这个数就是windy数,没有前导0。$区间界限<=2\cdot 10^9$。

      想法:数位dp裸题,何为数位dp?

      数位dp的意思就是我们交换一种dp的方法。通过数位进行dp。数位dp的大旨分为两点:1.对付所求答案的预措置惩罚惩罚。2.对付所求区间的界限特判。我们对付数位dp有几个显而易见但是却对照useful的性质:

        如果一个数的位数小于第二个数的位数,那么后者是大于前者的。

        如果两个数的位数相等且前者的最高位是小于后者的最高位的,那么后者是大于前者的。

        如果将两个数同时加减同一个数,他们之间的巨细关系显然是不乱的。

      通过以上几本性质,我们在枚举界限时可以先将位数小的全部枚举,然后对付高位到低位依次dp。

      回到这道题,我们先设状态dp[i][j]暗示i位数且最高位为j。在转移时,我们其实很容易想到

        $\sum\limits_{k=0}^{9}dp(i-1,k)\cdot [|j-k|\ge 2]$

      之后,关于界限的措置惩罚惩罚,看代码... ...

    最后,附上丑恶的代码... ...

#include <iostream> #include <cstdio> #include <cstring> #include <algorithm> typedef long long ll; using namespace std; int f[20][20]; void before_hand()//这个是预措置惩罚惩罚,,在上面说的很清楚了 { memset(f,0,sizeof f); for(int i=0;i<=9;i++) f[1][i]=1; for(int i=2;i<=10;i++) for(int j=0;j<=9;j++) for(int k=0;k<=9;k++) { if(abs(j-k)>=2) f[i][j]+=f[i-1][k]; } } int dig[20]; ll dispose(ll x)//对界限的措置惩罚惩罚 { int ans=0; int k=0; memset(dig,0,sizeof dig); if(!x) return 0; while(x) { dig[++k]=x%10; x/=10; } for(int i=1;i<=k-1;i++) for(int j=1;j<=9;j++)//我们用给出的第一条性质,发明 ans+=f[i][j];//ans这时必然是被所求答案笼罩的 for(int i=1;i<dig[k];i++)//第二本性质 ans+=f[k][i]; for(int i=k-1;i>=1;i--)//重复运用第二、三本性质 { for(int j=0;j<dig[i];j++) { if(abs(j-dig[i+1])>=2) ans+=f[i][j]; } if(abs(dig[i+1]-dig[i])<2) break; if(i==1) ans++; } return ans; } int main() { ll l,r; scanf("%lld%lld",&l,&r); before_hand(); printf("%lld\n",dispose(r)-dispose(l-1)); return 0; }

    小结:前面预措置惩罚惩罚的界限不要忘记特判。数位dp的第一道题,加油,JZYshuraK