hihocoder第229周:最大连续字母个数
给定一个仅包含小写字母的字符串s(长度小于1e5),你可以交换任意两个字符的位置,现在允许交换k次,要求交换之后,s中最长的连续相同字符个数尽量多,求这个最长连续区间的长度。
样例
输入
1 :表示k
bababbaa:表示s
输出
4
只需要把s[0]处的b移动到s[3],能够达成长度为4的连续区间。
思路
小写字母只有26种,这是一个重要信息。最后的答案会是哪个小写字母“达成”的呢? 只需要枚举26种小写字母。
最后的答案会是在哪个位置达成的呢?只需要枚举|s|个起始位置。
因为枚举连续区间起始位置的时候,连续区间是谁达成的就已经确定了(显然是由连续区间的第一个字符达成的),所以只需要枚举|s|个起始位置。
当起始位置为beg时,只需要求出连续区间的end来,从beg到end总共需要移动的次数是end-beg+1-从beg到end已经存在了的字符的个数。需要移动的次数需要小于等于k,关键在于寻找满足约束的end,这个过程可以二分实现,从beg到end已经存在的字符个数可以用前缀和数组O(1)实现。
下面代码是错误的
bbaaaabbbbbbbb,这种样例无法通过。
双向扫描一遍才能通过。
import java.util.Scanner;
public class Main {
int[][] dp;
char[] a;
int[] count;
int k;
int needMove(int beg, int end) {
int ch = a[beg] - 'a';
int nowCount = dp[end][ch] - dp[beg][ch] + 1;
int regionLength = end - beg + 1;
int move = regionLength - nowCount;
return move;
}
int maxContinue(int ind) {
//以ind开头移动k次最多能够达成的最大连续个数
int left = ind, right = Math.min(ind + count[a[ind] - 'a'] - 1, a.length - 1);
while (left + 1 < right) {
int mid = (left + right) >> 1;
int move = needMove(ind, mid);
if (move > k) {
right = mid - 1;
} else if (move < k) {
left = mid + 1;
} else {
left = mid;
}
}
int rightMove = needMove(ind, right);
int end = left;
if (rightMove <= k) end = right;
return end - ind + 1;
}
int solve() {
int ans = 0;
for (int i = 0; i < a.length; i++) {
if (i > 0 && a[i] == a[i - 1]) {
continue;
}
ans = Math.max(ans, maxContinue(i));
}
return ans;
}
Main() {
Scanner cin = new Scanner(System.in);
k = cin.nextInt();
a = cin.next().trim().toCharArray();
dp = new int[a.length][27];
for (int i = 0; i < dp[0].length; i++) {
dp[0][i] = 0;
}
dp[0][a[0] - 'a'] = 1;
for (int i = 1; i < a.length; i++) {
System.arraycopy(dp[i - 1], 0, dp[i], 0, dp[0].length);
dp[i][a[i] - 'a'] += 1;
}
count = dp[a.length - 1];
System.out.println(solve());
}
public static void main(String[] args) {
new Main();
}
}
优化:双指针单向移动枚举beg和end
如果一个区间[beg,end]是合法的(移动k次能够达成连续),那么这个区间的子区间也是合法的。这个原理保证了end向后单调移动而不会回溯。
当移动end时,我们只需要判断end能否向后移动(区间[beg+1,end+1]是否合法)。这个问题跟第一种思路中的判断移动次数原理是一样的,此法需要枚举[beg,end]区间上每种字符是否合法,复杂度为|s|*26
。
hihocoder第229周:最大连续字母个数的更多相关文章
- map集合修改其中元素 去除Map集合中所有具有相同值的元素 Properties长久保存的流操作 两种用map记录单词或字母个数的方法
package com.swift.lianxi; import java.util.HashMap; import java.util.Iterator; import java.util.Map; ...
- HihoCoder第三周与POJ2406:KMP算法总结
HihoCoder第三周: 输入 第一行一个整数N,表示测试数据组数. 接下来的N*2行,每两行表示一个测试数据.在每一个测试数据中,第一行为模式串,由不超过10^4个大写字母组成,第二行为原串,由不 ...
- 77 找出最大连续自然数个数[Longest Consecutive Sequence in an Unsorted Array]
[本文链接] http://www.cnblogs.com/hellogiser/p/Longest-Consecutive-Sequence-in-an-Unsorted-Array.html [题 ...
- [ACM_水题] ZOJ 3714 [Java Beans 环中连续m个数最大值]
There are N little kids sitting in a circle, each of them are carrying some java beans in their hand ...
- 求字符串空格、数字、字母个数--JAVA基础
相关内容:charAt()函数 package com.nxl123.www;public class NumString { public static void main(String[] arg ...
- 分享一个.NET(C#)按指定字母个数截断英文字符串的方法–提供枚举选项,可保留完整单词
分享一个.NET(C#)按字母个数截断英文字符串的方法,该方法提供枚举选项.枚举选项包括:可保留完整单词,允许最后一个单词超过最大长度限制,字符串最后跟省略号以及不采取任何操作等,具体示例实现代码如下 ...
- hdu4106 区间k覆盖问题(连续m个数,最多选k个数) 最小费用最大流 建图巧妙
/** 题目:hdu4106 区间k覆盖问题(连续m个数,最多选k个数) 最小费用最大流 建图巧妙 链接:http://acm.hdu.edu.cn/showproblem.php?pid=4106 ...
- java基础IO流 复制键盘录入的目录,复制其中的.java文件到指定目录,指定目录中有重名,则改名 对加密文件计算字母个数
package com.swift.jinji; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; im ...
- hihoCoder 第136周 优化延迟(二分答案+手写堆)
题目1 : 优化延迟 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho编写了一个处理数据包的程序.程序的输入是一个包含N个数据包的序列.每个数据包根据其重要程度不同 ...
随机推荐
- Tag Archives: 海明距离
在前一篇文章 <海量数据相似度计算之simhash和海明距离> 介绍了simhash的原理,大家应该感觉到了算法的魅力.但是随着业务的增长 simhash的数据也会暴增,如果一天100w, ...
- 【Spark】SparkStreaming-输出到Kafka
SparkStreaming-输出到Kafka sparkstreaming output kafka_百度搜索 SparkStreaming采用直连方式(Direct Approach)获取Kafk ...
- 判断checkbox选中的个数
直接看例子吧: shippingAddressList 为一个集合 <c:forEach items="${shippingAddressList }" var=" ...
- 使用tensorflow的lstm网络进行时间序列预测
https://blog.csdn.net/flying_sfeng/article/details/78852816 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog. ...
- python 字符串操作常用函数总结
说明:并不完善,只是记录自己使用到的,没使用到或会用的不会出现在本文. 1.字符串截取 (1)基于索引 s = 'ilovepython' s[0]='i' s[-1] = 'n' (2)取其中一段 ...
- 10gocm->session3->数据备份与恢复
这个实验考查我们当数据库出现问题宕机.数据丢失的情况下怎样挽回我们的损失,怎样在最短时间内恢复我们的数据库服务. RMAN规划 host target库 catalog库 ocm01 ocmdb ...
- Sqlite向MySql导入数据
想把手上的Sqlite数据库导入到MySql,想来应该很简单,结果发现非常麻烦. 1.工具直接导入.试着找了几个软件,都不行.网上有人开发的,但是要收费,也不能用. 2.用各自支持的方式,中转.我用的 ...
- (纪录片)光的故事 BBC Light Fantastic (2004)
简介: 导演: Jeremy Turner主演: Simon Schaffer / Dimitri Andreas ... Al Hazen / Edmund Dehn ... Priest/Old ...
- AngularJS中移动页面滚动穿透解决方案
()] + s[]) >= , preventDefault: false, click: IscrollAndroidBug.click() }); var _ele = document.g ...
- C#.NET常见问题(FAQ)-如何判断某个字符是否为汉字
字符强制转换成int可以判断字符数值大小,在下面所示范围内的就是中文 此外还可以判断是否是数字或者字母,用char.IsLetter和char.IsDigit方法 从先这个范例可以看出,中文也 ...