(ps:本集合为Star_F总结的dp进阶知识,持续更新~。 转载本文章需要联系我,否则视为侵权!!)

前置知识:线性dp,背包,树形dp,区间dp

内容预览:

  • 状压dp
  • 数位dp
  • dp优化(前缀和,单调队列,斜率优化)

1. 状压dp:

思路:如果题目中 \(n\) 的范围特别小(\(<=20\)) 大概率可以状压dp

状态:f[i][j]:考虑到前 \(i\) 个,且已考虑的集合为 \(j\) (j为二进制数,1表示考虑,0表示不考虑)

的集合

(辅助知识:位运算)

例题:

  • P1171 售货员的难题

    dp[i][j] 表示从起点到第j号点 且到达时状态恰好为i的最短路

    比较简单的状压dp,之间看代码吧:
点击查看代码
#include<bits/stdc++.h>
using namespace std;
int f[1<<20][20],w[20][20],n;
int main(){
cin>>n;
memset(f,0x3f,sizeof f);
f[1][0]=0;
for(int i=0;i<n;++i)
for(int j=0;j<n;++j)
cin>>w[i][j];
for(int i=1;i<(1<<n);i+=2) //枚举状态
for(int j=0;j<n;j++){ //枚举下一步到达的点
if(!((i >> j) & 1)) continue;
for(int k=0;k<n;k++){ 枚举中介点
if(j==k) continue;
if(!(i>>k &1)) continue;
f[i][j]=min(f[i][j],f[i^(1<<j)][k]+w[k][j]);
}
}
int minn=2e9;
for(int i=0;i<=n-1;++i) minn=min(minn,f[(1<<n)-1][i]+w[i][0]);
cout<<minn<<endl;
return 0;
}
  • P1896 [SCOI2005] 互不侵犯

    思路:f[i][j][s]就表示在只考虑前i行时,在前i行(包括第i行)有且仅有s个国王,且第i行国王的情况是编号为j的状态时情况的总数。而k就代表第i-1行的国王情况的状态编号。
点击查看代码
	#include<cstdio>
#include<iostream>
#include<cstring>
#include<cmath>
#include<string>
#include<algorithm>
using namespace std;
int sit[2000],gs[2000];
int cnt=0;
int n,yong;
long long f[10][2000][100]={0};
void dfs(int he,int sum,int node)//预处理出每一个状态
{
if(node>=n)//如果已经处理完毕(注意是大于等于)
{
sit[++cnt]=he;
gs[cnt]=sum;
return;//新建一个状态
}
dfs(he,sum,node+1);//不用第node个
dfs(he+(1<<node),sum+1,node+2);//用第node个,此时node要加2,及跳过下一个格子
}
int main()
{
scanf("%d%d",&n,&yong);
dfs(0,0,0);
for(int i=1;i<=cnt;i++)f[1][i][gs[i]]=1;//第一层的所有状态均是有1种情况的
for(int i=2;i<=n;i++)
for(int j=1;j<=cnt;j++)
for(int k=1;k<=cnt;k++)//枚举i、j、k
{
if(sit[j]&sit[k])continue;
if((sit[j]<<1)&sit[k])continue;
if(sit[j]&(sit[k]<<1))continue;//排除不合法国王情况
for(int s=yong;s>=gs[j];s--)f[i][j][s]+=f[i-1][k][s-gs[j]];//枚举s,计算f[i][j][s]
}
long long ans=0;
for(int i=1;i<=cnt;i++)ans+=f[n][i][yong];//统计最终答案,记得用long long
printf("%lld",ans);
return 0;
}

2. 数位dp:

以一个数的数位进行dp,一般为求一个区间满足某些条件的数

通常把区间改为\((1...r) - (1...l-1)\)

例题:

