目录

1 问题描述

2 解决方案

 


1 问题描述

问题描述
  在年轻的时候,我们故事中的英雄——国王 Copa——他的私人数据并不是完全安全地隐蔽。对他来说是,这不可接受的。因此,他发明了一种密码,好记又难以破解。后来,他才知道这种密码是一个长度为奇数的回文串。

  Copa 害怕忘记密码,所以他决定把密码写在一张纸上。他发现这样保存密码不安全,于是他决定按下述方法加密密码:他选定一个整数 X ,保证 X 不小于 0 ,且 2X 严格小于串长度。然后他把密码分成 3 段,最前面的 X 个字符为一段,最后面的 X 个字符为一段,剩余的字符为一段。不妨把这三段依次称之为 prefix, suffix, middle 。显然, middle 的长度为一个大于 0 的奇数,且 prefix 、 suffix 的长度相等。他加密后的密码即为 A + prefix + B + middle + C + suffix ,其中 A 、 B 、 C 是三个由 Copa 选定的字符串,且都有可能为空, + 表示字符串相连。

  许多年过去了。Copa 昨天找到了当年写下加密后字符串的那张纸。但是,Copa 把原密码、A、B、C 都忘了。现在,他请你找一个尽量长的密码,使得这个密码有可能被当年的 Copa 发明、加密并写下。

输入格式
  输入包含一个只含有小写拉丁字母的字符串,长度在 1 到 10^5 之内。
输出格式
  第一行包含一个整数 k ,表示你找到的原密码分成的 3 个部分中有多少个非空字符串。显然 k in {1, 3} 。接下来 k 行,每行 2 个用空格分开的整数 x_i l_i ,表示这一部分的起始位置和长度。要求输出的 x_i 递增。

  起始位置 x_i 应该在 1 到加密后的字符串长度之间。 l_i 必须是正整数,因为你只要输出非空部分的信息。 middle 的长度必须为奇数。

  如果有多组答案,任意一组即可。提示:你要最大化的是输出的 l_i 的总和,而不是 k 。

样例输入
abacaba
样例输出
1
1 7
样例输入
axbya
样例输出
3
1 1
2 1
5 1
样例输入
xabyczba
样例输出
3
2 2
4 1
7 2
数据规模和约定
  对于 10% 的数据: n <= 10

  对于 30% 的数据: n <= 100

  对于 100% 的数据: n <= 100000

  存在 20% 的数据,输出文件第一行为 1 。


2 解决方案

上述问题,我的解法在蓝桥杯中最终运行评分为40分,原因:运行超时。应该是解法思路没有排除某些情况,导致最终运行超时。这篇文章仅仅用于记录自己解决此问题的思考过程,不算作标答,如果能够对于其他同学解答此题带来一些启发,那也算体现出本文的价值了吧。

首先,说一下我解答本题的思路:

输入一段字符串组成为: A + prefix + B + middle + C + suffix 。现在要求取 prefix + middle + suffix ,由于题目说:他选定一个整数 X ,保证 X 不小于 0 ,且 2X 严格小于串长度。那么X可取0。即当X = 0时,也只有一段——middle。

现在要求取最长的密码串:

(1)首先求取原字符串中的最长回文串长度,用maxLen0表示。

(2)由于suffix在字符串尾部,先将suffix反转,然后使用KMP算法,在原字符串中找到第一次与suffix匹配的字符串,记录suffix长度文len1。然后,求取者两端匹配字符串中间剩余的一段字符串的最大回文串的长度len2。此时,求取的密码长度maxLen = 2*len1 + len2。

在这种情况下,影响maxLen长度的因素就是初始选择的len1长度,len1长度变化,也会导致len2长度的变化。所以,我这里采用1 <= len1 < 原始字符串一半的方法来枚举,求取其中的最大maxLen。个人感觉就是在这里,才导致运算超时...

不多说了,下面贴代码 。

具体代码如下:

package com.liuzhen.systemExe;

