zipper题解
-请奆佬们洁身自好,好好打代码从我做起 -
题目大意:
给三个字符串,判断C字符串是否由A B字符串顺序组成,
题意分析:
很容易想到的是,A的长度加上B的长度为C的长度
其实进一步想,这 提供了一个思路:
设f(i,j)为bool数组 ,表示 A的 1~i 部分 和B的 1~j 部分是否能组成 C的1~i+j部分
这也是符合上面的定理的
分析一下A的第 i 项 、B的第 j 项 、C的第 i+j 项 的关系
其实这决定了 什么情况用什么公式(情况很多)
因为如果 结果为Yes,那么 C的第 i+j 项 一定等于A的第 i 项 或 B的第 j 项
因为:
A的第 i 项是A的1~i部分的末尾,
B的第 j 项是B的1~j 部分的末尾,
C的1~ i + j 项是由A的1~i部分 和 B的1~j 部分 组成的,
C的1~ i + j 项的末尾必为 A的1~i部分的末尾 或 B的1~j 部分的末尾
也就是C的第 i+j 项 一定等于A的第 i 项 或 B的第 j 项
但事实是,C的第 i+j 项要么是A的第 i 项 要么是B的第 j 项
也可能,A的第 i 项 =B的第 j 项=C的第 i+j 项,我们要枚举两个情况,有一个为Yes即可
不过我们只是想知道A的 1~i 部分 和B的 1~j 部分是否能组成 C的1~i+j部分
所以我们看 f(i-1,j) (以a结尾)和f(i,j-1) (以b结尾) 即可
好了,根据上述,我们推出
当 C(i+j) = a(i) 且 C(i+j) = b(j )
\]
解析:三个相等时,二者皆有可能,有一个为Yes即可,所以是用或
当 C(i+j) = a(i) 且 C(i+j) ≠ b(j)
\]
解析:只有a和C相等时,只可能是以a结尾
当 C(i+j) ≠ a(i) 且 C(i+j) = b(j)
\]
解析:只有b和C相等时,只可能是以b结尾
当 C(i+j) ≠ a(i) 且 C(i+j) ≠ b(j)
\]
解析:皆不相等,无解
代码详解:
代码与理论的不同之处:
- 由于字符串是从0位开始的,所以调用 a,b,c 时记得位数减1
- 由于是记忆化,我的 f 略有不同——0表示未计算,1表示有解,2表示无解
- 上面问题的延伸——f 数组 mod 2 可转成bool
主函数:
int main() {
    cin >> n;//多组数据
    for (int i = 1; i <= n; i++) {
        memset(f, 0, sizeof(f));//每次都要初始化
        cin >> a >> b >> c;
        lena = a.size();
        lenb = b.size();
        lenc = c.size();//求位数
        if (lena + lenb == lenc && dg(lena, lenb)) {//判断
            cout << "Data set " << i << ":yes" << endl;
        } else {
            cout << "Data set " << i << ":no" << endl;
        }
    }
}
简单,不详细讲解
递归函数:
整体浏览
bool dg(int x, int y) //求f(x,y)
{
    if (f[x][y])
        return f[x][y] % 2;
    if (!x) {
        bool flag = true;
        for (int i = 0; i < y; i++) {
            if (c[i] != b[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
    if (!y) {
        bool flag = true;
        for (int i = 0; i < x; i++) {
            if (c[i] != a[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
    if (c[x + y - 1] == a[x - 1] && c[x + y - 1] == b[y - 1]) {
        f[x][y] = dg(x - 1, y) || dg(x, y - 1);
    } else {
        if (c[x + y - 1] == a[x - 1]) {
            f[x][y] = dg(x - 1, y);
        } else {
            if (c[x + y - 1] == b[y - 1]) {
                f[x][y] = dg(x, y - 1);
            } else {
                return false;
            }
        }
    }
    if (f[x][y] == 0)
        f[x][y] = 2;
    return f[x][y] % 2;
}
记忆化:
    if (f[x][y])
        return f[x][y] % 2;
简单,不讲
边界:
如果一个数为0了,只需比较另外一个数和c直接位对位比较
    if (!x) {//如果a匹配完了
        bool flag = true;
        for (int i = 0; i < y; i++) {
            if (c[i] != b[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
    if (!y) {//如果b匹配完了
        bool flag = true;
        for (int i = 0; i < x; i++) {
            if (c[i] != a[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
公式:
使用大量if,不解释
    if (c[x + y - 1] == a[x - 1] && c[x + y - 1] == b[y - 1]) {//三者相等
        f[x][y] = dg(x - 1, y) || dg(x, y - 1);
    } else {
        if (c[x + y - 1] == a[x - 1]) {//只与a匹配
            f[x][y] = dg(x - 1, y);
        } else {
            if (c[x + y - 1] == b[y - 1]) {//只与b匹配
                f[x][y] = dg(x, y - 1);
            } else {
                return false;//无解
            }
        }
    }
此时 f 可能为0,我们将其转成2,返回即可
上代码:
#include <bits/stdc++.h>
using namespace std;
int n, lena, lenb, lenc;
string a, b, c;
int f[205][205];
bool dg(int x, int y)
{
    if (f[x][y])
        return f[x][y] % 2;
    if (!x) {
        bool flag = true;
        for (int i = 0; i < y; i++) {
            if (c[i] != b[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
    if (!y) {
        bool flag = true;
        for (int i = 0; i < x; i++) {
            if (c[i] != a[i]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
    if (c[x + y - 1] == a[x - 1] && c[x + y - 1] == b[y - 1]) {
        f[x][y] = dg(x - 1, y) || dg(x, y - 1);
    } else {
        if (c[x + y - 1] == a[x - 1]) {
            f[x][y] = dg(x - 1, y);
        } else {
            if (c[x + y - 1] == b[y - 1]) {
                f[x][y] = dg(x, y - 1);
            } else {
                return false;
            }
        }
    }
    if (f[x][y] == 0)
        f[x][y] = 2;
    return f[x][y] % 2;
}
int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) {
        memset(f, 0, sizeof(f));
        cin >> a >> b >> c;
        lena = a.size();
        lenb = b.size();
        lenc = c.size();
        if (lena + lenb == lenc && dg(lena, lenb)) {
            cout << "Data set " << i << ":yes" << endl;
        } else {
            cout << "Data set " << i << ":no" << endl;
        }
    }
}
若有不懂,务必指出!
版权归某邓吖所有,禁止未经作者同意搬运!
zipper题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
		我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ... 
- noip2016十连测题解
		以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ... 
- BZOJ-2561-最小生成树 题解(最小割)
		2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ... 
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
		Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ... 
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
		题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ... 
- 2016ACM青岛区域赛题解
		A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ... 
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
		http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ... 
- 网络流n题 题解
		学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ... 
- CF100965C题解..
		求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ... 
随机推荐
- Oracle 数据库备份实战
			最近公司的客户希望使用oracle数据库,所以我们只好将数据从mysql数据库迁移到oracle数据库,并对oracle数据库制定了一个备份策略,之前虽然对oracle很熟悉,但做备份策略还是第一次, ... 
- Mybatis useGeneratedKeys无法返回主键解决
			1.项目环境--SpringBoot下的SSM+Maven 2.问题出现位置--Dao层和Mapper文件 错误代码如下图: dao层: mapper文件: 错误代码分析: 使用useGenerate ... 
- C++中sort()函数使用介绍
			sort()简介 为什么选择使用sort() 在刷题的时候我们经常会碰到排序的问题,如果我们不使用一些排序的方法那我们只能手撕排序,这样就会浪费一些时间.而且我们还需要根据需要去选择相关的排序方法: ... 
- HTML5打造原生应用——Ionic框架简介与Ionic Hello World
			试了试用Ionic框架打造了两个应用,然后在Google Play上架了. 程序语言答人 教你设计物联网 更有意思的是这是在一周的业余时间内完成的三个应用中的两个,接着让我们看看这个框架如何实现高效地 ... 
- java中为什么接口中的属性和方法都默认为public?
			4)为什么接口中的属性和方法都默认为public?Sun公司当初为什么要把java的接口设计发明成这样? [新手可忽略不影响继续学习]答:如上所述,马克-to-win:既然接口强于抽象类能胜任作为和外 ... 
- java JDK的安装和环境配置(windows10)
			1.下载JDK,安装.http://www.oracle.com/technetwork/java/javase/archive-139210.html 下载地址 2.配置JDK. (右键我的电脑 ... 
- java集合总览
			在编写java程序中,我们最常用的除了八种基本数据类型,String对象外还有一个集合类,在我们的的程序中到处充斥着集合类的身影!java中集合大家族的成员实在是太丰富了,有常用的ArrayList. ... 
- 使用localStorage缓存消息(聊天页面)
			可以在聊天页面的created生命周期里面写如下代码,使得刷新后的页面和之前的是一样的 created(){ alert(11) //在页面加载时读取localStorage里的状态 ... 
- Ncrystal Skill设计
			在使用allegro时一般都会听说过skill,使用合适的Skill会使事情事半功倍.但是现阶段所能看到的个人白嫖的Skill都有一些通病.所以我才开发符合自己操作习惯的Skill. 当前我们所能找的 ... 
- 6.S081-2021-Lab3 Pgtbl学习笔记
			Speed up system calls 根据hints查看kernel/proc.c中的函数proc_pagetable // kernel/proc.c // Create a user pag ... 
