acwing26动态规划

动态规划以前写过博客,最近发现以前的博客存在逻辑不完善的地方。借着学习acwing的算法的机会,再重新审视一下自己存在的问题。

动态规划

acwing大佬总结的模板:
acwing26动态规划_第1张图片

背包问题

最长公共子序列

给定两个长度分别为 N 和 M 的字符串 A 和 B,求既是 A 的子序列又是 B 的子序列的字符串长度最长是多少。

输入格式
第一行包含两个整数 N 和 M。

第二行包含一个长度为 N 的字符串,表示字符串 A。

第三行包含一个长度为 M 的字符串,表示字符串 B。

字符串均由小写字母构成。

输出格式
输出一个整数,表示最大长度。

数据范围
1≤N,M≤1000
输入样例:

4 5
acbd
abedc

输出样例:

3

这题之前写过博客,也算dp的一道入门题了。这次从集合的角度去理解问题,希望能有更深的感悟。
acwing26动态规划_第2张图片
先对问题分类:
集合表示:f[i][j]表示a的前i个字母,和b的前j个字母的最长公共子序列长度

集合划分:以a[i]a[i],b[j]b[j]是否包含在子序列当中为依据,因此可以分成四类:

  1. a[i]不在,b[j] 不在:max=f[i−1][j−1]。
  2. a[i]不在,b[j] 在:看似是max=f[i−1][j] , 无法实际上无法用f[i−1][j]表示,因为f[i−1][j]表示的是在a的前i-1个字母中出现,并且在b的前j个字母中出现,此时b[j]不一定出现,这与条件不完全相等,条件给定是a[i]一定不在子序列中,b[j]一定在子序列当中,但仍可以用f[i−1][j]来表示,原因就在于条件给定的情况被包含在f[i−1][j]中,即条件的情况是f[i−1][j]的子集,而求的是max,所以对结果不影响。例如:要求a,b,c的最大值可以这样求: max(max(a,b),max(b,c))虽然b被重复使用,但仍能求出max,求max只要保证不漏即可。
  3. a[i] 在,b[j] 不在 原理同②。
  4. a[i]在,b[j] 在:max=f[i−1][j−1]+ 1。
  5. 实际上,在计算时,①包含在②和③的情况中,所以①不用考虑
#include 
using namespace std;
const int N = 1010;
int n , m;
char a[N] , b[N];
int f[N][N];
int main()
{
     
    cin >> n >> m;
    cin >> a + 1 >> b + 1;
    for(int i = 1 ; i <= n ; i++)
        for(int j = 1 ; j <= m ; j++)
            {
     
                f[i][j] = max(f[i - 1][j] , f[i][j - 1]);//②和③的情况一定存在,所以可以无条件优先判断
                if(a[i] == b[j]) f[i][j] = max(f[i][j] , f[i - 1][j - 1] + 1);
            }                                                       
    cout << f[n][m] << endl;
    return 0;
}

石子合并(区间DP)

设有 N 堆石子排成一排,其编号为 1,2,3,…,N。

每堆石子有一定的质量,可以用一个整数来描述,现在要将这 N 堆石子合并成为一堆。

每次只能合并相邻的两堆,合并的代价为这两堆石子的质量之和,合并后与这两堆石子相邻的石子将和新堆相邻,合并时由于选择的顺序不同,合并的总代价也不相同。

例如有 4 堆石子分别为 1 3 5 2, 我们可以先合并 1、2 堆,代价为 4,得到 4 5 2, 又合并 1,2 堆,代价为 9,得到 9 2 ,再合并得到 11,总代价为 4+9+11=24;

如果第二步是先合并 2,3 堆,则代价为 7,得到 4 7,最后一次合并代价为 11,总代价为 4+7+11=22。

问题是:找出一种合理的方法,使总的代价最小,输出最小代价。

输入格式
第一行一个数 N 表示石子的堆数 N。

第二行 N 个数,表示每堆石子的质量(均不超过 1000)。

输出格式
输出一个整数,表示最小代价。

数据范围
1≤N≤300
输入样例:

4
1 3 5 2

输出样例:

22

区间 DP 常用模版:
所有的区间dp问题,第一维都是枚举区间长度,一般 len = 1 用来初始化,枚举从 len = 2 开始,第二维枚举起点 i (右端点 j 自动获得,j = i + len - 1)

for (int i = 1; i <= n; i++)
{
     
    dp[i][i] = 初始值
}
for (int len = 2; len <= n; len++)           //区间长度
    for (int i = 1; i + len - 1 <= n; i++) 
    {
      //枚举起点
        int j = i + len - 1;                 //区间终点
        for (int k = i; k < j; k++) 
        {
             //枚举分割点,构造状态转移方程
            dp[i][j] = max(dp[i][j], dp[i][k] + dp[k + 1][j] + w[i][j]);
        }
    }
