正题

题目链接:https://www.ybtoj.com.cn/contest/115/problem/2


题目大意

给出一个包含字母变量和若干种同级操作符的后缀表达式。求一个等价的表达式满足该表达式的连续相同段最少。

\(1\leq |S|\leq 2500\)


解题思路

构建出表达树先,然后看一下什么能够化简,

  1. 两个相邻的相同运算符可以合并
  2. 一个非叶子节点下的相同叶子节点(字母节点)可以合并

先把这些合并了,然后目前的最优解就是现在的节点数量,但是还有一种情况可以合并。

就是兄弟节点中,非叶子节点和叶子节点可以合并。

用类树形\(dp\)求出所有节点的子树中的所有表达式的最优答案,如果不考虑上面那种情况就有

\[ans_i=1+\sum_{x->y}ans_y
\]

然后考虑一个非叶子节点在最优情况下能否以某个字母作为开头,定义\(avl_{x,c}\)表示\(x\)节点在\(ans\)最大的情况下能否以\(c\)作为开头。(因为上面那种情况最多剩下一个费用,如果这里牺牲了子树的最优性那么至少需要增加一点费用,显然是一定不优的)

那么对于一个节点的所有儿子,将非叶子节点和叶子节点分成二分图,如果非叶子节点的\(x\)满足\(avl_{x,c}=1\),那么向\(c\)连边。

然后跑二分图匹配就是\(ans\)可以减去的价值。

如何求出\(avl\)?如果一个字母\(c\)是\(x\)的儿子那么显然可以作为开头,否则如果有一个字母\(c\)满足\(x\)的一个非叶子儿子\(y\)使得\(avl_{y,c}=1\),并且在二分图上删去\(y\)节点不会影响答案时,此时将该子树作为开头即可。

如何判断删除一个节点后最大匹配不变,如果原图中该点没有匹配显然可以直接删去。如果有匹配,那么将该节点打上禁止标记后从它的匹配点开始求一条增广路,如果有则可以删去。

时间复杂度\(O(n^2)\)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std;
const int N=3100;
struct node{
int to,next;
}a[N];
int n,cnt,tot,ls[N],ans[N];
bool del[N],leaf[N],ch[N][27],avl[N][27];
char s[N];stack<int> st;
namespace M{
node a[N*27];bool v[27];
int tot,ls[27],link[N];
void clear(){
for(int i=1;i<=cnt;i++)link[i]=0;
for(int i=1;i<=26;i++)ls[i]=0;
tot=cnt=0;return;
}
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
int find(int x){
if(v[x])return 0;
int p,q;v[x]=1;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
p=link[y];link[y]=x;
if(!p||find(p))return 1;
link[y]=p;
}
return 0;
}
int Match(int x){
memset(v,0,sizeof(v));
return find(x);
}
int Path(int x){
if(!link[x])return 1;
return Match(link[x]);
}
}
bool isabc(char c)
{return (c>='a')&&(c<='z');}
void addl(int x,int y){
if(leaf[y])ch[x][s[y]-'a'+1]=1;
else{
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;
}
return;
}
void Merge(int x,int y){
for(int i=1;i<=26;i++)ch[x][i]|=ch[y][i];
for(int i=ls[y];i;i=a[i].next)
if(!del[a[i].to])addl(x,a[i].to);
del[y]=1;return;
}
void dfs(int x){
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;dfs(y);
if(s[x]==s[y])Merge(x,y);
}
for(int i=1;i<=26;i++)ans[x]+=ch[x][i];
ans[x]++;return;
}
void dp(int x){
for(int i=ls[x];i;i=a[i].next)
if(!del[a[i].to])dp(a[i].to);
M::clear();
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(del[y])continue;
++cnt;ans[x]+=ans[y];
for(int j=1;j<=26;j++)
if(avl[y][j]&&ch[x][j])
M::addl(j,cnt);
}
for(int i=1;i<=26;i++)
if(ch[x][i])ans[x]-=M::Match(i);
for(int i=1;i<=26;i++)avl[x][i]=ch[x][i];
for(int i=ls[x],p=0;i;i=a[i].next){
int y=a[i].to;
if(del[y])continue;p++;
if(M::Path(p)){
for(int j=1;j<=26;j++)
avl[x][j]|=avl[y][j];
}
}
return;
}
int main()
{
freopen("expr.in","r",stdin);
freopen("expr.out","w",stdout);
scanf("%s",s+1);
int l=strlen(s+1);
for(int i=1;i<=l;i++){
++n;
if(isabc(s[i]))leaf[n]=1;
else{
addl(n,st.top());st.pop();
addl(n,st.top());st.pop();
}
st.push(n);
}
dfs(n);
dp(n);
printf("%d\n",ans[n]);
return 0;
}

