题目描述

考古学家发现了一堵写有未知语言的白色墙壁,上面有一个n行m列的格子,其中有些格子内被填入了某个A至Z的大写字母,还有些格子是空白的。

一直横着或竖着的连续若干个字母会形成一个单词,且每一行的阅读顺序可能是从左向右或从右向左,每一列的阅读顺序可能是从下往上或从上往下。也就是说对于每一行来说,从左向右可以被看做是若干个单词形成的句子,相邻两个单词被一个或多个空白格子分割开来;也有可能是从右向左被看成是一个句子,竖直方向类似。

遗憾的是,我们并不完全知道每一行每一列的阅读顺序是怎样的。但可以猜测,有些单词会满足反转过来也是一个单词。例如单词BOY,翻转过来的YOB也是一个英文单词。

此外观察者发现,对每一行(列)来说,按照确定后的阅读顺序读出的所有单词同时满足“自己的字典序不小于翻转后的字典序”,或同时满足“自己的字典序不大于翻转后的字典序”。

在确定了所有行列的阅读顺序之后,我们可以构造出关于这种未知语言的字典。

请问字典中出现的“翻转过来也是一个单词”的单词最少有多少种请注意,如果一个单词翻转后是不同的另外一个单词,它们需要被分别计入;而对于本身是回文的单词则不需要重复计入

输入输出格式

输入格式:

第一行一个整数T,表示T组测试数据。

对于每一组数据来说:第一行输入两个整数n,m。

第二行给出了n个数字,对应n行,其中若第i个数字为1,则表示第i行的阅读顺序从左往右;若为-1则为从右向左;若为0则表示无法确定

第三行给出了m个数字,对应n行,其中若第i个数字为1,则表示第i列的阅读顺序从上往下;若为-1则为从下向上;若为0则表示无法确定

之后n行,每行给出了长度为m的字符串,由A~Z和下划线组成,对应了每个格子的符号,其中下划线表示格子为空。

输出格式:

输出T行。每一组数据输出一行一个整数,表示最少有多少个单词,满足翻转后依然是单词。

注意,如果一个单词是回文,那么它一定满足“翻转后依旧是单词”

输入输出样例

输入样例#1:

1

2 10

0 0

0 0 0 0 0 0 0 0 0 0

ADA_JARVIS

ADA_SIVRAJ

输出样例#1:

>3

说明

对于100%的数据,1<=n,m<=72,T<=64


题解

最小割

好久不写网络流现在我的网络流水平真低==

然后想了想似乎\(dp\)并不容易记录状态,那就应该是一个网络流了

发现要求正着倒着都存在的单词的对数最小

那么要不是个费用流,要不就是最小割,要不就是反面计数了

然后分析搜题解一下觉得应该是最小割

最小割连的一条边就表示把ta们分在不同集合的代价

那么可以发现回文串无论怎么翻转贡献都是1

所以不考虑回文串

然后考虑其他的单词

对于一个单词,如果这个单词和翻转后的这个单词同时出现那么就会产生2的贡献

所以我们对于每对单词,字典序小的往字典序大的单词连一条边权为2的边

然后再考虑读法的问题

可以发现行列没啥关系,所以相同方法考虑即可

如果给定读法是从左往右且这样读的单词字典序比倒着读小

那么就从\(S\)往该行连一条\(INF\)的边,然后这行向从左往右读的这行的单词连一条\(INF\)的边

如果给定读法是从左往右且这样读的单词字典序比倒着读大

那么就从这一行从左往右读的单词往这一行连一条\(INF\)的边,这行往\(T\)连一条\(INF\)的边

这样在跑最大流的时候如果一种单词的字典序较小的部分有流量流入就说明读出了这个单词

如果一种单词的字典序较大的部分有流量流入同样也说明读出了这个单词

那么这样就要割掉这条连着两个单词的边产生\(2\)的代价

从右往左读也是同理的

那么问题就是如果ta不给定你读法应该怎么办?

也就是说ta可以正着读也可以反着读

那么就同时连ta正着读的边和反着读的边

但是不连\(S,T\)到ta的边

这样跑一边最大流然后加上回文串就是答案了

代码