#include 
#include 
using namespace std;
const int N = 310;
int n;
int s[N];
int f[N][N];
int main()
{
     
    scanf("%d", &n);
    for (int i = 1; i <= n; i ++ ) scanf("%d", &s[i]);
    for (int i = 1; i <= n; i ++ ) s[i] += s[i - 1];
    for (int len = 2; len <= n; len ++ )
        for (int i = 1; i + len - 1 <= n; i ++ )
        {
     
            int l = i, r = i + len - 1;
            f[l][r] = 1e8;
            for (int k = l; k < r; k ++ )
                f[l][r] = min(f[l][r], f[l][k] + f[k + 1][r] + s[r] - s[l - 1]);
        }
    printf("%d\n", f[1][n]);
    return 0;
}

记忆化搜索:

#include 
using namespace std;
int fd[510][5010];
int a[5010];
int n;
int dfs(int x, int y) {
     
    if (x == y) return 0;
    if (fd[x][y] != 0x3f3f3f3f) return fd[x][y];
    for (int k = x; k < y; k++) {
     
        fd[x][y] = min(fd[x][y], dfs(x, k) + dfs(k + 1, y) + a[y] - a[x - 1]);
    }
    return fd[x][y];
}
int main() {
     
    memset(fd, 0x3f, sizeof(fd));
    cin >> n;
    for (int i = 1; i <= n; i++) cin >> a[i];
    for (int i = 2; i <= n; i++) {
     
        a[i] = a[i - 1] + a[i];
    }
    cout << dfs(1, n) << endl;
}

最长上升子序列二分板子

#include 
#include 

using namespace std;

const int N = 100010;

int n;
int a[N];//存题目数据的数组 
int q[N];//存每个数对应上升子序列的数组 

int main()
{
     
    scanf("%d", &n);
    for (int i = 0; i < n; i ++ ) scanf("%d", &a[i]);

    int len = 0;
    for (int i = 0; i < n; i ++ )
    {
     
        int l = 0, r = len;
        while (l < r)
        {
     
            int mid = l + r + 1 >> 1;
            if (q[mid] < a[i]) l = mid;
            else r = mid - 1;
        }
        len = max(len, r + 1);
        q[r + 1] = a[i];//更新q这个数组
    }
    printf("%d\n", len); 
    return 0;
}


最短编辑距离

给定两个字符串 A 和 B,现在要将 A 经过若干操作变为 B,可进行的操作有:

删除–将字符串 A 中的某个字符删除。
插入–在字符串 A 的某个位置插入某个字符。
替换–将字符串 A 中的某个字符替换为另一个字符。
现在请你求出,将 A 变为 B 至少需要进行多少次操作。

输入格式
第一行包含整数 n,表示字符串 A 的长度。

第二行包含一个长度为 n 的字符串 A。

第三行包含整数 m,表示字符串 B 的长度。

第四行包含一个长度为 m 的字符串 B。

字符串中均只包含大写字母。

输出格式
输出一个整数,表示最少操作次数。

数据范围
1≤n,m≤1000
输入样例:

10 
AGTCTGACGC
11 
AGTAAGTAGGC

输出样例:

4

acwing26动态规划_第3张图片
一、状态表示
1.状态表示:dp[i][j]指的是把前i个字母变成b中前j个字母的集合,因为是两重循环枚举,可以包含所有组合。
2.属性:对于每个dp[i][j]进行的所有操作中,操作次数最少的方案的操作数。
二、状态计算
1.对任意一个i 串和 j 串,它们的子串为ii串和jj串,改变内部子串某一元素,如果使得 i串 和 j串 相等,那么它的解一定相等于 ii串 最后一个元素a[i] 变成 jj串最后一个元素的解,所以只需要对任意串 i , j :
acwing26动态规划_第4张图片
讨论使它们相等的最后一步:
(1)添加:如果a串添加当前字母变得与 j 串相同,说明a[i]已经与此时与 a[j-1]相同,即:dp[i][j]=dp[i][j-1] + 1
(2)删除:如果a串删除当前字母变得与 j 串相同,说明a[i-1]已经与a[j]相同,即:dp[i][j]=dp[i-1][j] + 1
(3)替换:如果当前结尾字母不同,则前i-1与j-1已经相同,即:dp[i][j]=dp[i-1][j-1] + 1
(4)a[i]与a[j] 已经相同,更新之前的状态:dp[i][j]=dp[i-1][j-1]

#include 
#include 

using namespace std;

const int N = 1010;

int n, m;
char a[N], b[N];
int f[N][N];

