hiho一下 第165周#1327 : 分隔相同字符
题目要求:
时间限制:10000ms
单点时限:1000ms
内存限制:256MB
描述
给定一个只包含小写字母'a'-'z'的字符串 S ,你需要将 S 中的字符重新排序,使得任意两个相同的字符不连在一起。
如果有多个重排后字符串满足条件,输出字典序最小的一个。
如果不存在满足条件的字符串,输出INVALID。
输入
字符串S。(1 ≤ |S| ≤ 100000)
输出
输出字典序最小的答案或者INVALID。
样例输入
aaabc
样例输出
abaca
《分隔相同字符》题目分析
首先我们需要分析INVALID的充分必要条件。容易看出充要条件是S中某个字符的数目超过了(|S|+1)/2。
换句话说,除非有一个字母数量 过多,否则一定可以有一个合法的重排方案。
例如样例"aaabc",5个字符中有3个'a'。这个例子'a'的数量就在临界值上,我们必须在第1、3、5个位置上都放'a'才能把所有的'a'分隔开。如果'a'的数量再多1个,比如"aaaabc",那就没办法把'a'都分隔开了,结果就是INVALID。
那么,当我们知道目前的S不是INVALID时,我们怎么求字典序最小的重排方案呢? 这里我们可以使用一个贪心策略:我们按照'a'-'z'的顺序枚举第一个字符c,如果除去c之后,S的剩余的字符仍然>不是INVALID,那么我们就把第一个字符定为c。之后我们可以用同样的策略确定第二位、第三位……,只是在确定第二位、第三位……的时候还需要要求当前字符不能与之前定下来的字符相同。
举个例子,假设S="bbbac",我们知道出现最多的'b'一共出现了3次,没超过(|S|+1)/2=3,所以至少存在一种重排方案。这时我们要确定重排之后的第一个字符。由于我们希望字典序最小,所以我们 会先尝试第一个字符是'a'行不行。假设第一个字符是'a',那么剩余的S'="bbbc",这时剩余的S是INVALID,所以我们不能把'a'放在一个字符。
然后我们再尝试把'b'放在第一个字符。这时剩余的S'是"bbac",我们知道"bbac"至少存在一个解,并且这个解不需要第一个字符一定是'b'(因为如果S'的解需要第一个字符一定是某个ch,那么S=S'+ch一定INVALID,大家可以仔细想想)。所以我们可以安全的令第一个字符是'b',一定是字典序最小的选择。
同样的贪心策略处理3个字符后之后,我们就会得到前3个字符是:"bab",这时余下的字符是"bc"。注意由于前一个字符是'b',所以此时字典序的第一选择'b'不能被选(尽管出去'b'之后剩余字符仍 然有解),所以第四个字符选'c'。
最终我们得到答案:"babcb"。
本题的关键就是每一个字符选择时,都需要判断剩余字符有没有解。由于只有'a'-'z'26个字母,判断有没有解可以认为是O(26)=O(1)即常数复杂度的。所以总复杂度是O(|S|)。
c语言代码:
#include<stdio.h>
#include<string.h>
char s[];
int az[]; char work(int x,int y,char a)
{
int i,j,k,l;
j=;
k=y/;
l=y%;
for(i=;i<;i++)
{
if(az[i]>k+l)
{
return ' ';
}else if(az[i]==k+l)
{
if(l==)
{
s[x]=i+'a';
az[i]--;
return i+'a';
}
/*else
{
j=0;
while(az[j]==0 || j+'a'==a)j++;
s[x]=j+'a';
az[j]--;
return j+'a';
}//*/
}
}
j=;
while(az[j]== || j+'a'==a)j++;
s[x]=j+'a';
az[j]--;
return j+'a';
} int main()
{
int i,j,k,l,L;
char c;
while(gets(s)!=NULL)
{
L=l=strlen(s);
memset(az,,sizeof(az));
for(i=;i<l;i++)az[s[i]-'a']++;
//for(j=0;j<26;j++)if(az[j]>0)printf("%c%d ",j+'a',az[j]);
//printf("\n");
k=;
c=' ';
for(i=;i<L;i++)
{
c=work(i,l,c);
l--;
if(c==' ')k=;
//for(j=0;j<26;j++)if(az[j]>0)printf("%c%d ",j+'a',az[j]);
//printf("\n");
}
if(k==)printf("%s\n",s);else printf("INVALID\n");
}
return ;
}
c++代码:
#include<iostream>
#include<cstring> char s[];
int letters[]; using namespace std; char deal_str(int num,int l,char c)
{
int max,i,j,k;
i=l%;
max=(l+)/;
for(k=;k<;k++){
if(letters[k]>max){
return ' ';
}else if(letters[k]==max){
if(i==){
s[num]=k+'a';
letters[k]--;
return k+'a';
}
}
}
j=;
while(letters[j]== || j+'a'==c){
j++;
}
s[num]=j+'a';
letters[j]--;
return j+'a';
} int main(){
int i,n,l,err;
char c;
while(cin>>s){
n=l=strlen(s);
memset(letters, , sizeof(letters));
for(i = ; i < n; ++i){
letters[s[i] - 'a']++;
}
err=;
c=' ';
for(i=;i<n;i++)
{
c=deal_str(i,l,c);
l--;
if(c==' '){
err = ;
}
}
if(err==){
cout<<s<<endl;
}else {
cout<<"INVALID"<<endl;;
}
}
return ;
}
c语言代码:
#include<stdio.h>
#include<string.h>
char s[100010];
int az[27];
char work(int x,int y,char a)
{
int i,j,k,l;
j=0;
k=y/2;
l=y%2;
for(i=0;i<26;i++)
{
if(az[i]>k+l)
{
return ' ';
}else if(az[i]==k+l)
{
if(l==1)
{
s[x]=i+'a';
az[i]--;
return i+'a';
}
/*else
{
j=0;
while(az[j]==0 || j+'a'==a)j++;
s[x]=j+'a';
az[j]--;
return j+'a';
}//*/
}
}
j=0;
while(az[j]==0 || j+'a'==a)j++;
s[x]=j+'a';
az[j]--;
return j+'a';
}
int main()
{
int i,j,k,l,L;
char c;
while(gets(s)!=NULL)
{
L=l=strlen(s);
memset(az,0,sizeof(az));
for(i=0;i<l;i++)az[s[i]-'a']++;
//for(j=0;j<26;j++)if(az[j]>0)printf("%c%d ",j+'a',az[j]);
//printf("\n");
k=0;
c=' ';
for(i=0;i<L;i++)
{
c=work(i,l,c);
l--;
if(c==' ')k=1;
//for(j=0;j<26;j++)if(az[j]>0)printf("%c%d ",j+'a',az[j]);
//printf("\n");
}
if(k==0)printf("%s\n",s);else printf("INVALID\n");
}
return 0;
}
c++代码:
#include<iostream>
#include<cstring>
char s[100010];
int letters[27];
using namespace std;
char deal_str(int num,int l,char c)
{
int max,i,j,k;
i=l%2;
max=(l+1)/2;
for(k=0;k<26;k++){
if(letters[k]>max){
return ' ';
}else if(letters[k]==max){
if(i==1){
s[num]=k+'a';
letters[k]--;
return k+'a';
}
}
}
j=0;
while(letters[j]==0 || j+'a'==c){
j++;
}
s[num]=j+'a';
letters[j]--;
return j+'a';
}
int main(){
int i,n,l,err;
char c;
while(cin>>s){
n=l=strlen(s);
memset(letters, 0, sizeof(letters));
for(i = 0; i < n; ++i){
letters[s[i] - 'a']++;
}
err=0;
c=' ';
for(i=0;i<n;i++)
{
c=deal_str(i,l,c);
l--;
if(c==' '){
err = 1;
}
}
if(err==0){
cout<<s<<endl;
}else {
cout<<"INVALID"<<endl;;
}
}
return 0;
}
hiho一下 第165周#1327 : 分隔相同字符的更多相关文章
- 圆内,求离圆心最远的整数点 hiho一下第111周 Farthest Point
// 圆内,求离圆心最远的整数点 hiho一下第111周 Farthest Point // 思路:直接暴力绝对T // 先确定x范围,每个x范围内,离圆心最远的点一定是y轴两端的点.枚举x的范围,再 ...
- hiho一下 第115周:网络流一•Ford-Fulkerson算法 (Edmond-Karp,Dinic,SAP)
来看一道最大流模板水题,借这道题来学习一下最大流的几个算法. 分别用Edmond-Karp,Dinic ,SAP来实现最大流算法. 从运行结过来看明显SAP+当前弧优化+gap优化速度最快. hi ...
- .NET如何将字符串分隔为字符
.NET如何将字符串分隔为字符 如果这是一道面试题,答案也许非常简单:.ToCharArray(),这基本正确-- 我们以"AB吉
- 【hiho一下第77周】递归-减而治之 (MS面试题:Koch Snowflake)
本题是一道微软面试题,看起来复杂,解出来会发现其实是一个很简单的递归问题,但是这道题的递归思路是很值得我们反复推敲的. 原题为hihocoder第77周的题目. 描述 Koch Snowflake i ...
- hiho一下 第207周
题目1 : The Lastest Time 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 What is latest time you can make with ...
- hiho一下第128周 后缀自动机二·重复旋律5
#1445 : 后缀自动机二·重复旋律5 时间限制:10000ms 单点时限:2000ms 内存限制:512MB 描述 小Hi平时的一大兴趣爱好就是演奏钢琴.我们知道一个音乐旋律被表示为一段数构成的数 ...
- 【hiho一下】第一周 最长回文子串
题目1:最长回文子串 题目原文:http://hihocoder.com/contest/hiho1/problem/1 [题目解读] 题目与 POJ 3974 palindrome 基本同样.求解最 ...
- Solution: 最近公共祖先·一 [hiho一下 第十三周]
题目1 : 最近公共祖先·一 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho最近发现了一个神奇的网站!虽然还不够像58同城那样神奇,但这个网站仍然让小Ho乐在其中 ...
- hiho一下十六周 RMQ-ST算法
RMQ-ST算法 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho在美国旅行了相当长的一段时间之后,终于准备要回国啦!而在回国之前,他们准备去超市采购一些当 ...
随机推荐
- centos /data目录扩容
/data盘被日志撑死了,必须扩容 有一块现成的100G的/dev/sdb盘,但是mount到了/data/test目录下,而且还有应用程序在上面进行读写操作 1.先查看哪些应用程序 在占用磁盘 #f ...
- date简述
Date 定义时间和日期的类 java.util.Date 1s=1000ms; 时间的原点:公元1970年1月1日 00点00分00秒: public class DateDemo { publ ...
- SpringMvc使用FastJson做为json的转换器(注解方式)
在使用XML方式配置项目,使用fastjson做为Json转换器时通常的在XML内添加如下的配置: <mvc:message-converters register-defaults=" ...
- HDU 6106 17多校6 Classes(容斥简单题)
Problem Description The school set up three elective courses, assuming that these courses are A, B, ...
- 1.Windows下使用VisualSVN Server搭建SVN服务器
使用 VisualSVN Server来实现主要的 SVN功能则要比使用原始的 SVN和Apache相配合来实现源代码的 SVN管理简单的多,下面就看看详细的说明. VisualSVN Server的 ...
- async 函数--学习笔记一
含义: ES2017 标准引入了 async 函数,使得异步操作变得更加方便.async 函数是什么?一句话,它就是 Generator 函数的语法糖. 前文有一个 Generator 函数,依次读取 ...
- TrueCrypt 7.1a Hashes
Here are the SHA256, SHA1, and MD5 hashes of all TrueCrypt version 7.1a files. The signature of the ...
- 安卓 dex 通用脱壳技术研究(三)
/* 此为DexHunter实现的主要功能,进行内存dump,将class_def_items中dump出classdef和extra部分 */ void* DumpClass(void *p ...
- Python之路PythonNet,第一篇,网络1
pythonnet 网络1 ARPAnet(互联网雏形)---> 民用 ISO(国际标准化组织)--->网络体系结构标准 OSI模型 OSI : 网络信息传输比较复杂需要很多功能协同 ...
- Unity 3D委托entrust
Unity 3D委托的多种用法 本文提供全流程,中文翻译. Chinar 坚持将简单的生活方式,带给世人!(拥有更好的阅读体验 -- 高分辨率用户请根据需求调整网页缩放比例) Chinar -- 心分 ...