import java.util.ArrayList;
import java.util.Scanner; public class Main{
//找出字符串arrayA[i]到arrayA[j]中的最大回文串,并返回其初始位置和长度(PS:此方法称作中心扩展法)
public int[] getMaxReverse(char[] arrayA, int i, int j) {
int[] result = new int[2];
int max = 1;
int begin = i+1; //最大回文串的开始位置,初始化为字符i的位置
int zero = i; //保存i的初始值
for(i = i + 1;i < j;i++) {
int start = i - 1;
int end = i + 1;
int count = 1;
while(start >= zero && end <= j) { //对于此种情况,只返回奇数位回文串
if(arrayA[start] == arrayA[end]) {
start--;
end++;
count = count + 2;
}
else
break;
}
if(count > max) {
max = count;
begin = i - (count-1)/2 + 1;
}
}
result[0] = begin; //最大回文串开始位置
result[1] = max; //最大回文串长度 return result;
}
//此处代码是使用Manacher算法求取最大回文串,但是在蓝桥杯练习系统中评分时,也是运行超时,所以在求最大回文串就采用上面更易理解的方法
// public int[] getManacher(char[] arrayA, int i, int j) {
// int[] result = new int[2];
// ArrayList<Character> listA = new ArrayList<Character>();
// for(int y = i;y <= j;y++) {
// listA.add('#');
// listA.add(arrayA[y]);
// }
// listA.add('#');
// int[] P = new int[listA.size() + 1];
// int mx = 0;
// int d = 0;
// for(int y = 1;y <= listA.size();y++) {
// if(mx > y) {
// P[y] = (P[2 * d - y] > (mx - y) ? (mx - y) : P[2 * d - y]);
// } else {
// P[y] = 1;
// }
//
// while(y + P[y] < listA.size() && y - P[y] >= 0 && listA.get(y + P[y]) == listA.get(y - P[y]))
// P[y]++;
//
// if(mx < y + P[y]) {
// mx = y + P[y];
// d = y;
// }
// }
//
// int max = 0;
// int tempY = 0;
// for(int y = 0;y < P.length;y++) {
// if(P[y] > max) {
// max = P[y];
// tempY = y;
// }
// }
// int begin = (tempY - 1)/2;
// begin = begin - (max - 1) / 2 + i + 1;
// result[0] = begin;
// result[1] = max - 1;
// return result;
// }
//使用KMP模式匹配,计算next函数值
public int[] getNext(char[] arrayB) {
int[] next = new int[arrayB.length + 1];
int j = 0;
for(int i = 1;i < arrayB.length;i++) {
while(j > 0 && arrayB[i] != arrayB[j]) j = next[j];
if(arrayB[i] == arrayB[j]) j++;
next[i+1] = j;
}
return next;
}
//使用KMP算法,找出字符数组arrayB,在arrayA中第一次出现匹配的位置
public int getKmpFirstPosition(char[] arrayA, char[] arrayB) {
int position = -1;
int j = 0;
int[] next = getNext(arrayB);
for(int i = 0;i < arrayA.length;i++) {
while(j > 0 && arrayA[i] != arrayB[j]) j = next[j];
if(arrayA[i] == arrayB[j])
j++;
if(j == arrayB.length) {
position = i - j + 1;
break;
}
}
return position;
} public void printResult(String A) {
//当选定整数X = 0时,输出第一行为1,此时只需在A中直接找到有段最长回文串即可,这时的密码最长
char[] arrayA = A.toCharArray();
int[] result0 = getMaxReverse(arrayA, 0, arrayA.length-1);
int maxLen0 = result0[1]; //获得此时回文串的最大长度
//当X != 0时,最长密码其尾部一定在输入的字符串尾部,即 suffix不为空
int j = arrayA.length - 1;
int maxLen = 0; //用于计算最长密码,初始化为0
int position1 = 0;
int position2 = 0;
int position3 = 0;
int len1 = 0;
int len2 = 0;
for(int lenS = 1;lenS < arrayA.length/2;lenS++) {
char[] tempS = new char[lenS];
for(int a = 0;a < lenS;a++)
tempS[a] = arrayA[j-a];
if(getKmpFirstPosition(arrayA, tempS) == -1) {
continue;
}
int tempPosition1 = getKmpFirstPosition(arrayA, tempS) + 1;
int endPosition1 = tempPosition1+lenS;
int startPosition3 = arrayA.length-lenS; if(startPosition3 <= endPosition1)
continue; int[] result = getMaxReverse(arrayA,tempPosition1+lenS-1,j-lenS);
int tempLen2 = result[1];
int tempPosition2 = result[0]; if(lenS * 2 + tempLen2 > maxLen) {
position1 = tempPosition1;
position2 = tempPosition2;
position3 = j - lenS + 2;
len1 = lenS;
len2 = tempLen2;
maxLen = lenS * 2 + tempLen2;
}
}
if(maxLen0 >= maxLen) {
System.out.println("1");
System.out.println(result0[0]+" "+result0[1]);
} else {
System.out.println("3");
System.out.println(position1+" "+len1);
System.out.println(position2+" "+len2);
System.out.println(position3+" "+len1);
}
} public static void main(String[] args){
Main test = new Main();
Scanner in = new Scanner(System.in);
// System.out.println("请输入一个字符串:");
String A = in.nextLine();
test.printResult(A);
}
}

运行结果:

请输入一个字符串:
asdfsfaaaaa
1
7 5

