纪念卡常把自己卡死的一次自闭模拟赛

QWQ

一开始看这个题,以为是个图论,仔细一想,貌似可以直接dp啊。

首先,因为规则只有从两个变为1个,貌似可以用类似区间\(dp\)的方式来\(check\)一段区间能不能合成某一个字母!

那我们定义\(f[i][j][k]\)表示第一个串,\([l,r]\)区间,是否可以合成\(k\)这个字母

然后转移的时候,枚举区间,枚举规则,枚举断点,满足\(f[l][k][p1]==1\)且\(f[k+1][r][p2]==1\) 才能使当前状态合法。

其中\(p1,p2\)表示当前规则的两个字母

for (int i=1;i<=n;i++) f[i][i][cc(s[i])]=1;
for (register int i=2;i<=n;++i)
for (register int l=1;l<=n-i+1;++l)
{
int r = l+i-1;
for (register int j=1;j<=26;++j)
{
for (register int p=1;p<=num[j];++p)
{
for (register int k=l;k<=r;++k)
{
f[l][r][j]=max(f[l][r][j],f[l][k][a[j][p].a]&f[k+1][r][a[j][p].b]);
if (f[l][r][j]) break;
}
if (f[l][r][j]) break;
}
}
}

同时定义\(g[l][r][k]\)数组表示第二个串区间\([l,r]\)能否合成k。处理和f类似。

统计答案的时候呢

还需要一个\(dp[i][j]\)表示第一个串的前i个字符和第二个串的前j个字符的最短公共祖先

那么,考虑枚举两个断点,两个串的后面两段能合成同一个字母,那么就可以从那个断点之前的状态转移过来

QWQ

详细直接看代码吧

memset(dp,127/3,sizeof(dp));
dp[0][0]=0;
for (register int i=1;i<=nn;++i)
{
for (register int k=1;k<=n;++k)
{
for (register int j=1;j<=i;++j)
for (register int p=1;p<=k;++p)
{
if (dp[j-1][p-1]==dp[maxn-3][maxn-3]) continue;
bool flag=false;
for (register int o=1;o<=26;o++)
if (g[j][i][o] && f[p][k][o]) flag=true;
if (flag) dp[i][k]=min(dp[i][k],dp[j-1][p-1]+1);
}
}
}

最后复杂度就是\(O(n^4*26)\)

我也不知道为啥能跑过啊

