NOIP2002 字串变换
2024-09-30 08:59:27
题二 字串变换 (存盘名: NOIPG2)
[问题描述]:
已知有两个字串 A$, B$ 及一组字串变换的规则(至多6个规则):
A1$ -> B1$
A2$ -> B2$
规则的含义为:在 A$中的子串 A1$ 可以变换为 B1$、A2$ 可以变换为 B2$ …。
例如:A$='abcd' B$='xyz'
变换规则为:
‘abc’->‘xu’ ‘ud’->‘y’ ‘y’->‘yz’
则此时,A$ 可以经过一系列的变换变为 B$,其变换的过程为:
‘abcd’->‘xud’->‘xy’->‘xyz’
共进行了三次变换,使得 A$ 变换为B$。
[输入]:
键盘输人文件名。文件格式如下:
A$ B$
A1$ B1$ \
A2$ B2$? |-> 变换规则
... ... /?
所有字符串长度的上限为 20。
[输出]:
输出至屏幕。格式如下:
若在 10 步(包含 10步)以内能将 A$ 变换为 B$ ,则输出最少的变换步数;否则输出"NO ANSWER!"
[输入输出样例]
b.in:
abcd wyz
abc xu
ud y
y yz
屏幕显示:
3
【思路】
Bfs。
隐式图的搜索,需要注意的是转移的时候状态u中可能有多处与A$匹配,也就是一个A$可以拓展多个点。
【代码】
#include<iostream>
#include<fstream>
#include<cstdlib>
#include<cstring>
#include<queue>
#include<map>
using namespace std; const int maxn = 1000+10;
struct Node{
string s;
int d;
}; string block[maxn];
map<string,int> X;
string a[7],b[7];
string A,B;
int n=0;
int pos[maxn];
void make_pos(string s,string t) {
pos[0]=1; int lens=s.size(),lent=t.size();
for(int i=0;i<=lens-lent;i++)
if(s.substr(i,lent)==t)
pos[pos[0]++]=i;
} void bfs() {
queue<Node> q;
q.push((Node){A,0});
X[A]=1;
while(!q.empty()) {
Node u=q.front(); q.pop();
if(u.s==B) { cout<<u.d; return ; }
for(int r=0;r<n;r++) {
make_pos(u.s,a[r]);
for(int i=1;i<pos[0];i++) {
string s=u.s;
s.replace(pos[i],a[r].size(),b[r]);
if(!X.count(s) && u.d+1<=10) {
X[s]=1;
q.push((Node){s,u.d+1});
}
}
}
}
cout<<"NO ANSWER!";
} int main() {
ios::sync_with_stdio(false);
freopen("NOIPG2.in","r",stdin);
freopen("NOIPG2.out","w",stdout);
cin>>A>>B;
while(cin>>a[n]>>b[n]) n++;
bfs();
return 0;
}
找出所有字符串匹配点也可以用KMP算法,就本题数据而言优化效果不大
int f[maxn];
void get_Fail(string P) {
int m=P.size();
f[0]=f[1]=0;
for(int i=1;i<m;i++) {
int j=f[i];
while(j && P[j]!=P[i]) j=f[j];
f[i+1]=P[i]==P[j]?j+1:0;
}
}
void KMP(string T,string P) {
pos[0]=1;
int n=T.size(),m=P.size();
get_Fail(P);
int j=0;
for(int i=0;i<n;i++) {
while(j && T[i]!=P[j]) j=f[j];
if(T[i]==P[j]) j++;
if(j==m) pos[pos[0]++]=i-m+1,j=0; //i-m+1
}
}
最新文章
- SVD奇异值分解的基本原理和运用
- Python基础篇【第7篇】: 面向对象(1)
- 云服务程序在启动的时候执行Powershell脚本
- IO中同步、异步与阻塞、非阻塞的区别
- oracle创建dblink问题
- UNC path
- 手机APP软件使用说明
- excel中VBA对多个文件的操作
- perl5 第十二章 Perl5中的引用/指针
- AlertDialog的写法
- Linux几个常用的目录结构
- 使用mongo-express管理mongodb数据库
- .NET学习日记【1】
- vue-cli脚手架之package.json
- 一、程序设计与C语言
- 从原理上理解如何由震源机制一个节面的解:strike,dip,rake可以求出另一个节面的解
- laravel 5.3升级5.4
- shell入门基础必备
- 死磕 Fragment 的生命周期
- spring-springMVC-MyBatis整合配置文件
热门文章
- javascript进阶——分离式DOM脚本编程
- CentOS安装mplayer
- nginx中的try_files指令解释
- Nginx+uWSIG+Django+websocket的实现
- 数组有N+M个数字, 数字的范围为1 ... N, 打印重复的元素, 要求O(M + N), 不可以用额外的空间
- Android全部权限详解(manifest.xml)
- iOS序列化与反序列化
- js中构造字符串若放入Grails中gsp的<;g:link>;标签出错
- 基于jQuery的上下左右无缝滚动应用(单行或多行)
- android 小米手机连接到电脑adb无法识别 解决方案