YbtOJ#573-后缀表达【二分图匹配】的更多相关文章

  1. cogs_14_搭配飞行员_(二分图匹配+最大流,网络流24题#01)

    描述 http://cojs.tk/cogs/problem/problem.php?pid=14 有一些正飞行员和副飞行员,给出每个正飞行员可以和哪些副飞行员一起飞.一架飞机上必须一正一副,求最多多 ...

  2. Cogs 14. [网络流24题] 搭配飞行员(二分图匹配)

    [网络流24题] 搭配飞行员 ★★☆ 输入文件:flyer.in 输出文件:flyer.out 简单对比 时间限制:1 s 内存限制:128 MB [问题描述] 飞行大队有若干个来自各地的驾驶员,专门 ...

  3. UVA 12549 - 二分图匹配

    题意:给定一个Y行X列的网格,网格种有重要位置和障碍物.要求用最少的机器人看守所有重要的位置,每个机器人放在一个格子里,面朝上下左右四个方向之一发出激光直到射到障碍物为止,沿途都是看守范围.机器人不会 ...

  4. POJ 1274 裸二分图匹配

    题意:每头奶牛都只愿意在她们喜欢的那些牛栏中产奶,告诉每头奶牛愿意产奶的牛棚编号,求出最多能分配到的牛栏的数量. 分析:直接二分图匹配: #include<stdio.h> #includ ...

  5. BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2375  Solved: 1005[Submit][Sta ...

  6. HDU1281-棋盘游戏-二分图匹配

    先跑一个二分图匹配,然后一一删去匹配上的边,看能不能达到最大匹配数,不能这条边就是重要边 /*----------------------------------------------------- ...

  7. HDU 1083 网络流之二分图匹配

    http://acm.hdu.edu.cn/showproblem.php?pid=1083 二分图匹配用得很多 这道题只需要简化的二分匹配 #include<iostream> #inc ...

  8. hdu 5727 Necklace dfs+二分图匹配

    Necklace/center> 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5727 Description SJX has 2*N mag ...

  9. BZOJ 1059 & 二分图匹配

    题意: 判断一个黑白染色的棋盘能否通过交换行或列使对角线上都是黑色. SOL: 真是有点醉...这种问题要么很神要么很水...第一眼感觉很水但就是不造怎么做...想了10分钟怎么感觉就是判断个数够不够 ...

随机推荐

  1. C# 插件编写

    //加载插件 private void LoadPlugins() { string path = Path.Combine(Path.GetDirectoryName(Assembly.GetExe ...

  2. C#中Finalize方法的问题

    ninputer在关于"值类型的Finalize不会被调用"中(http://blog.joycode.com/lijianzhong/archive/2005/01/13/429 ...

  3. LeetCoded第739题题解--每日温度

    每日温度 请根据每日 气温 列表,重新生成一个列表.对应位置的输出为:要想观测到更高的气温,至少需要等待的天数.如果气温在这之后都不会升高,请在该位置用 0 来代替. 例如,给定一个列表 temper ...

  4. js之window对象(慕课网学习笔记)

    javaScript定义了一个变量一个函数都会变成window中的一个成员 var a=1; alert(window.a) //会输出a的值 window基础 创建窗口.调整窗口.移动窗口.关闭窗口 ...

  5. CompletionService简介、原理以及小案例

    博客1:http://www.oschina.net/question/12_11255 博客2: CompletionService简介 CompletionService与ExecutorServ ...

  6. CSS布局中最小高度的妙用

    CSS布局中最小高度的妙用 --最小高度可以设定一个BOX的最小高度,当其内容较少时时,也能保持BOX的高度为一定,超出就自动向下延伸最小高度可以设定一个BOX的最小高度,当其内容较少时时,也能保持B ...

  7. Shiro03

    1.shiro授权角色.权限 2.Shiro的注解式开发 shiro权限思路 授权 ShiroUserMapper中定义两个方法 // 通过用户ID查询角色 Set<String> get ...

  8. Docker下制作一个容器镜像

    操作过程描述: (1)先基于centos的镜像启动一个centos容器 (2)在这个容器中安装nginx (3)然后把这个已经安装了nginx的容器制作成一个docker的镜像 操作:docker c ...

  9. [考试总结]noip模拟45

    真开心,挂没了.. 考完:"你们怎么第二题打了这么点分,明明一个爆搜就有65pts!!!怎么跟别人打?!" 然后我看了看我的爆搜,30pts. 然后认为自己打爆了... 我又想为什 ...

  10. jooq使用示例

    一.说明 最近使用的项目,采用了jooq. 通过学习api文档和自我调试,写了一些代码,在此处进行记录. 二.代码 一切尽在代码中-- 参考文档:http://www.jooq.org/doc/3.1 ...