算法笔记_055:蓝桥杯练习 Tricky and Clever Password (Java)的更多相关文章

  1. 算法笔记_077:蓝桥杯练习 K好数(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 如果一个自然数N的K进制表示中任意的相邻的两位都不是相邻的数字,那么我们就说这个数是K好数.求L位K进制数中K好数的数目.例如K = 4, ...

  2. 算法笔记_061:蓝桥杯练习 字串统计(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 给定一个长度为n的字符串S,还有一个数字L,统计长度大于等于L的出现次数最多的子串(不同的出现可以相交),如果有多个,输出最长的,如果仍然 ...

  3. 算法笔记_123:蓝桥杯第七届省赛(Java语言B组部分习题)试题解答

     目录 1 凑算式 2 方格填数 3 四平方和   1 凑算式 凑算式 B DEF A + --- + ------- = 10 C GHI (如果显示有问题,可以参见[图1.jpg]) 这个算式中A ...

  4. 算法笔记_057:蓝桥杯练习 最大的算式 (Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 题目很简单,给出N个数字,不改变它们的相对位置,在中间加入K个乘号和N-K-1个加号,(括号随便加)使最终结果尽量大.因为乘号和加号一共就 ...

  5. 算法笔记_086:蓝桥杯练习 9-2 文本加密(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 先编写函数EncryptChar,按照下述规则将给定的字符c转化(加密)为新的字符:"A"转化"B" ...

  6. 算法笔记_063:蓝桥杯练习 送分啦(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 这题想得分吗?想,请输出“yes”:不想,请输出“no”. 输出格式 输出包括一行,为“yes”或“no”. 2 解决方案 初步一看,这题 ...

  7. 算法笔记_083:蓝桥杯练习 合并石子(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 在一条直线上有n堆石子,每堆有一定的数量,每次可以将两堆相邻的石子合并,合并后放在两堆的中间位置,合并的费用为两堆石子的总数.求把所有石子 ...

  8. 算法笔记_107:蓝桥杯练习 算法提高 学霸的迷宫(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 学霸抢走了大家的作业,班长为了帮同学们找回作业,决定去找学霸决斗.但学霸为了不要别人打扰,住在一个城堡里,城堡外面是一个二维的格子迷宫,要 ...

  9. 算法笔记_096:蓝桥杯练习 算法提高 求最大值(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 给n个有序整数对ai bi,你需要选择一些整数对 使得所有你选定的数的ai+bi的和最大.并且要求你选定的数对的ai之和非负,bi之和非负 ...

随机推荐

  1. 洛谷P1392 取数 [堆]

    题目传送门 取数 题目描述 在一个n行m列的数阵中,你须在每一行取一个数(共n个数),并将它们相加得到一个和.对于给定的数阵,请你输出和前k小的取数方法. 输入输出格式 输入格式: 第一行,三个数n, ...

  2. 【tomcat】获取访问者真实IP

    1.直接访问tomcat的情况 通过下面这段代码可以获取: String ip = request.getHeader("x-forwarded-for"); if(ip == n ...

  3. 【DFS】【DP】岳麓山上打水

    [vijos1159]岳麓山上打水 描述 今天天气好晴朗,处处好风光,好风光!蝴蝶儿忙啊,蜜蜂也忙,信息组的同学们更加忙.最近,由于XX原因,大家不得不到岳麓山去提水.55555555~,好累啊. 信 ...

  4. 【最小割】BZOJ3438-小M的作物(Rank 2???!!!)(含新款Dinic模板)

    一开始被T掉了之后,才害怕地发现之前写的Dinic基本上都是错的??!!!正确的写在注释里了,注意一下(;3<)馬鹿やろ 一个丧心病狂的优化前后效率对比:

  5. JAVA EE 项目常用知识 之AJAX技术实现select下拉列表联动的两种用法(让你真正理解ajax)

    ajax 下拉列表联动的用法. ajax的定义: AJAX 是一种用于创建快速动态网页的技术. 通过在后台与服务器进行少量数据交换,AJAX 可以使网页实现异步更新.这意味着可以在不重新加载整个网页的 ...

  6. NOI 二分算法练习

    1.NOI 二分法求函数的零点 总时间限制:  1000ms 内存限制:  65536kB 描述 有函数: f(x) = x5 - 15 * x4+ 85 * x3- 225 * x2+ 274 * ...

  7. [转] Spring@Autowired注解与自动装配

    1   配置文件的方法 我们编写spring 框架的代码时候.一直遵循是这样一个规则:所有在spring中注入的bean 都建议定义成私有的域变量.并且要配套写上 get 和 set方法. Boss ...

  8. 使用Python SocketServer快速实现多线程网络服务器

    Python SocketServer使用介绍 1.简介: SocketServer是python的一个网络服务器框架,可以减少开发人员编写网络服务器程序的工作量. SocketServer总共有4个 ...

  9. 一种计算MD5的实现方法

    1.在需要用到加密的地方可以使用.net中的md5相关的类生成md5给文件加密. 2.基本思路: 将文件也好,字符串也好,转成字节数组,再利用.net的md5相关类生成md5相关字符串,再将字符串转成 ...

  10. NAND Flash大容量存储器K9F1G08U的坏块管理方法

    转: http://www.360doc.com/content/11/0915/10/7715138_148381804.shtml 在进行数据存储的时候,我们需要保证数据的完整性,而NAND Fl ...