题目链接

给定一个仅包含小写字母的字符串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. jQuery对象初始化的多种传参数形式

    jQuery对象初始化的传参方式包括:1.$(DOMElement)2.$('<h1>...</h1>'), $('#id'), $('.class') 传入字符串, 这是最常 ...

  2. 最全的spark基础知识解答

    原文:http://www.36dsj.com/archives/61155 一. Spark基础知识 1.Spark是什么? UCBerkeley AMPlab所开源的类HadoopMapReduc ...

  3. 不可不知的Python模块: collections

    原文:http://www.zlovezl.cn/articles/collections-in-python/ Python作为一个“内置电池”的编程语言,标准库里面拥有非常多好用的模块.比如今天想 ...

  4. 大智慧专业财务PFFIN(N,M)函数N的取值一览表

    每股指标 1001 摊薄每股收益 1002 净资产收益率 1003 每股经营活动现金流量 1004 每股净资产 1005 每股资本公积金 1006 每股未分配利润 1007 每股主营收入 1008 扣 ...

  5. "garbage at end of line" on Windows 10

      在windows 10上运行docker-machine scp myvm1 docker-compose.yml myvm1:~的时候报错: "garbage at end of li ...

  6. java多线程分块上传并支持断点续传最新修正完整版本[转]

    package com.test; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.Fi ...

  7. ASP.NET MVC 基于页面的权限管理

    菜单表 namespace AspNetMvcAuthDemo1.Models { public class PermissionItem { public int ID { set; get; } ...

  8. 【Android界面实现】使用PagerTabStrip实现有滑动标签的Viewpager

    在ViewPager这样的能够滑动的控件上,总是有非常多的文章能够做.上次的文章,我们实现了一个自己定义的ViewPager的指示器,这篇文章,我们主要是想利用Android自带的控件,实现一个指示器 ...

  9. uni-app - 支付(app支付、小程序支付、h5(微信端)支付)

    App支付.小程序支付.h5(微信端)支付 APP支付(内置) appPay.js /** * 5+App支付,仅支持支付宝以及微信支付 * * 支付宝Sdk集成,微信sdk未集成 * * @para ...

  10. 微信小程序 - 骨架屏

    骨架屏 - “与其等待网络加载,不如提前给点暗示” 注:不适用复杂交互效果 演示 示例解释以及使用全在index.wxml中,观看需了解组件使用. 示例下载:微信小程序-骨架屏演示