[TC11326]ImpossibleGame

题目大意:

一类字符串仅由'A','B','C','D'四种字母组成。对于这样的一个字符串\(S\),可以用以下两种方式之一修改这个字符串:

  1. 交换\(S\)中的相邻两个字母;
  2. 使用一种魔法。魔法共\(m(m\le50)\)种,其中第\(i\)种可以将\(S\)中一个等于\(X_i\)的子串替换为\(y_i\),保证\(|X_i|=|Y_i|\)。

求对于所有这样的长度为\(n(n\le30)\)的字符串\(S\),修改过程中出现的不同的字符串最多能有多少?

思路:

对于第一种操作,显然若两个字符串中各字符出现次数对应相等,则这两个字符串之间可以相互转化。

因此我们可以用一个三元组\((a,b,c)\)表示字符串的状态,其中\(a,b,c\)分别表示字符'A','B','C'出现的次数。

魔法就相当于在这些状态间连边。

缩点后在DAG上DP即可。

时间复杂度\(\mathcal O(n^3m)\)

源代码:

#include<stack>
#include<queue>
#include<string>
#include<vector>
class ImpossibleGame {
private:
using int64=long long;
static constexpr int N=31,S=5457;
int64 fac[N],w[S],f[S];
int id[N][N][N],tot,a[S],b[S],c[S],d[S];
void init(const int &n) {
for(register int i=fac[0]=1;i<=n;i++) {
fac[i]=fac[i-1]*i;
}
for(register int i=0;i<=n;i++) {
for(register int j=0;i+j<=n;j++) {
for(register int k=0;i+j+k<=n;k++) {
const int l=n-i-j-k;
id[i][j][k]=++tot;
a[tot]=i;
b[tot]=j;
c[tot]=k;
d[tot]=l;
w[tot]=fac[n]/fac[i]/fac[j]/fac[k]/fac[l];
}
}
}
}
struct Edge {
int u,v;
};
std::vector<Edge> edge;
std::vector<int> e[S];
inline void add_edge(const int &u,const int &v) {
e[u].emplace_back(v);
}
bool ins[S];
std::stack<int> s;
int dfn[S],low[S],scc[S],ind[S];
void tarjan(const int &x) {
dfn[x]=low[x]=++dfn[0];
s.push(x);
ins[x]=true;
for(auto &y:e[x]) {
if(!dfn[y]) {
tarjan(y);
low[x]=std::min(low[x],low[y]);
} else if(ins[y]) {
low[x]=std::min(low[x],dfn[y]);
}
}
if(low[x]==dfn[x]) {
int y;
scc[0]++;
do {
y=s.top();
s.pop();
ins[y]=false;
scc[y]=scc[0];
f[scc[0]]+=w[y];
} while(y!=x);
}
}
std::queue<int> q;
int64 dis[S];
int64 kahn() {
for(register int i=1;i<=scc[0];i++) {
if(!ind[i]) {
dis[i]=f[i];
q.push(i);
}
}
int64 ret=0;
while(!q.empty()) {
const int x=q.front();
q.pop();
for(auto &y:e[x]) {
dis[y]=std::max(dis[y],dis[x]+f[y]);
if(!--ind[y]) {
q.push(y);
}
}
ret=std::max(ret,dis[x]);
}
return ret;
}
public:
int64 getMinimum(const int &n,const std::vector<std::string> &s,const std::vector<std::string> &t) {
init(n);
const int m=s.size();
for(register int i=0;i<m;i++) {
int w=0,x=0,y=0,z=0,o=0,p=0,q=0,r=0;
const int l=s[i].length();
for(register int j=0;j<l;j++) {
if(s[i][j]=='A') w++;
if(s[i][j]=='B') x++;
if(s[i][j]=='C') y++;
if(s[i][j]=='D') z++;
if(t[i][j]=='A') o++;
if(t[i][j]=='B') p++;
if(t[i][j]=='C') q++;
if(t[i][j]=='D') r++;
}
for(register int i=1;i<=tot;i++) {
if(a[i]<w||b[i]<x||c[i]<y||d[i]<z) continue;
const int &j=id[a[i]-w+o][b[i]-x+p][c[i]-y+q];
if(i==j) continue;
add_edge(i,j);
edge.emplace_back((Edge){i,j});
}
}
for(register int i=1;i<=tot;i++) {
if(!dfn[i]) tarjan(i);
}
for(register int i=1;i<=tot;i++) {
e[i].clear();
}
for(auto &e:edge) {
const int &u=e.u,&v=e.v;
if(scc[u]!=scc[v]) {
add_edge(scc[u],scc[v]);
ind[scc[v]]++;
}
}
return kahn();
}
};

