数位\(dp\)搞了一上午才搞懂。靠这种傻\(X\)的东西竟然花了我一上午的时间。

数位\(dp\)

概念

数位\(dp\)就是强制你分类一些数,例如给你一段区间,然后让你求出不包含\(2\)的数的个数。

思想

利用前缀和的思想,然后求出区间端点的前缀和这样作差就可以了。

实现方式

有两种实现方式。

记忆化搜索版

这种方法比较好理解(例题不要\(37\))

int dfs(int pos,int pre,int sta,bool limit) {
if (pos==-1) return 1;
if (!limit && dp[pos][sta]!=-1) return dp[pos][sta];
int u = limit?a[pos]:9;
int cnt = 0;
for (int i=0; i<=u; ++i) {
if (i==4 || (pre==3&&i==7)) continue;
cnt += dfs(pos-1,i,i==3,limit&&i==a[pos]);
}
if (!limit) dp[pos][sta] = cnt;
return cnt;
}

查询区间\([0 \ldots n]\)

动态规划版

(例题不要\(62\))

void get_dp()
{
dp[0][0]=1;
for (int i=1;i<10;i++)
{
for (int j=0;j<10;j++)
{
if (j==4) dp[i][j]=0;
else if (j==6)
{
for (int k=0;k<10;k++)
dp[i][j]+=dp[i-1][k];
dp[i][j]-=dp[i-1][2];
}
else
{
for (int k=0;k<10;k++)
dp[i][j]+=dp[i-1][k];
}
}
}
}

查询区间\([0\ldots n)\)

完整代码

动态规划

#include<cstdio>

const int maxn=10;
long long dp[maxn][10]; void get_dp()
{
dp[0][0]=1;
for (int i=1;i<10;i++)
{
for (int j=0;j<10;j++)
{
if (j==4) dp[i][j]=0;
else if (j==6)
{
for (int k=0;k<10;k++)
dp[i][j]+=dp[i-1][k];
dp[i][j]-=dp[i-1][2];
}
else
{
for (int k=0;k<10;k++)
dp[i][j]+=dp[i-1][k];
}
}
}
} int a[maxn];
long long solve(int n)
{
a[0]=0;
while (n)
{
a[++a[0]]=n%10;
n/=10;
}
long long ans=0;
a[a[0]+1]=0;
for (int i=a[0];i>=1;i--)
{
for (int j=0;j<a[i];j++)
if(j!=4 && !(a[i+1]==6 && j==2))
ans+=dp[i][j];
if (a[i]==4) break;
if (a[i+1]==6 && a[i]==2) break;
}
return ans;
} int main()
{
int n,m;
get_dp();
while (scanf("%d %d",&n,&m)==2 && (n||m))
{
long long k1=solve(m+1);
long long k2=solve(n);
printf("%I64d\n",k1-k2);
}
return 0;
}

关于动态规划版有个难点,就是\(solve\)函数里面的两个\(break\)。因为我们枚举的是最高位,所以当我们最高位枚举到不合法数的时候,我们就会固定,那么剩下数的任何数都是不合法的,所以\(break\)

记忆化搜索

#include<cstdio>
#include<iostream>
#include<cstring> using namespace std;
typedef long long LL; int a[20],p;
int dp[20][2]; int dfs(int pos,int pre,bool sta,bool limit) {
if (pos==-1) return 1;
if (!limit && dp[pos][sta]!=-1) return dp[pos][sta];
int u = limit?a[pos]:9;
int cnt = 0;
for (int i=0; i<=u; ++i) {
if (i==4 || (pre==3&&i==7)) continue;
cnt += dfs(pos-1,i,i==3,limit&&i==a[pos]);
}
if (!limit) dp[pos][sta] = cnt;
return cnt;
} int work(int x) {
p = 0;
while (x) {
a[p++] = x%10;
x /= 10;
}
return dfs(p-1,-1,0,true);
} int main() {
memset(dp,-1,sizeof(dp));
int l,r;
cin>>l>>r;
printf("%d",work(r)-work(l-1));
return 0;
}

以上代码借鉴。

