题目链接:http://hihocoder.com/problemset/problem/1039

题意:给定一个只由{A, B, C}组成的字符串s,长度为n, 故包含n+1个空隙;现要求在某个空隙插入一个来自{A, B, C}的字符,然后按照以下“消除规则”对插入后的字符串进行消除操作,问最多能消掉几个字符(包含插入的一个)。

  消除规则:

  1. 自左至右扫描当前字符串,若字符v从某个位置开始连续出现了2次及以上,则此区间内的v全部消除;

  2. 重复步骤1直至不再有可消除的字符。

思路:模拟,枚举。

  这个题有点像之前数据结构编程作业的“祖玛”一题,同样是会产生连锁反应的“消除规则”,不过那道题是给定一个方案,包含一连串的插入操作,直接用动态链表模拟就好。而这道题要求最佳方案,所以可以枚举所有可能的插入方案,对每个方案都要模拟出消除后的结果。动态申请内存显然不合适,而节点数又固定为n+1,所以我用静态链表来实现。

有两个值得注意的地方:

1. 之前以为可以这样剪枝:在挑选每个位置插入的字符时,只考虑和相邻字符相同的。然而一直WA。。。感谢这篇题解给出的反例 BBBAB http://blog.csdn.net/Lu597203933/article/details/44245411

2. 关于复杂度分析:开始时不敢枚举,总感觉是指数的复杂度,但听了Kirai同学的聚合分析后明白了。简述如下:

  (1)首先有n+1个空隙,每个空隙有3种选择,故共有3*(n+1)种插入方案;

  (2)其次对于每个插入方案,要经历若干趟扫描,而停止扫描的标志为上一趟扫描是否发生过消除;

  具体地,设第 i 趟扫描前字符串长度为 m ,由于一趟扫描过程中指针不回溯,所以一趟扫描花费的时间为线性的,记为t[i] = T(m)。若此趟发生了消除,则字符串的长度必然至少减1,故t[i+1] <= T(m-1)。而初始字符串长度为n+1,故至多经过n趟扫描后,必然达到最终状态。

  至此观察所有趟扫描的时间花费序列t,从t[n]到t[1],呈算术级数,故求和后与末项平方同阶,为T(n2)。

  (3)总的时间复杂度为O(n2 * 3 * (n+1)) = O(n3)

关于代码:Debug的时间太长了,需要多写些模拟题

 #include <cstdio>
#include <cstring>
#include <string>
#include <algorithm>
#include <map>
#define REP(N) for(int i=0; i<(N); i++)
#define REPE(N) for(int i=1; i<=(N); i++)
#define CLEAR(A, X) memset(A, X, sizeof(A))
#define FREAD(FN) freopen((FN), "r", stdin)
#define pb(a) push_back(a)
#define SINT(X) scanf("%d", &(X))
using namespace std;
const int MAX_C = ; struct Node
{
char c;
int next;
}nodes[MAX_C];//静态链表
int n;//节点个数
int head; int T;
char s[MAX_C];
int ans; void reset(){//重置为原始序列
n = ;
for(int i = ; s[i] != '\0'; i++){//[1, n]
nodes[i].next = i+;//来自前驱的引用
nodes[i+].c = s[i];//[0]为头结点
n++;
}
nodes[n].next = -;//末元的后继
} int main()
{
FREAD("1039.txt");
head = ;
SINT(T);
while(T--){
scanf("%s", s);
ans = ;
reset();//初始化原始序列
for(char t = 'A'; t <= 'C'; t++){
nodes[n+].c = t;//待插入的元素统一放在[n+1]
for(int i=; i<=n; i++){//插在i与i+1之间
// if(nodes[i].c != t && nodes[i+1].c != t) continue;//这里不能剪枝!!BBBAB这个可以在第一个B后插入一个A而全部消掉
//printf("insert %c after index %d\n", t, i);
nodes[n+].next = nodes[i].next;
nodes[i].next = n + ;
int cnt = ;
bool updated = ;//这趟扫描是否发生了消除
head = ;
// printf("before disappear\n");
// for(int j = nodes[head].next; j != -1; j = nodes[j].next){
// printf("%c", nodes[j].c);
// }
// printf("\n");
while(updated){//这趟扫描没有发生消除,则退出
updated = ;
int discovered = ;
int pre = head;//cur的前驱
int cur = nodes[head].next;
while(cur != - && nodes[cur].next != -){//一趟自左到右的扫描
int j = nodes[cur].next;
//printf("%c %c\n", nodes[cur].c, nodes[j].c);
while(j != - && nodes[j].c == nodes[cur].c){
updated = ;//发现可消
discovered = ;//发现了以cur为开始可消串
cnt++;
//printf("delete %c\n", nodes[j].c);
nodes[cur].next = nodes[j].next;//删除j
j = nodes[j].next;
}
if(discovered){//发现连续的cur,可消
cnt++;//加上开头的cur
discovered = ;
nodes[pre].next = nodes[cur].next;//删除cur
cur = nodes[pre].next;//新的起点
}else{//未发现,cur只有一个
pre = cur;
cur = nodes[cur].next;
}
// printf("delete one cluster\n");
// for(int j = nodes[head].next; j != -1; j = nodes[j].next){
// printf("%c", nodes[j].c);
// }
// printf("\n");
}
// printf("disappear once\n");
// for(int j = nodes[head].next; j != -1; j = nodes[j].next){
// printf("%c", nodes[j].c);
// }
// printf("\n");
}
nodes[i].next = nodes[n+].next;//删除插入的[n+1]
ans = max(ans, cnt);
//printf("ans this %d\n", cnt);
reset();
}
}
printf("%d\n", ans);
}
return ;
}