[TC11326]ImpossibleGame的更多相关文章

  1. NOI.AC NOIP模拟赛 第二场 补记

    NOI.AC NOIP模拟赛 第二场 补记 palindrome 题目大意: 同[CEOI2017]Palindromic Partitions string 同[TC11326]Impossible ...

随机推荐

  1. 35个java代码性能优化总结

    前言 代码优化,一个很重要的课题.可能有些人觉得没用,一些细小的地方有什么好修改的,改与不改对于代码的运行效率有什么影响呢?这个问题我是这么考虑 的,就像大海里面的鲸鱼一样,它吃一条小虾米有用吗?没用 ...

  2. 【Linux】MySQL安装及允许远程访问

    安装环境/工具  Linux( centOS 版) MySQL(MySQL-5.6.28-1.el7.x86_64.rpm-bundle.tar版) 安装步骤 1.解压mysql安装文件 命令:tar ...

  3. c++ 函数指针简单实例

    一开始看函数指针的时候我是很懵的,因为不知道它有什么用,之后慢慢就发现了自己的愚昧无知. 假设我们想实现一个数据结构,比如二叉搜索树,堆.又或者是一个快排,归并排序. 我们一般是直接在两个数要比较的时 ...

  4. 【linux kernel】 中断处理-中断下半部【转】

    转自:http://www.cnblogs.com/embedded-tzp/p/4453987.html 欢迎转载,转载时需保留作者信息,谢谢. 邮箱:tangzhongp@163.com 博客园地 ...

  5. win10 安装IIS说明操作

    1.点左下角的Windows,所有应用,找到Windows系统,打开控制面板. 2.进入控制面板之后点击程序,可能你的控制面板和图片里的不太一样,不过没关系,找到程序两个字点进去就行. 3.接下来,在 ...

  6. pwd、ln和重定向命令

    pwd命令 命令功能: ​ 使用pwd命令可以显示当前的工作目录,该命令很简单,直接输入pwd即可,后面不带参数. ​ pwd命令以绝对路径的方式显示用户当前工作目录.命令将当前目录的全路径名称(从根 ...

  7. easyui表格,单元格合并

    easyui的合并单元格比较麻烦,官网提供一下方法 $('#tt').datagrid({ onLoadSuccess:function(){ var merges = [{ index:2, row ...

  8. ASP.NET中Request.ApplicationPath、Request.FilePath、Request.Path、.Request.MapPath、

    1.Request.ApplicationPath->当前应用的目录    Jsp中, ApplicationPath指的是当前的application(应用程序)的目录,ASP.NET中也是这 ...

  9. IOC创建对象的几种方式

    接上一篇IOC入门 IOC创建对象的几种方式 1)调用无参数构造器 2)带参数构造器 3)工厂创建对象 工厂类:静态方法创建对象 工厂类:非静态方法创建对象 1.对之前的User类进行一些修改,加上一 ...

  10. 定制自己的new和delete:operator new 和 operator delete

    new和delete不同用法 基本用法 int * aptr = new int(10); delete aptr, aptr = nullptr; 上面的代码是我们最基本也是最常见的使用new和de ...