题目链接

给定一个仅包含小写字母的字符串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周:最大连续字母个数的更多相关文章

  1. map集合修改其中元素 去除Map集合中所有具有相同值的元素 Properties长久保存的流操作 两种用map记录单词或字母个数的方法

    package com.swift.lianxi; import java.util.HashMap; import java.util.Iterator; import java.util.Map; ...

  2. HihoCoder第三周与POJ2406:KMP算法总结

    HihoCoder第三周: 输入 第一行一个整数N,表示测试数据组数. 接下来的N*2行,每两行表示一个测试数据.在每一个测试数据中,第一行为模式串,由不超过10^4个大写字母组成,第二行为原串,由不 ...

  3. 77 找出最大连续自然数个数[Longest Consecutive Sequence in an Unsorted Array]

    [本文链接] http://www.cnblogs.com/hellogiser/p/Longest-Consecutive-Sequence-in-an-Unsorted-Array.html [题 ...

  4. [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 ...

  5. 求字符串空格、数字、字母个数--JAVA基础

    相关内容:charAt()函数 package com.nxl123.www;public class NumString { public static void main(String[] arg ...

  6. 分享一个.NET(C#)按指定字母个数截断英文字符串的方法–提供枚举选项,可保留完整单词

    分享一个.NET(C#)按字母个数截断英文字符串的方法,该方法提供枚举选项.枚举选项包括:可保留完整单词,允许最后一个单词超过最大长度限制,字符串最后跟省略号以及不采取任何操作等,具体示例实现代码如下 ...

  7. hdu4106 区间k覆盖问题(连续m个数,最多选k个数) 最小费用最大流 建图巧妙

    /** 题目:hdu4106 区间k覆盖问题(连续m个数,最多选k个数) 最小费用最大流 建图巧妙 链接:http://acm.hdu.edu.cn/showproblem.php?pid=4106 ...

  8. java基础IO流 复制键盘录入的目录,复制其中的.java文件到指定目录,指定目录中有重名,则改名 对加密文件计算字母个数

    package com.swift.jinji; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; im ...

  9. hihoCoder 第136周 优化延迟(二分答案+手写堆)

    题目1 : 优化延迟 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho编写了一个处理数据包的程序.程序的输入是一个包含N个数据包的序列.每个数据包根据其重要程度不同 ...

随机推荐

  1. Tag Archives: 海明距离

    在前一篇文章 <海量数据相似度计算之simhash和海明距离> 介绍了simhash的原理,大家应该感觉到了算法的魅力.但是随着业务的增长 simhash的数据也会暴增,如果一天100w, ...

  2. Nginx网站常见的跳转配置实例

    相信大家在日常运维工作中如果你用到nginx作为前端反向代理服务器的话,你会对nginx的rewrite又爱又恨,爱它是因为你搞定了它,完成了开发人员的跳转需求后你会觉得很爽,觉得真的很强大,恨它是因 ...

  3. [NL系列] RNN & LSTM 网络结构及应用

    http://www.jianshu.com/p/f3bde26febed/ 这篇是 The Unreasonable Effectiveness of Recurrent Neural Networ ...

  4. thinkphp5在URL地址里隐藏模块名

    新的Thinkphp5的路由功能很强大,完全可以自定义以满足自己的要求   ThinkPHP5.0的路由规则如下:http://serverName/index.php/module/controll ...

  5. (转)Unity3D研究院之Assetbundle的实战(六十三)

    上一篇文章中我们相惜讨论了Assetbundle的原理,如果对原理还不太了解的朋友可以看这一篇文章:Unity3D研究院之Assetbundle的原理(六十一) 本篇文章我们将说说assetbundl ...

  6. border-color: transparent rgb(255, 48, 48) transparent transparent;

    border-color: transparent rgb(255, 48, 48) transparent transparent;

  7. 4、Cocos2dx 3.0游戏开发找小三之Hello World 分析

    尊重开发人员的劳动成果.转载的时候请务必注明出处:http://blog.csdn.net/haomengzhu/article/details/27186557 Hello World 分析 打开新 ...

  8. C#.NET常见问题(FAQ)-如何让listView如何选中一行

    把FullRowSelect设置为True     更多教学视频和资料下载,欢迎关注以下信息: 我的优酷空间: http://i.youku.com/acetaohai123   我的在线论坛: ht ...

  9. 微信小程序 - 日期(起止)选择器组件

    2019-01-03 : 修复了日期day-1,新增了年月日(除去时分秒),删除了不必要的touchmove 新增: column: ""(年月日) 配置: pickerConfi ...

  10. Linux下验证码无法显示,Could not initialize class sun.awt.X1 解决方案

    环境:Oracle Linux 6.4,JDK1.6,Weblogic11g 在通过java.awt生成图片验证码时,提示: Could not initialize class sun.awt.X1 ...