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个数据包的序列.每个数据包根据其重要程度不同 ...
随机推荐
- iOS开发-UITabBarController详解
我们在开发中经常会使用到UITabBarController来布局App应用,使用UITabBarController可以使应用看起来更加的清晰,iOS系统的闹钟程序,ipod程序都是非常好的说明和A ...
- Java去掉Html标签的方法
content = content.replaceAll("\\&[a-zA-Z]{1,10};", "").replaceAll("< ...
- 使用checkstyle来规范你的项目
Checkstyle是什么 自从做了程序员,关于格式化的讨论就不曾中断过,到底什么才是正确的,什么才是错误的,到现在也没有完整的定论.但随着时间发展,渐渐衍生出一套规范出来.没有什么绝对的正确和错误, ...
- Kafka:ZK+Kafka+Spark Streaming集群环境搭建(十五)Spark编写UDF、UDAF、Agg函数
Spark Sql提供了丰富的内置函数让开发者来使用,但实际开发业务场景可能很复杂,内置函数不能够满足业务需求,因此spark sql提供了可扩展的内置函数. UDF:是普通函数,输入一个或多个参数, ...
- ACM~排列组合&&hdu例子
排列组合是数学中的一个分支.在计算机编程方面也有非常多的应用,主要有排列公式和组合公式.错排公式.母函数.Catalan Number(卡特兰数)等. 一.有关组合数学的公式 1.排列公式 P(n ...
- php获取网址
#测试网址: http://localhost/blog/testurl.php?id=5 //获取域名或主机地址 echo $_SERVER['HTTP_HOST']."<br> ...
- spring 判断非空提示断言
org.springframework.util.Assert Assert.notNull(object, "Bean object must not be null");
- MyBatis对于Java对象里的枚举类型处理
平时咱们写程序实体类内或多或少都会有枚举类型属性,方便嘛.但是mybatis里怎么处理他们的增删改查呢? 要求: 插入的时候,会用枚举的定义插入数据库,我们希望在数据库中看到的是数字或者其他东西: 查 ...
- 从C# 2.0新特性到C# 3.5新特性
一.C# 2.0 新特性: 1.泛型 List<MyObject> obj_list=new List(); obj_list.Add(new MyObject()); 2.部分类(par ...
- Mongodb3安装授权
(1) mongodb 官网下载解压包mongodb-win32-x86_64-3.0.7.zip解压释放在d盘,目录为mongodb,接下来手动创建data文件夹和log文件夹分别用于存放数据和日志 ...