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在美国旅行了相当长的一段时间之后,终于准备要回国啦!而在回国之前,他们准备去超市采购一些当 ...
随机推荐
- oracle 12c 警告日志位置
Oracle 12c环境下查询,alert日志并不在bdump目录下,看到网上和书上都写着可以通过初始化参数background_dump_dest来查看alter日志路径,还说警告日志文件的缺省位置 ...
- HDU 6098 17多校6 Inversion(思维+优化)
Problem Description Give an array A, the index starts from 1.Now we want to know Bi=maxi∤jAj , i≥2. ...
- 【Python】爬虫-2
8. urllib2.urlopen可以接受一个Request对象或者url,(在接受Request对象时候,并以此可以来设置一个URL的headers),urllib.urlopen只接收一个url ...
- sizeof和strlen区别(转)
char str[20]="0123456789"; int a=strlen(str); /*a=10;strlen 计算字符串的长度,以\0'为字符串结束标记. int ...
- TX2-static-dhcp-network
前言 之前是使用私有的镜像而不是按照nvidia官网中JetPack3.2套件的镜像对TX2板子进行刷机,刷机之后发现网络不能正常连接,即使已经连接了有线网口,经过大师的指点才终于明白了其中的一点点奥 ...
- Vue的新启之笔
之前就有接触Vue这一语言,作为一个摊薄饼的我,觉得其基础性的知识体系与其他语言是相通的.且由于贵阳这一城市的地理位置的特殊性,我不得不承认想要从事软件开发这一行业,不精通一门语言不行.因为,任何一家 ...
- 动态开点线段树(陕西师范18k题)---get new skill
思想: 每次开点的时候:左右孩子都开辟新空间 注意懒惰标记tag: 因为会向下传递 提前在值中减去懒惰标记,避免重复计算 链接:https://www.nowcoder.com/acm/ ...
- grep 的学习 正则
grep 命令: grep "name" /path/file_name 从file_name文件中中查找 name 字符 grep -c "name ...
- Atom编辑神器
最近喜欢上了Atom编辑神器,安装就不说了,重点讲配置. 一:软件配置 1.先将欢迎界面去掉,每次打开Atom的时候都会出现,实在是很烦人. 就在欢迎界面里面有个复选框,去掉选中就可以了. 2.让At ...
- 踩坑tomcat8.5的cookie机制
https://www.pomelolee.com/1601.html tomcat升级到8.5版本 发现登录和退出报错,报错日志为下 [http-nio-8080-exec-20] 2016 Aug ...