【hihocoder 1039 字符串消除】模拟的更多相关文章

  1. hihoCoder 1039:字符消除(字符串处理)

    #1039 : 字符消除 时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消 ...

  2. hihocoder #1039 : 字符消除 ( 字符串处理类 ) 好久之前做的题目,具体的算法代码中阅读吧

    #1039 : 字符消除 时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消 ...

  3. hihoCoder 1039字符消除 (字符串处理)

    http://hihocoder.com/problemset/problem/1039 因为字符串只由3种字母组成,并且插入的字符也只能是这三种字符的其中一个,那么可以考虑枚举这三个字符其中一个字符 ...

  4. [hihoCoder]#1039 : 字符消除

    Description 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消除过程是如下进行的: 1)如果s包含长度超过1的由相同字母组成的子串,那么这些 ...

  5. hihoCoder 1039 字符消除 最详细的解题报告

    题目来源:字符消除 解题思路: 1.在给定字符串中的任意位置插入'A'.'B'.'C'中的任意一个字符,然后计算插入后的字符经过消除后最短的字符串长度: 2.在计算字符消除后最短长度时,智能一遍一遍的 ...

  6. 【hihoCoder】#1039 : 字符消除 by C solution

    #1039 : 字符消除 时间限制:1000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消 ...

  7. ACdream 1188 Read Phone Number (字符串大模拟)

    Read Phone Number Time Limit:1000MS     Memory Limit:64000KB     64bit IO Format:%lld & %llu Sub ...

  8. 【hihoCoder】1039 : 字符消除

    题目:http://hihocoder.com/problemset/problem/1039 给定一个字符串s,只包含'A', 'B', 'C'三种字符 1. 向 s 的任意位置 (包括头和尾) 中 ...

  9. 【HIHOCODER 1039】 字符消除

    链接 问题描述 小Hi最近在玩一个字符消除游戏.给定一个只包含大写字母"ABC"的字符串s,消除过程是如下进行的: 1)如果s包含长度超过1的由相同字母组成的子串,那么这些子串会被 ...

随机推荐

  1. Raid1源代码分析--开篇总述

    前段时间由于一些事情耽搁了,最近将raid1方面的各流程整理了一遍.网上和书上,能找到关于MD下的raid1的文档资料比较少.决定开始写一个系列的关于raid1的博客,之前写过的一篇读流程也会在之后加 ...

  2. testlink 下载地址

    testlink 下载地址 https://sourceforge.net/projects/testlink/files/TestLink%201.9/

  3. Qt Mac 下软件Release 公布dmg

    1.首先当然是用Qt Creator.编译一个Release版本号的软件 注意到编译出来的大小非常小,才420KB,由于一些类库还没包括进去的原因.如今还仅仅能在你本地执行,复制到其它Mac电脑就不能 ...

  4. 页面显示磁盘空间使用情况-Agedu

    下载:http://www.chiark.greenend.org.uk/~sgtatham/agedu/ [root@localhost ~]# tar zxvf agedu-r9723.tar.g ...

  5. Impala 源码分析-FE

    By yhluo 2015年7月29日 Impala 3 Comments Impala 源代码目录结构 SQL 解析 Impala 的 SQL 解析与执行计划生成部分是由 impala-fronte ...

  6. mysql 主从复制配置步骤

    1.准备两台数据库环境,或者单台多实例环境,能否正常启动和登录. 2.配置my.cnf文件,主库配置log-bin和server-id参数,从库配置server-id,不能和主库及其他从库一样,一般不 ...

  7. 针对淡入淡出的定时轮播效果js

    如果不使用jquery的fadeIn和fadeOut的接口和不适用animate情况下,如果要做用js实现淡入淡出轮播效果,我所想到的办法就是使用css3新特性transition(注意好兼容性). ...

  8. html与css的移动端与pc端需要注意的事项

    一个移动端与pc端之间最主要的也就是尺寸问题,苹果与安卓的机型尺寸大小相差甚多,一个尺寸都会影响用户的体验.那么我们来了解一下一些常用的解决方法. 一般在网页中都会在头部有一些这样的代码 <me ...

  9. WebApi2官网学习记录---OData中的查询

    EMD安全 查询语法是基于entity data model(EDM),不是基于底层的model类型,可以从EDM排除一个属性,这样这个属性在client就不能被查询了. 有两种方式可以从EDM中排除 ...

  10. Android Studio ---------------常用快捷键(更新中。。。。。。)

    ##常用快捷键: Ctrl+X(或Y) 删除行 Ctrl+D 复制行 Ctrl+Alt+L 格式化代码 Ctrl + Alt + V 提取变量 Shift+F6 重命名 Ctrl+F12显示当前文件的 ...