最近刷题的同时还得填填坑,说来你们也不信,我还不会数位dp。




照例推几篇博客:

数位DP讲解

数位dp 的简单入门

这两篇博客讲的都很好,不过代码推荐记搜的形式,不仅易于理解,还短。




数位dp的式子一般是这样的:dp[i][][]表示到第\(i\)位,而后面几维就因题而异了。

不过通用的思想就是利用前缀相减求出区间信息。

算了上题吧。




[SCOI2009]windy数

这都说是数位dp入门题。

根据这题,受到影响的数只有相邻两个,因此dp[i][j]表示到第\(i\)位(从高往低)上一位的数\(j\)的答案。

接下来的关键在于怎么判断到达上界的情况。那么我们在搜索的时候加一个bool变量_Max表示是否到达上界,如果是,那么这一位的最大值就是该位上的数,否则就是9。

然后关于前导零,我又开了一个bool变量zero判断他之前有没有过非0的数。

接下来就是记搜的内容了。先写一个爆搜,然后把答案存在dp里,基本就是记搜了。

#include<cstdio>
#include<iostream>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<cctype>
#include<vector>
#include<stack>
#include<queue>
using namespace std;
#define enter puts("")
#define space putchar(' ')
#define Mem(a, x) memset(a, x, sizeof(a))
#define In inline
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-8;
const int maxn = 12;
inline ll read()
{
ll ans = 0;
char ch = getchar(), last = ' ';
while(!isdigit(ch)) last = ch, ch = getchar();
while(isdigit(ch)) ans = (ans << 1) + (ans << 3) + ch - '0', ch = getchar();
if(last == '-') ans = -ans;
return ans;
}
inline void write(ll x)
{
if(x < 0) x = -x, putchar('-');
if(x >= 10) write(x / 10);
putchar(x % 10 + '0');
} int dp[maxn][maxn], num[maxn], cnt = 0; In int dfs(int len, int las, bool _Max, bool zero)
{
if(!len) return 1; //表示出现了一个符合条件的数,就返回1
if(!_Max && !zero && dp[len][las] != -1) return dp[len][las];
//只有普通情况才可以返回记搜的答案.
int pos, ret = 0, Max = _Max ? num[len] : 9;
for(int i = 0; i <= Max; ++i)
{
if(abs(i - las) < 2) continue;
pos = (zero && !i) ? -INF : i; //如果前面都是0且这一位还填0,就标记为INF
ret += dfs(len - 1, pos, _Max && i == Max, pos == -INF);
}
if(!_Max && !zero) dp[len][las] = ret;
return ret;
} In int solve(int n)
{
Mem(dp, -1); cnt = 0;
while(n) num[++cnt] = n % 10, n /= 10; //处理每一位
return dfs(cnt, -INF, 1, 1);
} int main()
{
int a = read(), b = read();
write(solve(b) - solve(a - 1)), enter;
return 0;
}