qwqwqwq

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk makr_pair
#define ll long long
#include<ctime>
using namespace std;
inline int read()
{
int x=0,f=1;char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
const int maxn = 110;
struct Node{
int a,b;
};
Node a[maxn][maxn];
int num[maxn];
int f[maxn][maxn][maxn];
int g[maxn][maxn][maxn];
int n,m;
char s[maxn];
char ss[maxn];
int nn;
string ans;
int dp[maxn][maxn];
inline int cc(char c)
{
return c-'a'+1;
}
int main()
{
scanf("%s",s+1);
scanf("%s",ss+1);
s[0]=ss[0]='*';
n=strlen(s+1);
nn=strlen(ss+1);
m=read();
for (register int i=1;i<=m;++i)
{
char ymh[10];
scanf("%s",ymh+1);
int now = ymh[1]-'a'+1;
num[now]++;
a[now][num[now]].a = ymh[4]-'a'+1;
a[now][num[now]].b = ymh[5]-'a'+1;
}
for (int i=1;i<=n;i++) f[i][i][cc(s[i])]=1;
for (register int i=2;i<=n;++i)
for (register int l=1;l<=n-i+1;++l)
{
int r = l+i-1;
for (register int j=1;j<=26;++j)
{
for (register int p=1;p<=num[j];++p)
{
for (register int k=l;k<=r;++k)
{
f[l][r][j]=max(f[l][r][j],f[l][k][a[j][p].a]&f[k+1][r][a[j][p].b]);
if (f[l][r][j]) break;
}
if (f[l][r][j]) break;
}
}
}
for (int i=1;i<=nn;i++) g[i][i][cc(ss[i])]=1;
for (register int i=2;i<=nn;i++)
for (register int l=1;l<=nn-i+1;++l)
{
int r = l+i-1;
for (register int j=1;j<=26;++j)
{
for (register int p=1;p<=num[j];++p)
{
for (register int k=l;k<=r;++k)
{
g[l][r][j]=max(g[l][r][j],g[l][k][a[j][p].a]&g[k+1][r][a[j][p].b]);
if (g[l][r][j]) break;
}
if (g[l][r][j]) break;
}
}
}
memset(dp,127/3,sizeof(dp));
dp[0][0]=0;
for (register int i=1;i<=nn;++i)
{
for (register int k=1;k<=n;++k)
{
for (register int j=1;j<=i;++j)
for (register int p=1;p<=k;++p)
{
if (dp[j-1][p-1]==dp[maxn-3][maxn-3]) continue;
bool flag=false;
for (register int o=1;o<=26;o++)
if (g[j][i][o] && f[p][k][o]) flag=true;
if (flag) dp[i][k]=min(dp[i][k],dp[j-1][p-1]+1);
}
}
}
if(dp[nn][n]==dp[maxn-3][maxn-3]) dp[nn][n]=-1;
cout<<dp[nn][n]<<endl;
return 0;
}

CF49E Common ancestor(dp+dp+dp)的更多相关文章

  1. [PAT] 1143 Lowest Common Ancestor(30 分)

    1143 Lowest Common Ancestor(30 分)The lowest common ancestor (LCA) of two nodes U and V in a tree is ...

  2. [CF49E]Common ancestor

    [CF49E]Common ancestor 题目大意: 有两个由小写字母构成的字符串\(S\)和\(T(|S|,|T|\le50)\).另有\(n(n\le50)\)个形如\(a\to bc\)的信 ...

  3. 【Aizu2292】Common Palindromes(回文树)

    [Aizu2292]Common Palindromes(回文树) 题面 Vjudge 神TMD日语 翻译: 给定两个字符串\(S,T\),询问\((i,j,k,l)\)这样的四元组个数 满足\(S[ ...

  4. PAT A1143 Lowest Common Ancestor (30 分)——二叉搜索树,lca

    The lowest common ancestor (LCA) of two nodes U and V in a tree is the deepest node that has both U ...

  5. 【POJ3415】 Common Substrings(后缀数组|SAM)

    Common Substrings Description A substring of a string T is defined as: T(i, k)=TiTi+1...Ti+k-1, 1≤i≤ ...

  6. POJ 1330 Nearest Common Ancestors(Targin求LCA)

    传送门 Nearest Common Ancestors Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 26612   Ac ...

  7. [PAT] 1143 Lowest Common Ancestor(30 分)1145 Hashing - Average Search Time(25 分)

    1145 Hashing - Average Search Time(25 分)The task of this problem is simple: insert a sequence of dis ...

  8. LeetCode 236. 二叉树的最近公共祖先(Lowest Common Ancestor of a Binary Tree)

    题目描述 给定一棵二叉树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义: “对于有根树T的两个结点u.v,最近公共祖先表示一个结点x,满足x是u.v的祖先且x的深度尽可能大. ...

  9. ACM学习历程—HDU 3092 Least common multiple(数论 && 动态规划 && 大数)

    Description Partychen like to do mathematical problems. One day, when he was doing on a least common ...

随机推荐

  1. Kickstart部署之FTP架构

    原文转自:https://www.cnblogs.com/itzgr/p/10029551.html作者:木二 目录 一 准备 1.1 完整架构:Kickstart+DHCP+VSFTP+TFTP+P ...

  2. MySQL-存储引擎-MERGE

    MERGE存储引擎是一组Myisam表的组合,这些Myisam表必须结构完全相同,MERGE表本身并没有数据,对MERGE类型的表可以进行查询.更新.删除操作,这些操作实际上是对内部的Myisam表进 ...

  3. Docker私有镜像仓库Harbor

    一.安装Harbor(离线安装包的方式安装) 1.解压离线包 2.进入harbor目录中编辑harbor.yml 3.安装docker-compose yum -y install docker-co ...

  4. Identity角色管理三(创建角色)

    首先创建视图模型 using System.ComponentModel; using System.ComponentModel.DataAnnotations; namespace Shop.Vi ...

  5. [转]SpringBoot系列——花里胡哨的banner.txt

    Creating ASCII Text Banners from the Linux Command Line In Ubuntu, Debian, Linux Mint etc. $ sudo ap ...

  6. Java统计文件中字母个数

    import java.text.DecimalFormat; import java.io.File; import java.io.FileReader; import java.io.Buffe ...

  7. [NOIP2015 普及组] 扫雷游戏

    [NOIP2015 普及组] 扫雷游戏 难度:入门 题目描述 扫雷游戏是一款十分经典的单机小游戏.在nn行mm列的雷区中有一些格子含有地雷(称之为地雷格),其他格子不含地雷(称之为非地雷格).玩家翻开 ...

  8. git介绍-常用操作(一)

    Table of Contents 1  系列文章 2  git说明 3  git常用命令 3.1  基本操作 3.2  远程操作 4  查看git的配置 4.1  查看已配置项 4.2  其他配置 ...

  9. mysql 基础配置经验

    创建库: 排序:utf8_unicode_ci和utf8_general_ci对中.英文来说没有实质的差别.utf8_general_ci校对速度快,但准确度稍差. 普遍的意思utf8_unicode ...

  10. Mysql实现排序

    排序   SELECT     obj.user_id,obj.score,@rownum := @rownum + 1 AS rownum FROM     (         SELECT     ...