动态规划——数位dp
通过先前在《动态规划——背包问题》中关于动态规划的初探,我们其实可以看到,动态规划其实不是像凸包、扩展欧几里得等是具体的算法,而是一种在解决问题中决策的思想。在不同的题目中,我们都需要根据题设恰到好处的把整个过程分割成小的状态,然后找到对应的状态转移方程,尽管都是这个过程,但是有时候条件稍微一遍,我们分析状态并找状态转移方程的思路都会发生改变,因此动态规划的题目呈现出很大的灵活性。
除了典型那背包问题涉及动态规划,还有很多其他的模型——概率dp、数位dp、区间dp、插头dp,这些都是在不同的情景中利用动态规划的思想来解决的具体模型,在接下来的文章中,笔者将一一介绍。
直接通过题目来学习数位dp。(Problem source : Light OJ 1122)
Description
Given a set of digits S, and an integer n, you have to find how many n-digit integers are there, which contain digits that belong to S and the difference between any two adjacent digits is not more than two.
Input
Input starts with an integer T (≤ 300), denoting the number of test cases.
Each case contains two integers, m (1 ≤ m < 10) and n (1 ≤ n ≤ 10). The next line will contain m integers (from 1 to 9) separated by spaces. These integers form the set S as described above. These integers will be distinct and given in ascending order.
Output
For each case, print the case number and the number of valid n-digit integers in a single line.
题目大意:给出一个数组S,和一个整数n,需要你计算用数组S中的数组成一个长度为n的整数,要求这个整数相邻两位的差要小于2。
数理分析:所谓好的开始是成功过的一半,动态规划问题的解决往往也是起始于好的dp数组的定义,这个dp数组用来记录各个状态的值。类似含有n种物体的背包问题,在这里我们需要排n位数字,显然从第1位开始,一直构造到第n位。可以说,这是dp数组的一个维度了,可以表征一种状态。我们在想,在这种状态下,整个问题还可以分解成哪些状态,我们从第i位(从后往前构造,即:最高位)放的数入手,它可以是序列S中的任意一个数字,这又是dp数组的一个维度了。
因此我们可以设置数组dp[i][j],表示i位整数,且最高位上的整数是j的个数。那么我们容易看到,状态转移方程为:
dp[i][j] = dp[i-1][j-2] + dp[i-1][j-1] + dp[i-1][j] + dp[i-1][j+1] + dp[i-1][j+2]。
基于状态转移方程的给出,我们就可以很容易变成实现了。
参考代码如下。
#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std;
int m,n,ans;
bool a[];
int dp[][];
void init()
{
memset(dp,,sizeof(dp));
for(int i=;i<=;i++)
if(a[i]) dp[][i]=;
}
void solve()
{
for(int i=;i<=n;i++)
for(int j=;j<=;j++)
if(a[j]){
for(int k=j-;k<=j+;k++)
if(k>=&&k<=)
dp[i][j]+=dp[i-][k];
}
ans=;
for(int i=;i<=;i++)
ans+=dp[n][i];
}
int main()
{
int T,t,i,x,j;
int temp;
cin>>T;
for(j=;j<=T;j++)
{
cin>>m>>n;
memset(a,false,sizeof(a));
for(i=;i<=m;i++)
{
cin>>temp;
a[temp]=;
}
init();
solve();
printf("Case %d: %d\n",j,ans);
}
return ;
}
我们早来看一道有关数位dp的问题。(Problem source : hdu 2089)
题意似乎简单暴力,然是如果真得用暴力来解决问题的话会非常费时的,因此我们这里讨论如何利用动态规划的思想来求解。
从题目的限制条件我们不难看出,我们需要从位的角度来看十进制整数。动态规划除了能够动态保存最优解,另一大作用便是将全局性的问题给子问题化。
我们依然从整个过程的中间开始,对于一个i位整数a,在区间[0,a]上满足题设限制的整数的个数是f(a),那么对于一个i-1位的整数b,在区间[0,b]上满足题设限制的整数的个数是f(b),那么f(a)和f(b)是否有着什么联系呢?如果有的话,我们就找到了递推关系,也就是状态转移方程,便能够将全局问题给子问题化了。
那么下面我们来看看f(a)和f(b)有着怎样的递推关系,这其实有些类似组合数学方面的问题,如果学习过错排的读者可能会感触更深。我们假设i位的整数的最高位是k,那么显然对于f()函数就有了第二维的限制,那么我们在这里设置dp[i][j]来表示0到最高位是j的i位整数满足条件的整数个数。那么结合简单的组合数学的思想,我们可以看到如下的递推关系。
dp[i][j] = ∑dp[i-1][k] , 其中k∈[1,9]。
基于dp[][]的得到,我们要求0到某个具体的数字x(注意dp数组表征的并不是某个具体数字)x范围内满足限制的整数个数,只需从该数字的最高位开始按位依次往下记录数据即可。有读者可能会问,为何不从最低位开始呢?我们看到,从高位往低位记录,我们控制当前位的数字小于x对应位的数字,这样保证我们构造的数处在[0,x)内,而如果从低位往高位记录,那记录的末了就非常难以控制当前构造的数字是否在[0,x)的范围内。而正是基于这个特点,我们通过这种方法得到是[0,x)范围内的解,而显然题设想让我们找到[l,r]范围内的解,在这里我们容易想到,可以通过一个中间量来嫁接一下,[l,r]上的解其实就是[0,r]上的解减掉[0 , l - 1]上的解。如果设Fun作为求解的函数,基于Fun(0,x)函数其实求的是[0,x)上的解,记录F[l,r]表示[l,r]上的解,那么则有F[l,r] = Fun(0,r+1) - Fun(0,l)。
其实概括地来看上面的分析过程不难发现,对于数位dp的求解,相对于背包问题,这种模型并没用用动态规划来求解什么最优解,而是利用这种思想通过预处理来将问题子化,并保存子问题的解,然后对于输入不同的值,再来通过子问题间不同的组合来解决问题。
参考代码如下。
#include <iostream>
#include <string>
#include <string.h>
#include <algorithm>
using namespace std;
int dp[][];
void init()
{
memset(dp,,sizeof(dp));
dp[][] = ;
for(int i=;i<=;i++)
{
for(int j=;j<;j++)//枚举第i位可能出现的数
{
for(int k=;k<;k++)//枚举第i-1位可能出现的数
{
if(j!=&&!(j==&&k==))
dp[i][j] += dp[i-][k];
}
}
}
}
int solve(int n)
{
init();
int digit[];
int len = ;
while(n>)
{
digit[++len] = n%;
n/=;
}
digit[len+]=; //为下面判断最大数位是否含62做个预处理
int ans = ;
for(int i=len;i;i--)
{
for(int j=;j<digit[i];j++)//这里一定要是小于而不是小于等于,至于理由读者可以简单的思考一下
{
if(j!=&&!(digit[i+]==&&j==))
ans+=dp[i][j];
}
if(digit[i]==||(digit[i]==&&digit[i+]==))
break;
}
return ans;
}
int main()
{
int l,r;
while(cin>>l>>r)
{
if(l== && r == )
break;
else
cout<<solve(r+) - solve(l)<<endl;
}
return ; }
动态规划——数位dp的更多相关文章
- 模板 - 动态规划 - 数位dp
#include<bits/stdc++.h> using namespace std; #define ll long long ]; ll dp[][/*可能需要的状态2*/];//不 ...
- 动态规划-数位dp
大佬讲的清楚 [https://blog.csdn.net/wust_zzwh/article/details/52100392] 例子 不要62或4 l到r有多少个数不含62或者4 代码 #incl ...
- 有关动态规划(主要是数位DP)的一点讨论
动态规划(dynamic programming)是运筹学的一个分支,是求解决策过程(decision process)最优化的数学方法.20世纪50年代初美国数学家在研究多阶段决策过程的优化问题时, ...
- 动态规划晋级——HDU 3555 Bomb【数位DP详解】
转载请注明出处:http://blog.csdn.net/a1dark 分析:初学数位DP完全搞不懂.很多时候都是自己花大量时间去找规律.记得上次网络赛有道数位DP.硬是找规律给A了.那时候完全不知数 ...
- 动态规划:数位DP
数位dp一般应用于: 求出在给定区间[A,B]内,符合条件P(i)的数i的个数 条件P(i)一般与数的大小无关,而与 数的组成 有关 例题是一道BZOJ1833,让求出区间所有整数每个数字出现的次数 ...
- 动态规划专题(三)——数位DP
前言 数位\(DP\) 真的是最恶心的\(DP\). 简介 看到那种给你两个数,让你求这两个数之间符合条件的数的个数,且这两个数非常大,这样的题目一般就是 数位\(DP\) 题. 数位\(DP\)一般 ...
- 「动态规划」-数位dp专题
数位dp,今天学长讲的稍玄学,课下花了一会时间仔细看了一下,发现板子是挺好理解的,就在这里写一些: 数位dp主要就是搞一些在区间中,区间内的数满足题目中的条件的数的个数的一类题,题目一般都好理解,这时 ...
- P4317 花神的数论题 动态规划?数位DP
思路:数位$DP$ 提交:5次(其实之前A过,但是调了调当初的程序.本次是2次AC的) 题解: 我们分别求出$sum(x)=i$,对于一个$i$,有几个$x$,然后我们就可以快速幂解决. 至于求个数用 ...
- 数位dp/记忆化搜索
一.引例 #1033 : 交错和 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 给定一个数 x,设它十进制展从高位到低位上的数位依次是 a0, a1, ..., an ...
随机推荐
- 如何在安卓/data(而不是/data/data)目录下进行文件的读写操作
分析:Android默认是无法直接操作/data目录的,只能读写程序自己的私有目录,也就是/data/data/package name/下,默认只能操作这个目录下的文件,也就是我们想直接读写/dat ...
- ASP.NET 3.5路由总结篇
URL Routing是非常重要的一块技术体系,笔者将URL Routing的知识进行梳理后得出本文,旨在同大家分享,希望能够起到抛砖引玉的作用. 1. 什么是URL Routing? 所谓UR ...
- tomcat 正常启动,无法访问。且项目启动无问题。。。的解决办法。。
Eclipes解决方法: 1.右击项目,选择propreties选项 2.在弹出的首选项窗口的左侧选择“Web Project Settings” 3.修改context root:输入框,修改成自己 ...
- while read line无法循环read文件
while read line 与for循环的区别 例子:要从一个ip列表中获取ip.port,然后ssh ip 到目标机器进行特定的command操作ssh -o StrictHostKeyChec ...
- JavaScript 计时事件
JavaScript 计时事件 通过使用 JavaScript,我们有能力作到在一个设定的时间间隔之后来执行代码,而不是在函数被调用后立即执行.我们称之为计时事件. 在 JavaScritp 中使用计 ...
- jQuery慢慢啃之属性(三)
1.attr(name|properties|key,value|fn)设置或返回被选元素的属性值. $("img").attr("src");//获取属性 $ ...
- js 的其它运算符和优先级
三元运算符: 语法为 exp1? exp2:exp3 判断 exp1是true 和 false 如果true,则返回exp2 ,如果false ,则返回exp3 <script> if ...
- truncate 命令删除恢复
truncate命令可以一次性删除当前表中所有记录并且不留任何日志,同时这个表的ID就自动初化从1开始,今天我就来给大家尝试一个利用truncate清除记录之后恢复过程. 实际线上的场景比较复杂,当时 ...
- 【技术宅11】php入门运算
//1.空bool $a=''; $b=NULL; $c=false; $d=0; $e='0'; $f=array(); $g=array(array()); $h='NULL'; var_dump ...
- bootstrap日期时间插件datetimepicker
<!DOCTYPE HTML> 02 <html> 03 <head> 04 <link href="http://netdna.boo ...