**[luogu P3413 SAC#1 - 萌数](https://www.luogu.org/problemnew/show/P3413)**
这题反正我是不会,看了题解也没懂。最后在dukelv的帮助下搞明白了。
首先题解中有很多篇用了正难则反的思想,但其实没必要。
考虑最小的回文串,无非两种:aa, aba。
所以我们只用找出这两种回文串即可。因为大的回文串比如acbbca,在搜bb的时候,下几个状态就包括这个回文串了。
那么就要讨论奇偶,所以爆搜的时候开两个变量pre,per分别记录上一位和上上一位,然后判断和当前位是否构成回文串。
别忘了还可能会出现构不成萌数的情况,所以dp方程除了dp[i][j]表示到了第$i$位,这一位填$i$,还应再加一位[0/1]表示当前是否存在萌数,这样能剪枝不少。
```c++
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
#define enter puts("")
#define space putchar(' ')
#define Mem(a, x) memset(a, x, sizeof(a))
#define In inline
typedef long long ll;
typedef double db;
const int INF = 0x3f3f3f3f;
const db eps = 1e-8;
const int maxn = 1e3 + 5;
const ll mod = 1e9 + 7;
inline ll read()
{
ll ans = 0;
char ch = getchar(), last = ' ';
while(!isdigit(ch)) last = ch, ch = getchar();
while(isdigit(ch)) ans = (ans = 10) write(x / 10);
putchar(x % 10 + '0');
}

char a[maxn], b[maxn];

int num[maxn], len = 0;

ll dp[maxn][10][2];

In ll dfs(int pos, int pre, int per, bool _t, bool _k, bool _Max)

{

if(!pos) return _t; //_t表示当前是否存在萌数

if(!_Max && dp[pos][pre][_t] != -1) return dp[pos][pre][_t];

int Max = _Max ? num[pos] : 9;

ll ret = 0;

for(int i = 0; i <= Max; ++i)

ret += dfs(pos - 1, i, _k ? pre : -1, _t || (i == pre && _k) || (i == per && _k), _k || i, _Max && i == Max), ret %= mod;

if(!_Max && _k && per != -1) dp[pos][pre][_t] = ret % mod;

return ret;

}

In ll solve(char* s)

{

len = strlen(s + 1);

for(int i = 1; i <= len; ++i) num[i] = s[i] - '0';

reverse(num + 1, num + len + 1);

Mem(dp, -1);

return dfs(len, -1, -1, 0, 0, 1);

}

int main()

{

scanf("%s%s", a + 1, b + 1);

int tp = strlen(a + 1);

reverse(a + 1, a + tp + 1);

int pos = 1; //别忘了是高精减1……

--a[pos];

while(pos < tp && a[pos] < '0') a[pos++] = '9', --a[pos];

while(tp > 1 && a[tp] <= '0') a[tp--] = '\0';

reverse(a + 1, a + tp + 1);

write((solve(b) - solve(a) + mod) % mod), enter;

return 0;

}

数位dp小练的更多相关文章

  1. 【树形dp小练】HDU1520 HDU2196 HDU1561 HDU3534

    [树形dp]就是在树上做的一些dp之类的递推,由于一般须要递归处理.因此平庸情况的处理可能须要理清思路.昨晚開始切了4题,作为入门训练.题目都很easy.可是似乎做起来都还口以- hdu1520 An ...

  2. 牛客寒假算法基础集训营3处女座和小姐姐(三) (数位dp)

    链接:https://ac.nowcoder.com/acm/contest/329/G来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言52428 ...

  3. 处女座和小姐姐(三)(数位dp)

    链接:https://ac.nowcoder.com/acm/contest/329/G 来源:牛客网 题目描述 经过了选号和漫长的等待,处女座终于拿到了给小姐姐定制的手环,小姐姐看到以后直呼666! ...

  4. [bzoj3209][花神的数论题] (数位dp+费马小定理)

    Description 背景众所周知,花神多年来凭借无边的神力狂虐各大 OJ.OI.CF.TC …… 当然也包括 CH 啦.描述话说花神这天又来讲课了.课后照例有超级难的神题啦…… 我等蒟蒻又遭殃了. ...

  5. P2188 小Z的 k 紧凑数 题解(数位DP)

    题目链接 小Z的 k 紧凑数 解题思路 数位DP,把每一个数位的每一个数对应的可能性表示出来,然后求\(num(1,r)-num(1,l-1)\),其中\(num(i,j)\)表示\([i,j]\)区 ...

  6. 数位DP之小小结

    资料链接:http://wenku.baidu.com/view/9de41d51168884868662d623.html http://wenku.baidu.com/view/d2414ffe0 ...

  7. 牛客训练三:处女座和小姐姐(三)(数位dp)

    题目链接:传送门 思路:数位dp的记忆化搜索模板 从高位向低位枚举,逐位确定每一位的6的个数,dp[i][s]表示处理到第i条边,状态为s时的数字的个数. 注意,要使用long long类型. #in ...

  8. SCUT - 289 - 小O的数字 - 数位dp

    https://scut.online/p/289 一个水到飞起的模板数位dp. #include<bits/stdc++.h> using namespace std; typedef ...

  9. [CSP-S模拟测试]:小L的数(数位DP+模拟)

    题目传送门(内部题132) 输入格式 第一行一个整数$t$. 接下来$t$行每行一个整数$n$. 输出格式 $t$行,每行一个整数表示答案. 样例 样例输入: 41818231232691052109 ...

随机推荐

  1. python的Web框架,html分页

    使用简单的算法得出页码数,然后在html中获取即可.仅供参考. views的写法 def crm_stu(request): section = '教师后台管理页' search = request. ...

  2. nginx配置指南

    nginx(读作engine x)是一款设计优秀的Http服务器, 其占用内存少, 负载能力强且稳定性高, 正在被越来越多的用户所采用. nginx可以为HTTP, HTTPS, SMTP, POP3 ...

  3. S3C2440的中断体系结构

    概述 S3C2440A中的中断控制器接受来自60个中断源的请求.提供这些中断源的可以是内部外设,如DMA控制器.UART.IIC等等.在这些中断源中,UARTn.AC97和EINTn中断对于中断控制器 ...

  4. c# 导出表格

    var record = m_editor.getMasterRecord(); var Check_Id = record.Check_Id; var url = "/Storage/St ...

  5. Linux中ansible批量管理软件部署及剧本编写

    服务器版本信息: Centos6.9 [root@db02 ~]# uname -a Linux db02 2.6.32-696.el6.x86_64 #1 SMP Tue Mar 21 19:29: ...

  6. 本地计算机上的MySQL服务启动后停止。某些服务在未由其他服务或程序使用时将自动

    重新安装MySQL数据库,由于安装的时候马虎,一路next(事实上,某些地方需要严格的配置,我忘记注意了),导致现在出了很多麻烦. 错误信息: 本地计算机上的MySQL服务启动后停止.某些服务在未由其 ...

  7. 乐字节-Java8新特性之Date API

    上一篇文章,小乐给大家带来了Java8新特性之Optional,接下来本文将会给大家介绍Java8新特性之Date API 前言: Java 8通过发布新的Date-Time API来进一步加强对日期 ...

  8. Spring全家桶系列–SpringBoot之AOP详解

    //本文作者:cuifuan //本文将收录到菜单栏:<Spring全家桶>专栏中 面向方面编程(AOP)通过提供另一种思考程序结构的方式来补充面向对象编程(OOP). OOP中模块化的关 ...

  9. JavaScript复杂判断的更优雅写法

    摘要: 写代码是一门艺术. 原文:JavaScript 复杂判断的更优雅写法 作者:Think. 公众号:大转转fe Fundebug经授权转载,版权归原作者所有. 前提 我们编写js代码时经常遇到复 ...

  10. 浏览器兼容性问题——IE不支持却很实用的CSS属性Outline和Child

    1. Outline(适用范围:鼠标悬浮hover加外边框) 我们在布局的时候,常常会因为添加边框border影响宽高的布局. 那么,outline是完美的替代品,因为它可以在不影响文档流的情况下呈现 ...