题意:给定4个长度为N的字符串( N <= 100000),然后构成一个“中国结”,给定目标串,问能否从某个节点出发走一遍得到目标串,其中不能连续通过3个中心节点,也就是从字符串一个端点转移到其他端点后必须沿着这条字符串走。

分析:对4个字符串正反两面和目标串建立相应hash函数,然后暴力枚举每一个位置出发就可以了,可以有正反两个方向的走法。中间注意一下细节差不多就可以了。

代码:

 #include <bits/stdc++.h>
#pragma comment(linker, "/STACK:102400000,102400000")
#define in freopen("F:\\rootial\\data\\data.txt", "r", stdin);
#define bug(x) printf("Line %d: >>>>>>\n", (x));
#define pb push_back
#define mp make_pair using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef pair<int, int> PII;
typedef vector<pair<int, int > > VII; const ULL seed = ; const int maxn = + ;
char s[][maxn], ta[maxn];
ULL hh[][maxn], tt[maxn], HH[maxn];
void pre()
{
tt[] = ;
for(int i = ; i < maxn; i++)
tt[i] = tt[i-]*seed;
}
int n, m;
VII ans;
bool dfs(int x, int pos, int dir, int len)
{
if(len == )
return true;
int len1 = n-pos+;
int mlen = min(len, len1);
int ok = ;
if(hh[x<<|dir][pos]-hh[x<<|dir][pos+mlen]*tt[mlen] == HH[m-len+]-HH[m-len+mlen+]*tt[mlen])
{
ok = ;
ans.pb(mp(x*n+(dir ? n+-pos : pos), dir));
len -= mlen;
if(len == )
return true;
for(int k = ; k < ; k++)
for(int j = ; j < ; j++)
{
if(k == x && (j^) == dir)
continue;
if(dfs(k, , j, len))
return true;
}
}
if(ok)
ans.pop_back();
return false;
}
int main()
{ pre();
int T;
for(int t(scanf("%d", &T)); t <= T; t++)
{
scanf("%d%d", &n, &m);
for(int i = ; i < ; i++)
scanf("%s", s[i]+);
scanf("%s", ta+);
for(int i = ; i < ; i++)
hh[i][n+] = ;
HH[m+] = ;
for(int i = ; i < ; i++)
{
for(int j = n; j >= ; j--)
{
hh[i<<][j] = hh[i<<][j+]*seed+(s[i][j]-'a');
hh[i<<|][j] = hh[i<<|][j+]*seed+(s[i][n+-j]-'a');
}
}
for(int i = m; i >= ; i--)
{
HH[i] = HH[i+]*seed+(ta[i]-'a');
}
int ok = ;
for(int k = ; k < ; k++)
{
for(int pos = ; pos <= n; pos++)
{
for(int j = ; j < ; j++)
{
ok = ;
ans.clear();
if(dfs(k, pos, j, m))
{
ok = ;
break;
}
if(ok)
break;
}
if(ok)
break;
}
if(ok)
{
break;
}
}
int ll = ;
if(ok)
{
for(int i = ; i < ans.size(); i++)
{
PII x = ans[i];
int st = x.first;
int dir = x.second;
int k = (st-)/n+; if(dir == )
{
for(int j = st; j <= k*n && ll < m; j++, ll++)
{
if(ll)
putchar(' ');
printf("%d", j);
}
}
else
{
for(int j = st; j > (k-)*n && ll < m; j--, ll++)
{
if(ll)
putchar(' ');
printf("%d", j);
}
}
}
cout<<endl;
}
else
{
puts("No solution!");
}
}
return ;
}