#include<map>
#include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
# define ull unsigned long long
const int N = 80 ;
const int M = 3005 ;
const ull Base = 233 ;
const int INF = 1e8 ;
using namespace std ;
inline int read() {
char c = getchar() ; int x = 0 , w = 1 ;
while(c>'9'||c<'0') { if(c=='-') w = -1 ; c = getchar() ; }
while(c>='0'&&c<='9') { x = x*10+c-'0' ; c = getchar() ; }
return x*w ;
} char s[N][N] ;
int n , m , S , T ;
int ans , cnt , num = 1 ;
int dirh[N] , dirl[N] , hea[M] ;
int hnum[N] , lnum[N] , d[M] ;
int hw[2][N][N] , lw[2][N][N] , isbig[M] ;
ull pw[N] , hsh1[N] , hsh2[N] ;
map < ull , int > p , hap ; struct E {
int nxt , to , dis ;
} edge[M * 20] ;
inline void Insert_edge(int from , int to , int dis) {
edge[++num].nxt = hea[from] ; edge[num].to = to ;
edge[num].dis = dis ; hea[from] = num ;
}
inline void add_edge(int u , int v , int w) {
Insert_edge(u , v , w) ;
Insert_edge(v , u , 0) ;
}
inline void Clear() {
ans = 0 ; cnt = 0 ; num = 1 ;
memset(isbig , false , sizeof(isbig)) ;
memset(hea , 0 , sizeof(hea)) ;
memset(hnum , 0 , sizeof(hnum)) ;
memset(lnum , 0 , sizeof(lnum)) ;
p.clear() ; hap.clear() ;
} inline bool Bfs() {
queue < int > q ; q.push(S) ;
memset(d , 0 , sizeof(d)) ; d[S] = 1 ;
while(!q.empty()) {
int u = q.front() ; q.pop() ;
for(int i = hea[u] ; i ; i = edge[i].nxt) {
int v = edge[i].to ;
if(!d[v] && edge[i].dis > 0) {
d[v] = d[u] + 1 ;
q.push(v) ;
}
}
}
return d[T] ;
}
int Dfs(int u , int dis) {
if(u == T || !dis) return dis ;
int Sum = 0 ;
for(int i = hea[u] ; i ; i = edge[i].nxt) {
int v = edge[i].to ;
if(d[v] == d[u] + 1 && edge[i].dis > 0) {
int diss = Dfs(v , min(dis , edge[i].dis)) ;
if(diss > 0) {
edge[i].dis -= diss ; edge[i ^ 1].dis += diss ;
dis -= diss ; Sum += diss ; if(!dis) break ;
}
}
}
if(!Sum) d[u] = -1 ;
return Sum ;
}
inline void dinic() {
while(Bfs())
ans += Dfs(S , INF) ;
}
int main() {
int Case = read() ;
pw[0] = 1 ; for(int i = 1 ; i <= 75 ; i ++) pw[i] = pw[i - 1] * Base ;
while(Case --) {
Clear() ;
n = read() ; m = read() ;
for(int i = 1 ; i <= n ; i ++) dirh[i] = read() ;
for(int i = 1 ; i <= m ; i ++) dirl[i] = read() ;
for(int i = 1 ; i <= n ; i ++) {
scanf("%s",s[i] + 1) ;
for(int j = 1 ; j <= m ; j ++)
hsh1[j] = hsh1[j - 1] * Base + s[i][j] ;
for(int j = m ; j >= 1 ; j --)
hsh2[j] = hsh2[j + 1] * Base + s[i][j] ;
int lst = 1 ; ull tp1 , tp2 ;
for(int j = 1 , l , r ; j <= m ; j ++) {
if(s[i][j] == '_' || j == m) {
l = lst , r = j - 1 ; lst = j + 1 ;
if(s[i][j] != '_') ++ r ; if(l > r) continue ;
tp1 = hsh1[r] - hsh1[l - 1] * pw[r - l + 1] ;
tp2 = hsh2[l] - hsh2[r + 1] * pw[r - l + 1] ;
if(tp1 == tp2) {
if(hap[tp1]) continue ;
++ ans ; hap[tp1] = 1 ; continue ;
}
if(!p[tp1]) {
int vit1 = 1 ;
for(int k = 1 ; k <= r - l + 1 ; k ++) {
if(s[i][l + k - 1] > s[i][r - k + 1]) break ;
else if(s[i][l + k - 1] < s[i][r - k + 1]) {
vit1 = -1 ;
break ;
}
}
p[tp1] = ++ cnt ; isbig[cnt] = vit1 ;
p[tp2] = ++ cnt ; isbig[cnt] = - vit1 ;
}
hw[0][i][++hnum[i]] = p[tp1] ; hw[1][i][hnum[i]] = p[tp2] ;
}
}
}
for(int j = 1 ; j <= m ; j ++) {
for(int i = 1 ; i <= n ; i ++)
hsh1[i] = hsh1[i - 1] * Base + s[i][j] ;
for(int i = n ; i >= 1 ; i --)
hsh2[i] = hsh2[i + 1] * Base + s[i][j] ;
int lst = 1 ; ull tp1 , tp2 ;
for(int i = 1 , l , r ; i <= n ; i ++) {
if(s[i][j] == '_' || i == n) {
l = lst , r = i - 1 ; lst = i + 1 ;
if(s[i][j] != '_') ++ r ; if(l > r) continue ;
tp1 = hsh1[r] - hsh1[l - 1] * pw[r - l + 1] ;
tp2 = hsh2[l] - hsh2[r + 1] * pw[r - l + 1] ;
if(tp1 == tp2) {
if(hap[tp1]) continue ;
++ ans ; hap[tp1] = 1 ; continue ;
}
if(!p[tp1]) {
int vit1 = 1 ;
for(int k = 1 ; k <= r - l + 1 ; k ++) {
if(s[l + k - 1][j] > s[r - k + 1][j]) break ;
else if(s[l + k - 1][j] < s[r - k + 1][j]) {
vit1 = -1 ;
break ;
}
}
p[tp1] = ++ cnt ; isbig[cnt] = vit1 ;
p[tp2] = ++ cnt ; isbig[cnt] = -vit1 ;
}
lw[0][j][++lnum[j]] = p[tp1] ; lw[1][j][lnum[j]] = p[tp2] ;
}
}
}
S = 0 , T = cnt + n + m + 1 ;
for(int i = 1 , x , y ; i <= cnt ; i += 2) {
x = i ; y = i + 1 ;
if(isbig[x] == 1) swap(x , y) ;
add_edge(x , y , 2) ;
}
for(int i = 1 ; i <= n ; i ++) {
if(!hnum[i]) continue ;
if((dirh[i] == 1 && isbig[hw[0][i][1]] < 0) || (dirh[i] == -1 && isbig[hw[1][i][1]] < 0)) {
add_edge(S , cnt + i , INF) ;
for(int j = 1 ; j <= hnum[i] ; j ++) {
if(dirh[i] == 1) add_edge(cnt + i , hw[0][i][j] , INF) ;
else add_edge(cnt + i , hw[1][i][j] , INF) ;
}
}
else if((dirh[i] == 1 && isbig[hw[0][i][1]] > 0) || (dirh[i] == -1 && isbig[hw[1][i][1]] > 0)) {
add_edge(cnt + i , T , INF) ;
for(int j = 1 ; j <= hnum[i] ; j ++) {
if(dirh[i] == 1) add_edge(hw[0][i][j] , cnt + i , INF) ;
else add_edge(hw[1][i][j] , cnt + i , INF) ;
}
}
else {
for(int j = 1 , x , y ; j <= hnum[i] ; j ++) {
x = hw[0][i][j] , y = hw[1][i][j] ;
if(isbig[x] > 0) swap(x , y) ;
add_edge(cnt + i , x , INF) ;
add_edge(y , cnt + i , INF) ;
}
}
}
for(int i = 1 ; i <= m ; i ++) {
if(!lnum[i]) continue ;
if((dirl[i] == 1 && isbig[lw[0][i][1]] <= 0) || (dirl[i] == -1 && isbig[lw[1][i][1]] <= 0)) {
add_edge(S , cnt + n + i , INF) ;
for(int j = 1 ; j <= lnum[i] ; j ++) {
if(dirl[i] == 1) add_edge(cnt + n + i , lw[0][i][j] , INF) ;
else add_edge(cnt + n + i , lw[1][i][j] , INF) ;
}
}
else if((dirl[i] == 1 && isbig[lw[0][i][1]] >= 0) || (dirl[i] == -1 && isbig[lw[1][i][1]] >= 0)) {
add_edge(cnt + n + i , T , INF) ;
for(int j = 1 ; j <= lnum[i] ; j ++) {
if(dirl[i] == 1) add_edge(lw[0][i][j] , cnt + n + i , INF) ;
else add_edge(lw[1][i][j] , cnt + n + i , INF) ;
}
}
else {
for(int j = 1 , x , y ; j <= lnum[i] ; j ++) {
x = lw[0][i][j] , y = lw[1][i][j] ;
if(isbig[x] > 0) swap(x , y) ;
add_edge(cnt + n + i , x , INF) ;
add_edge(y , cnt + n + i , INF) ;
}
}
} dinic() ;
printf("%d\n",ans) ;
}
return 0 ;
}