数位$dp$的更多相关文章

  1. BZOJ 1911: [Apio2010]特别行动队 [斜率优化DP]

    1911: [Apio2010]特别行动队 Time Limit: 4 Sec  Memory Limit: 64 MBSubmit: 4142  Solved: 1964[Submit][Statu ...

  2. BZOJ 1597: [Usaco2008 Mar]土地购买 [斜率优化DP]

    1597: [Usaco2008 Mar]土地购买 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4026  Solved: 1473[Submit] ...

  3. Light OJ 1031---Easy Game(区间DP)

    题目链接 http://lightoj.com/volume_showproblem.php?problem=1031 Description You are playing a two player ...

  4. HDU 5807 Keep In Touch DP

    Keep In Touch Problem Description   There are n cities numbered with successive integers from 1 to n ...

  5. 17996 Daily Cool Run (dp)

    时间限制:1000MS  内存限制:65535K 提交次数:0 通过次数:0 题型: 编程题   语言: 不限定 Description Daily Cool Run is a popular gam ...

  6. hdu 1028 Ignatius and the Princess III 简单dp

    题目链接:hdu 1028 Ignatius and the Princess III 题意:对于给定的n,问有多少种组成方式 思路:dp[i][j],i表示要求的数,j表示组成i的最大值,最后答案是 ...

  7. BC.5200.Trees(dp)

    Trees  Accepts: 156  Submissions: 533  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 65536/6 ...

  8. hdu 5115 区间dp ***

    题意:有n只狼,每只狼有两种属性,一种攻击力一种附加值,我们没杀一只狼,那么我们受到的伤害值为这只狼的攻击值与它旁边的两只狼的附加值的和,求把所有狼都杀光受到的最小的伤害值. 枚举中间k作为最后杀死的 ...

  9. HDU 4258 Covered Walkway 斜率优化DP

    Covered Walkway Problem Description   Your university wants to build a new walkway, and they want at ...

  10. HDU 3853(期望DP)

    题意: 在一个r*c的网格中行走,在每个点分别有概率向右.向下或停止不动.每一步需要的时间为2,问从左上角走到右下角的期望时间. SOL: 非常水一个DP...(先贴个代码挖个坑 code: /*== ...

随机推荐

  1. python-Pymyslql-requests_html:把腾讯新闻的今日推荐和链接存进数据库

    使用pumsql封装的部分功能 request_html:打开www.qq.com的例子 结果: PS:记得处理数据

  2. GitHub客户端和Shell的基本操作和理解

    GitHub客户端和Shell指令的简单实用 客户端操作, web端操作, shell指令操作. 掌握了这三种操作,基本上就可以很好的运用gitHub了. 创建项目, 可以通过web端进行创建. 可以 ...

  3. 使用NFS共享硬盘

    1. 安装 sudo apt install nfs-kernel-server   2. 配置   sudo vi /etc/exports   /mnt/NewDisk *(rw,sync,no_ ...

  4. POJ 3044单调栈

    题意: 思路: 单调栈 // by SiriusRen #include <stack> #include <cstdio> using namespace std; stac ...

  5. (转载)15 个 Android 通用流行框架大全

    15 个 Android 通用流行框架大全 时间:2017-03-20 11:36来源:未知 作者:admin 点击: 2089 次 15 个 Android 通用流行框架大全   1. 缓存 Dis ...

  6. *hiho 1475 - 数组拆分,dp,由N^2降到NlogN

    题目链接 描述 小Ho得到了一个数组作为他的新年礼物,他非常喜欢这个数组! 在仔细研究了几天之后,小Ho成功的将这个数组拆成了若干段,并且每段的和都不为0! 现在小Ho希望知道,这样的拆分方法一共有多 ...

  7. Twilio介绍和使用

    1.Twilio是?需要如何才能通过Twilio打国际网络电话 http://uuxn.com/twilio-toll-free-sms介绍了通过网页来收取和发送信息 需求:通过TWILIO拨打国外座 ...

  8. STM8S汇编代码分析

    转载:http://blog.csdn.net/u010093140/article/details/50021897使用STVD建立完汇编工程项目之后(具本建立方法可以看我的另一篇博文http:// ...

  9. swift语言点评四-Closure

    总结:整个Closure的作用在于简化语言表述形式. 一.闭包的简化 Closure expression syntax has the following general form: { () -& ...

  10. Dispatch Groups

    Dispatch Groups are objects that allow several tasks to be grouped for later joining. Tasks can be a ...