int main()
{
     
    scanf("%d%s", &n, a + 1);
    scanf("%d%s", &m, b + 1);

    for (int i = 0; i <= m; i ++ ) f[0][i] = i;//0变成i串,增加i次
    for (int i = 0; i <= n; i ++ ) f[i][0] = i;//i串变成0,删除i次

    for (int i = 1; i <= n; i ++ )
        for (int j = 1; j <= m; j ++ )
        {
     
            f[i][j] = min(f[i - 1][j] + 1, f[i][j - 1] + 1);
            if (a[i] == b[j]) f[i][j] = min(f[i][j], f[i - 1][j - 1]);
            else f[i][j] = min(f[i][j], f[i - 1][j - 1] + 1);
        }

    printf("%d\n", f[n][m]);

    return 0;
}

编辑距离

和上题一样,多个字符串先存起来,求一个在Limit操作次数内实现的方案数。

#include
#include
#include
#include
using namespace std;
const int N=1010;
char str[N][15];
int f[N][N];
int count(char a[],char b[])
{
       int n=strlen(a+1),m=strlen(b+1);
   for(int i=0;i<=n;i++) f[i][0]=i;
   for(int i=0;i<=m;i++) f[0][i]=i;
   for(int i=1;i<=n;i++)
   {
     
      for(int j=1;j<=m;j++)
      {
     
         f[i][j]=min(f[i][j-1]+1,f[i-1][j]+1);
         if(a[i]==b[j]) f[i][j]=min(f[i][j],f[i-1][j-1]);
         else f[i][j]=min(f[i][j],f[i-1][j-1]+1);
      }
   }
   return f[n][m];
}
int main()
{
     
     int n,m;
     cin>>n>>m;
     for(int i=1;i<=n;i++) scanf("%s",str[i]+1);
     while(m--)
     {
       
        int cnt=0,lim;
        char st[15];
        scanf("%s%d",st+1,&lim);
        for(int i=1;i<=n;i++)
        {
     
           if(count(str[i],st)<=lim) cnt++;
        }
        cout<<cnt<<endl;
     }
}

计数问题(数位统计DP)

给定两个整数 a 和 b,求 a 和 b 之间的所有数字中 0∼9 的出现次数。

例如,a=1024,b=1032,则 a 和 b 之间共有 9 个数如下:

1024 1025 1026 1027 1028 1029 1030 1031 1032

其中 0 出现 10 次,1 出现 10 次,2 出现 7 次,3 出现 3 次等等…

输入格式
输入包含多组测试数据。

每组测试数据占一行,包含两个整数 a 和 b。

当读入一行为 0 0 时,表示输入终止,且该行不作处理。

输出格式
每组数据输出一个结果,每个结果占一行。

每个结果包含十个用空格隔开的数字,第一个数字表示 0 出现的次数,第二个数字表示 1 出现的次数,以此类推。

数据范围
0 输入样例:

1 10
44 497
346 542
1199 1748
1496 1403
1004 503
1714 190
1317 854
1976 494
1001 1960
0 0

输出样例:

1 2 1 1 1 1 1 1 1 1
85 185 185 185 190 96 96 96 95 93
40 40 40 93 136 82 40 40 40 40
115 666 215 215 214 205 205 154 105 106
16 113 19 20 114 20 20 19 19 16
107 105 100 101 101 197 200 200 200 200
413 1133 503 503 503 502 502 417 402 412
196 512 186 104 87 93 97 97 142 196
398 1375 398 398 405 499 499 495 488 471
294 1256 296 296 296 296 287 286 286 247

数位dp分类讨论:
比如说我要找[1,abcdefg]中的数中1出现的个数
就得先求1在每一个位置上出现的次数

比如我要找第4位上出现的1的数有几个
就是要找满足 1 <= xxx1yyy <= abcdefg
(1) xxx∈[000,abc-1] , yyy∈[000,999] :如果前三位没填满,则后三位就可以随便填 :ans += abc*1000
(2)xxx == abc :如果前三位填满了,后三位怎么填取决于当前这一位

  1. if(d<1) yyy不存在 , ans += 0
  2. if(d==1) yyy∈[000,efg] , ans += efg+1
  3. if(d>1) yyy∈[000,999] , ans += 1000
#include
#include
#include
using namespace std;
int power10(int x)//求x的位数是10的几次方
{
     
    int res=1;
    while(x--) res*=10;
    return res;
}
int get(vector<int >num,int l,int r)//求某个区间内的数是谁
{
     
    int res=0;
    for(int i=l;i>=r;i--)
        res=res*10+num[i];
    return res;
}
int count(int n,int x)//求某个数在某个区间的每一位上的出现次数
{
     
    if(!n) return 0;
    vector<int >num;
    while(n)
    {
     
        num.push_back(n%10);
        n/=10;
    }
    n=num.size();
    int res=0;
    for(int i=n-1-!x;i>=0;i--)
    {
     
        if(i<n-1)
        {
     
            res+=get(num,n-1,i+1)*power10(i);
            if(!x) res-=power10(i);
        }
        if(num[i]==x) res+=get(num,i-1,0)+1;
        else if(num[i]>x) res+=power10(i);
       
    } 
    return res;
}
int main()
{
     
    int a,b;
    while(cin>>a>>b,a||b)
    {
     
        if(a>b) swap(a,b);
        for(int i=0;i<10;i++)
            cout<<count(b,i)-count(a-1,i)<<" ";
        cout<<endl;
    }
}