ZOJ 3817 Chinese Knot的更多相关文章

  1. 数学+高精度 ZOJ 2313 Chinese Girls' Amusement

    题目传送门 /* 杭电一题(ACM_steps 2.2.4)的升级版,使用到高精度: 这次不是简单的猜出来的了,求的是GCD (n, k) == 1 最大的k(1, n/2): 1. 若n是奇数,则k ...

  2. zoj 2313 Chinese Girls' Amusement 解题报告

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1313 题目意思:有 N 个人(编号依次为1~N)围成一个圆圈,要求求 ...

  3. ASC #1

    开始套题训练,第一套ASC题目,记住不放过每一题,多独立思考. Problem A ZOJ 2313 Chinese Girls' Amusement 循环节 题意:给定n,为圆环长度,求k < ...

  4. 使用MySQL数据库将汉字转换成拼音的一个C语言小程序

    环境: mysql:mysql-5.1.65 centos:centos 6.5 编译命令: gcc -o chinesetopinyin chinesetopinyin.c -L/usr/lib/m ...

  5. A - Chinese Girls' Amusement ZOJ - 2313(大数)

    You must have heard that the Chinese culture is quite different from that of Europe or Russia. So so ...

  6. ZOJ 3817Chinese Knot(The 2014 ACM-ICPC Asia Mudanjiang Regional First Round)

    思路: 将4个串每个串都反向这样得到新的四个串一共8个串,对于母串每个位置检测这个串能不能放进去,hs或者后缀数组都可以.然后dp[i][j]  (0<i<len  0<=j< ...

  7. ZOJ People Counting

    第十三届浙江省大学生程序设计竞赛 I 题, 一道模拟题. ZOJ  3944http://www.icpc.moe/onlinejudge/showProblem.do?problemCode=394 ...

  8. CTRL-Space always toggles Chinese IME (Windows 7、10)

    一.window占用了ctrl+空格的快捷键,影响开发工具的只能提示的使用. 二.解决方式: Go to Start > Type in regedit and start it (打开运行输入 ...

  9. ZOJ 3686 A Simple Tree Problem

    A Simple Tree Problem Time Limit: 3 Seconds      Memory Limit: 65536 KB Given a rooted tree, each no ...

随机推荐

  1. (译文)12个简单(但强大)的JavaScript技巧(一)

    原文连接: 12 Simple (Yet Powerful) JavaScript Tips 我将会介绍和解析12个简单但是强大的JavaScript技巧. 这些技巧所有的JavaScript程序员都 ...

  2. HTML5拖拽功能drag

    1.创建拖拽对象 给需要拖拽的元素设置draggable属性,它有三个值: true:元素可以被拖拽:false:元素不能被拖拽:auto: 浏览器自己判断元素是否能被拖拽. 2.处理拖拽事件当我们拖 ...

  3. ACM——Digital Roots

    http://acm.njupt.edu.cn/acmhome/problemdetail.do?&method=showdetail&id=1028 Digital Roots 时间 ...

  4. Opencv——将摄像头拍摄写成视频文件

    这里主要利用了Opencv打开摄像头的代码,以及写入视频的函数,只是这里要注意的是摄像头好像没有帧率,在cvCreateVideoWriter,时要自己设置 #include"cv.h&qu ...

  5. 对象创建型模式------Singleton(单例模式)

    地址:http://blog.csdn.net/wuzhekai1985/article/details/6665869.仅供自己参考学习. 单例模式:保证一个类仅有一个实例,并提供一个访问它的全局节 ...

  6. ios registerNib: and registerClass:

    先看看apple官网简述: registerNib:forCellWithReuseIdentifier: Register a nib file for use in creating new co ...

  7. OpenJudge/Poj 1088 滑雪

    1.链接地址: bailian.openjudge.cn/practice/1088 http://poj.org/problem?id=1088 2.题目: 总Time Limit: 1000ms ...

  8. OpenJudge/Poj 1159 Palindrome

    1.链接地址: http://bailian.openjudge.cn/practice/1159/ http://poj.org/problem?id=1159 2.题目: Palindrome T ...

  9. [java学习笔记]JDK的安装和环境变量的配置

    1.JDK的下载和安装 jdk(java development kit)是java提供给我们的一套java开发工具,它必运行在JVM(java虚拟机)上,java语言的跨平台性就是利用java运行在 ...

  10. linux系统使用密钥登录设置

    使用密钥登录linux的操作步骤(使用putty): 1.用putty远程登录linux服务器,然后使用puttygen生成密钥,将生成的密钥保存,保存私钥将公钥复制保存到linux服务器的autho ...