Equivalent Strings CodeForces - 559B

思维,递归

题意:

Cgg特别喜欢收集特别的字符串。这天,lfgg给了cgg两个字符串,字符串A,字符串B,声称这是一对神奇的字符串,他们满足如下两个条件的其中之一:
1、 A与B相等。
2、 如果我们把字符串A分成两个长度相等的字符串A1,A2,并将字符串B分成两个长度相等的字符串B1,B2 然后他们满足以下的其中一项条件:
(1) A1与B1是一对神奇的字符串,并且A2与B2是一对神奇的字符串。
(2) A2与B1是一对神奇的字符串,并且A1与B2是一对神奇的字符串。
Lfgg确信他找到了一对神奇字符串,cgg却并不相信。
希望你能帮帮cgg判断,这对字符串是否是一对神奇字符串。

Input
一共两行,一对字符串。
字符串长度为1到200000且仅包含小写英文字母,保证两个字符串长度一致。

Output
如果这是一对神奇字符串,输出“YES”(不含引号)。否则,输出“NO”(不含引号)。

Examples
Input
aaba
abaa
Output
YES
Input
aabb
abab
Output
NO
Note
在第一组样例中,将第一个字符串拆成aa,ba,将第二个字符串拆成ab,aa,显然aa与aa是一对神奇字符串,ba与ab是一对神奇字符串,因为ba与ab可以被拆分成b,a与a,b。

分析

我们的第一反应肯定是递归搜索,直接暴力做了他!
但是,真的能行吗?
代码如下:

#include<iostream>
#include<string>
#include<cstring>
using namespace std;
string s1;
string s2;

bool solve(int l1,int r1,int l2,int r2) {
    if ( (r1 - l1) == 0)return true;
    int i;
    for (i = 0;i < r1 - l1;i++)
        if (s1[i + l1] != s2[i + l2])break;
    if (i == r1 - l1)return true;
    if ((r1 - l1) & 1)return false;
    int mid1 = (l1 + r1) >> 1;
    int mid2 = (l2 + r2) >> 1;
    if (solve(l1, mid1, l2, mid2) && solve(mid1, r1, mid2, r2))return true;
    if (solve(l1, mid1, mid2, r2) && solve(mid1, r1, l2, mid2))return true;
    return false;
}

int main() {
    ios::sync_with_stdio(0);
    cin >> s1 >> s2;
    if (solve(0,s1.size(),0,s2.size()))cout << "YES\n";
    else cout << "NO\n";
}

其实当我们看到

if (solve(l1, mid1, l2, mid2) && solve(mid1, r1, mid2, r2))return true;
if (solve(l1, mid1, mid2, r2) && solve(mid1, r1, l2, mid2))return true;

时就已经感到有点不妙了。。
每一次处理下去,所需处理的数据量都需要加倍!!!!!!
(可以画个递归树看看)
这会导致我们的数据处理时间复杂度向O(n^2)方向靠拢!!!!
所以这是不行的!!!!!!

下面是正确的思路

我们再次回顾一下我们需要解决的问题:
我们需要知道字符串s1与字符串s2是否等价!
等价有一些十分重要的性质:传递性,反身性,等价性
这里我们利用传递性!
如果字符串s1等价于字符串s3且字符串s2等价于字符串s3那么字符串s1等价于字符串s3
按照这个性质我们可以确定,对于等价的s1与s2,他们一定身处在同一个等价字符串集合中!!!!

那么我们求等价于字符串s1的最小的字符串ss1,求等价与s2的最小字符串ss2
我们看一看ss1是否相等于ss2,就ok了!!!!!

下面给出代码,注意细节

#include<iostream>
#include<string>
#include<cstring>
using namespace std;
string s1;
string s2;

bool solve(int l1,int r1,int l2,int r2) {
    if ( (r1 - l1) == 0)return true;
    int i;
    for (i = 0;i < r1 - l1;i++)
        if (s1[i + l1] != s2[i + l2])break;
    if (i == r1 - l1)return true;
    if ((r1 - l1) & 1)return false;
    int mid1 = (l1 + r1) >> 1;
    int mid2 = (l2 + r2) >> 1;
    if (solve(l1, mid1, l2, mid2) && solve(mid1, r1, mid2, r2))return true;
    if (solve(l1, mid1, mid2, r2) && solve(mid1, r1, l2, mid2))return true;
    return false;
}

int main() {
    ios::sync_with_stdio(0);
    cin >> s1 >> s2;
    if (solve(0,s1.size(),0,s2.size()))cout << "YES\n";
    else cout << "NO\n";
}
全部评论

相关推荐

其实本来打算等lastday的时候再写的,但是现在提笔写下这篇总结完全是出于自己的想法,今天上午自己被学校发的签到吵醒时才突然想明白了很多事情,遂决定写下本文进行总结,虽然现在顶多算2.5个月,但也大差不差喵。回看这段时间的日常实习,我的关键词是:遗憾,焦虑。当然也有快乐的时候,不过大部分时间都是前面这两种情绪主导。为了避免后人再次踩坑,我将在本文详细解释我遇到的困难&nbsp;+&nbsp;产生的原因&nbsp;+&nbsp;应对的措施。同时总结新人实习时除了业务本身,还有如何处理生活与工作上的平衡,调控自身的情绪,让自己恢复到最好的工作状态。本文不会教你实习怎么去做产出,因为有产出的前提是你的心态足够健康,且在工作之余还有时间去...
wuwuwuoow:你的经历跟挺像,但我实力远没你强,现在只能干外包。但解决焦虑这块我应该比你更有经验,因为我曾经也非常迷茫和焦虑: 1.规律作息。无论节假日,都必须在同一时间点睡觉,同一时间点起床。放假睡的多,工作睡的少,这就是典型的作息不规律。将直接干扰前额叶皮层功能,导致情绪波动(易怒、焦虑)。无论上班还是周末,我都是 11:30 睡,7 点起床。7.5h 睡眠,完全足够了。 2.运动。缓解压力,强身健体,提高免疫力。不要觉得每天没有时间锻炼,都是懒惰的借口。 3.冥想。长期练习会增厚前额叶皮层(理性决策区),缩小杏仁核体积(减少情绪过敏反应,核心),增强情绪调控能力。 方法很简单,任何时候都能做。就是闭上眼睛,只专注自己的呼吸,不去想其他任何事情。你可以尝试一下,你会发现非常难只专注呼吸,会有大量的想法涌现出来(什么走马灯),不要去压抑它们,而是放平心态,把注意力继续放在呼吸上面。 而且最重要的是,冥想让你学会“活在当下”。因为处于冥想的你,除了专注呼吸你还能做什么呢?你什么都做不了。生活也是这样,我们无法改变过去,无法预知未来会发生什么,我们能做的只有手头的事情,除此之外什么都别想,因为你无法去改变它们。 4.工作与生活分离。工作不是生活的全部,生活可不是只有工作。像我放假的时候,从不带电脑回去。放假该玩就玩吧。 上面要是都能做到,其实完全解决不了你工作上的问题,完不成的需求还是完不成,面试该挂还是得挂。不过呢,当你再次迷茫,再次焦虑的时候,你会发现,诶,还好,没这么难受。比如面试挂了,可能以前的你会感觉非常难受。但如果你做到以上 4 点,你还是会难受的,但其实又没这么难受,可能你会这样想:既然挂了我还能怎么样?这公司不要我,有的是公司要我!
投递腾讯等公司6个岗位 >
点赞 评论 收藏
分享
勤奋努力的椰子这就开摆:这些经历跟硬件都没啥关系呀
点赞 评论 收藏
分享
评论
1
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务