没有上司的舞会(树形DP)

状态表示
f[i][j]:其表示的是以 i 为树根,及其左右子树可以符合无上司条件的个数最大值,j 用来表示根节点 i 是不是存在与这个最大值当中。
比如求红圈内的无上司最大值:
acwing26动态规划_第5张图片
首先需要递归的求出s1,s2的最大值。
然后当前u树的答案个数一定是对u树的子树里的个数以某种方式累加求和。
状态计算
若当前u结点不选,子结点可选可不选

f[u][0]=∑max(f[si,0],f[si,1])

若当前u结点选,子结点一定不能选

f[u][1]=∑(f[si,0])

#include
#include
#include
using namespace std;
const int N=6010;
int n;
int h[N],e[N],ne[N],idx;
int happy[N];
int f[N][2];
bool has_father[N];//先看看谁没有父节点谁就是根
void add(int a,int b)
{
     
    e[idx]=b,ne[idx]=h[a],h[a]=idx++;
}
void dfs(int u)
{
     
    f[u][1]=happy[u];//如果选择这个点,加上这个点的快乐值
    for(int i=h[u];i!=-1;i=ne[i])
    {
     
        int j=e[i];
        dfs(j);
        f[u][0]+=max(f[j][0],f[j][1]);
        f[u][1]+=f[j][0];
    }
}
int main()
{
     
    cin>>n;
    for(int i=1;i<=n;i++) cin>>happy[i];
    memset(h,-1,sizeof h);
    for(int i=0;i<n-1;i++)
    {
     
        int a,b;
        cin>>a>>b;
        has_father[a]=true;
        add(b,a);
    }
    int root=1;
    while(has_father[root]) root++;
    dfs(root);
    printf("%d\n",max(f[root][0],f[root][1]));
    return 0;
}

整数划分(计数类dp)

一个正整数 n 可以表示成若干个正整数之和,形如:n=n1+n2+…+nk,其中 n1≥n2≥…≥nk,k≥1。

我们将这样的一种表示称为正整数 n 的一种划分。

现在给定一个正整数 n,请你求出 n 共有多少种不同的划分方法。

输入格式
共一行,包含一个整数 n。

输出格式
共一行,包含一个整数,表示总划分数量。

由于答案可能很大,输出结果请对 109+7 取模。

数据范围
1≤n≤1000
输入样例:

5

输出样例:

7

完全背包同思路推导。
f[i][j]表示的是用前 i 个数,组成 j 这个正整数,其中可行的方案数量。
acwing26动态规划_第6张图片
在这里插入图片描述

#include 
#include 

using namespace std;

const int N = 1010, mod = 1e9 + 7;

int n;
int f[N];

int main()
{
     
    cin >> n;

    f[0] = 1;//任何一个数组成0 为一种方案-全不选
    for (int i = 1; i <= n; i ++ )
        for (int j = i; j <= n; j ++ )
            f[j] = (f[j] + f[j - i]) % mod;//完全背包思路

    cout << f[n] << endl;

    return 0;
}

第二种思路的f[i][j]表示的是总和为 i ,并且由 j 个数组成的方案数。
acwing26动态规划_第7张图片
当最小值为1时,去掉最后一个1 可以得到f[i][j]=f[i-1][j-1]。
当最小值大于1时,给 j 个数每个数减去1,f[i][j]=f[i-j][j],这两种方案一一对应。
将两种情况加和即为f[1][j]的答案。
而最终的答案则是把f[n][1]到f[n][n]求和。

#include 
#include 

using namespace std;

const int N = 1010, mod = 1e9 + 7;

int n;
int f[N][N];

int main()
{
     
    cin >> n;

    f[1][1] = 1;
    for (int i = 2; i <= n; i ++ )
        for (int j = 1; j <= i; j ++ )//枚举到i即可,不可能用超过i个数表示i这个数
            f[i][j] = (f[i - 1][j - 1] + f[i - j][j]) % mod;

    int res = 0;
    for (int i = 1; i <= n; i ++ ) res = (res + f[n][i]) % mod;

    cout << res << endl;

    return 0;
}

你可能感兴趣的:(笔记,c++)