点击查看代码
#include <cstring>
#include <iostream>
#include <algorithm>
#include <vector> using namespace std; const int N = 11; int f[N][10]; void init()
{
for (int i = 0; i <= 9; i ++ ) f[1][i] = 1; for (int i = 2; i < N; 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 dp(int n)
{
if (!n) return 0; vector<int> nums;
while (n) nums.push_back(n % 10), n /= 10; int res = 0;
int last = -2;
for (int i = nums.size() - 1; i >= 0; i -- )
{
int x = nums[i];
for (int j = i == nums.size() - 1; j < x; j ++ )
if (abs(j - last) >= 2)
res += f[i + 1][j]; if (abs(x - last) >= 2) last = x;
else break; if (!i) res ++ ;
} for (int i = 1; i < nums.size(); i ++ )
for (int j = 1; j <= 9; j ++ )
res += f[i][j]; return res;
} int main()
{
init(); int l, r;
cin >> l >> r;
cout << dp(r) - dp(l - 1) << endl; return 0;
}
  • CF628D

    递归做法,枚举到那一位,现在的属性,有没有什么限制
点击查看代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int mod = 1e9+7;
ll m,d,a[2005],len,f[2005][2005];
char l[2005],r[2005];
ll dfs(int u,int sum,int yaoqiu){
if(u>len) return sum==0?1:0;
if(!yaoqiu&&f[u][sum]!=-1) return f[u][sum];
ll tmp=0,maxx=yaoqiu==1?a[u]:9;
if(u%2==1){
for(int i=0;i<=maxx;i++)
if(i!=d) tmp=(tmp+dfs(u+1,(sum*10+i)%m,yaoqiu&&(i==maxx)))%mod;
}
else{
for(int i=0;i<=maxx;i++)
if(i==d) tmp=(tmp+dfs(u+1,(sum*10+i)%m,yaoqiu&&(i==maxx)))%mod;
}
if(!yaoqiu) f[u][sum]=tmp;
return tmp;
}
ll dp(char *s){
memset(f,-1,sizeof(f));
len=strlen(s+1);
for(int i=1;i<=strlen(s+1);i++) a[i]=s[i]-'0';
return dfs(1,0,1);
}
bool check(char *s){
len=strlen(s+1);
int x=0;
for(int i=1;i<=len;i++){
int y=s[i]-'0';
x=(x*10+y)%m;
if(i&1){
if(y==d)
return false;
}
else{
if(y!=d)
return false;
}
}
return !x;
}
int main(){
cin>>m>>d>>l+1>>r+1;
cout<<(dp(r)-dp(l)+check(l)+mod)%mod;
return 0;
}

3.dp优化:

DP进阶合集的更多相关文章

  1. Codeforces - tag::dp 大合集 [占坑 6 / inf]

    Gym - 100753J 某国家仅有金币和银币两种货币,起汇率为g,纪念品市场有n个商人和商品,商人结帐只用银币,并且把一堆银币装在袋子里,分为三种类型,分别按向下/向上/四舍五入取整(其中向上的优 ...

  2. DP小合集

    1.Uva1625颜色的长度 dp[i][j]表示前一个串选到第i个 后一个串选到第j个 的最小价值 记一下还有多少个没有结束即dp2 记一下每个数开始和结束的位置 #include<cstdi ...

  3. 【视频合集】极客时间 react实战进阶45讲 【更新中】

    https://up2.v.sharedaka.com/video/ochvq0AVfpa71A24bmugS5EewhFM1553702519936.mp4 01 React出现的历史背景及特性介绍 ...

  4. dp合集 广场铺砖问题&&硬木地板

    dp合集 广场铺砖问题&&硬木地板 很经典了吧... 前排:思想来自yali朱全民dalao的ppt百度文库免费下载 后排:STO朱全民OTZ 广场铺砖问题 有一个 W 行 H 列的广 ...

  5. 9.15 DP合集水表

    9.15 DP合集水表 显然难了一些啊. 凸多边形的三角剖分 瞄了一眼题解. 和蛤蛤的烦恼一样,裸的区间dp. 设f[i][j]表示i~j的点三角剖分最小代价. 显然\(f[i][i+1]=0,f[i ...

  6. 9.14 DP合集水表

    9.14 DP合集水表 关键子工程 在大型工程的施工前,我们把整个工程划分为若干个子工程,并把这些子工程编号为 1. 2. --. N:这样划分之后,子工程之间就会有一些依赖关系,即一些子工程必须在某 ...

  7. DP+贪心水题合集_C++

    本文含有原创题,涉及版权利益问题,严禁转载,违者追究法律责任 本次是最后一篇免费的考试题解,以后的考试题目以及题解将会以付费的方式阅读,题目质量可以拿本次作为参考 本来半个月前就已经搞得差不多了,然后 ...

  8. 掘金 Android 文章精选合集

    掘金 Android 文章精选合集 掘金官方 关注 2017.07.10 16:42* 字数 175276 阅读 50053评论 13喜欢 669 用两张图告诉你,为什么你的 App 会卡顿? - A ...

  9. Android 自定义View合集

    自定义控件学习 https://github.com/GcsSloop/AndroidNote/tree/master/CustomView 小良自定义控件合集 https://github.com/ ...

  10. [题解+总结]动态规划大合集II

    1.前言 大合集总共14道题,出自江哥之手(这就没什么好戏了),做得让人花枝乱颤.虽说大部分是NOIP难度,也有简单的几道题目,但是还是做的很辛苦,有几道题几乎没思路,下面一道道边看边分析一下. 2. ...

随机推荐

  1. Docker 启动 Redis 就停止解决方案(2022-3)

    启动命令如下: docker run -itd \ -p 6379:6379 \ --name myredis \ -v /home/redis/redis.conf:/etc/redis/redis ...

  2. .NET 6+Semantic Kernel快速接入OpenAI接口

    大家好,我是Edison. 今天我们快速地使用Semantic Kernel来集成OpenAI,使用20来行代码快速实现一个简单的AIGC应用. 这里,我就不多介绍Semantic Kernel了,包 ...

  3. 8.18考试总结(NOIP模拟43)[第一题·第二题·第三题·第四题]

    愿你和重要的人,在来日重逢. 前言 题目名字起的很随意... 这天 Luogu 的运势好像是大凶(忌:打模拟赛,注意报零). 但是考得还不错,拿到了这么多场模拟赛以来第二三个场上AC. 所以说,我爱大 ...

  4. Opencv笔记(11)随机数发生器cv::RNG

    一个随机数对象(RNG)用来产生随机数的伪随机序列.这样做的好处是你可以方便地得到多重伪随机数流.一旦随机数发生器创建,就会开始按需提供产生随机数的"服务",无论是平均分布还是正态 ...

  5. 网页唤起qq加群

    今天在网上大浪淘沙了一番,90%的教程都是使用的鹅厂官方的加群组件,但是有个致命bug就是这个加群组件只能唤起自己创建的群,这就很尴尬了,后来偶然发现在qq群资料那边就可以直接获取到加群的链接,害得我 ...

  6. java.io.File类中分隔符区别

    1.separator File.separator是系统默认的文件分隔符号,在UNIX系统上,这个字段的值是'/';在Microsoft Windows系统上,它是''. 类型:String 2.s ...

  7. 网易面试:SpringBoot如何开启虚拟线程?

    虚拟线程(Virtual Thread)也称协程或纤程,是一种轻量级的线程实现,与传统的线程以及操作系统级别的线程(也称为平台线程)相比,它的创建开销更小.资源利用率更高,是 Java 并发编程领域的 ...

  8. SQL 如何去掉字段中千位的逗号(比如set @= '1,320.00' 想得到@= '1320.00' )

    1/去掉字段里的逗号.(比如set @= '1,320.00' 想得到@= '1320.00' )UPDATE table SET fieldA = REPLACE(fieldA, ',', '') ...

  9. 338 warnings potentially fixable with the `--fix` option.

    将lint的值设置为eslint --fix --ext .js,.vue src,重启就可以了或者直接把lint这行删除掉

  10. TiKV 源码分析之 PointGet

    作者:来自 vivo 互联网存储研发团队-Guo Xiang 本文介绍了TiDB中最基本的PointGet算子在存储层TiKV中的执行流程. 一.背景介绍 TiDB是一款具有HTAP能力(同时支持在线 ...