[SDOI2016]墙上的句子的更多相关文章

  1. luogu P4076 [SDOI2016]墙上的句子

    luogu loj 题意看了我半天(逃 (应该是我语文太差了) 题意是要确定每一行和每一列的看单词的顺序,使得同时正着出现和反着出现在里面的单词数量最少,每行和每列的性质是这一行所有单词反过来的单词要 ...

  2. 【LOJ】#2066. 「SDOI2016」墙上的句子

    题解 我一直也不会网络流--orz 我们分析下这道题,显然和行列没啥关系,就是想给你n + m个串 那么我们对于非回文单词之外的单词,找到两两匹配的反转单词(即使另一个反转单词不会出现也要建出来) 具 ...

  3. Solution -「SDOI 2016」「洛谷 P4076」墙上的句子

    \(\mathcal{Description}\)   Link.   (概括得说不清话了还是去看原题吧 qwq. \(\mathcal{Solution}\)   首先剔除回文串--它们一定对答案产 ...

  4. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  5. [LeetCode] Sentence Screen Fitting 调整屏幕上的句子

    Given a rows x cols screen and a sentence represented by a list of words, find how many times the gi ...

  6. BZOJ4516: [Sdoi2016]生成魔咒 后缀自动机

    #include<iostream> #include<cstdio> #include<cstring> #include<queue> #inclu ...

  7. bzoj-4518 4518: [Sdoi2016]征途(斜率优化dp)

    题目链接: 4518: [Sdoi2016]征途 Description Pine开始了从S地到T地的征途. 从S地到T地的路可以划分成n段,相邻两段路的分界点设有休息站. Pine计划用m天到达T地 ...

  8. vim 长句子中的上下移动

    当一个句子很长的时候,屏幕显示不下,就会分为多行,这个时候,你又想找到中间几行某部分的字母,怎么办?这个时候,先按下一个 g ,在按下 j / k ,就可以实现长句子的上下移动了.

  9. BZOJ 4517: [Sdoi2016]排列计数

    4517: [Sdoi2016]排列计数 Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 911  Solved: 566[Submit][Status ...

随机推荐

  1. mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样

    Mybatis批量更新数据 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批量更新两种方式:1.修改值全部一样 2.修改每条记录值不一样 mybatis批 ...

  2. eclipse工程设置项目jre

    Eclipse 是一个开放源代码的.基于Java的可扩展开发平台.就其本身而言,它只是一个框架和一组服务,用于通过插件组件构建开发环境.当我们导入已经存在项目或者通过svn引入项目时经常出现红叉叉的情 ...

  3. 随机森林、gbdt算法

    http://blog.csdn.net/songzitea/article/details/10035757 http://blog.csdn.net/holybin/article/details ...

  4. office outlook 無法開啟 outlook 視窗

    例如[無法啟動Microsoft Office Outlook.無法開啟Outlook 視窗.] 1.啟動 Outlook 安全模式outlook.exe /safe2.清除並重新產生目前設定檔的功能 ...

  5. 常见Python运行时错误

    1)忘记在 if , elif , else , for , while , class ,def 声明末尾添加 :(导致 “SyntaxError :invalid syntax”) 该错误将发生在 ...

  6. win8系统 如何不显示这台电脑的文件夹

    在win8系统中,默认有下面这种文件夹   只要打开注册表编辑器,找到下面所示的项目,删除所有子文件夹即可(最后剩下一个DelegateFolders不用管) [HKEY_LOCAL_MACHINE\ ...

  7. Hadoop DistributedCache使用案例

    背景 公司数据处理具有两个计算框架,单机框架和MR框架.眼下我已经抽象出一套API interface, 供业务计算开发者使用. 并分别在两个计算框架下实现了API的运行调度.应用开发者有时间须要通过 ...

  8. A Simple Example About Privileged Methods in JavaScript

    Douglas Crockford classified the "class methods" in JavaScript into three types: private, ...

  9. xcode10的那些事

    前言 这里主要介绍一下Xcode10 版本主要更新的内容.随着iOS12的发布,Xcode10已经可以从Mac App Store下载.Xcode10包含了iOS12.watchOS 5.macOS1 ...

  10. BZOJ 2460: [BeiJing2011]元素 线性基

    2460: [BeiJing2011]元素 Description 相传,在远古时期,位于西方大陆的 Magic Land 上,人们已经掌握了用魔法矿石炼制法杖的技术.那时人们就认识到,一个法杖的法力 ...