“新智认知”杯上海高校程序设计竞赛暨第十七届上海大学程序设计春季联赛----D-CSL的字符串

本文主要是介绍“新智认知”杯上海高校程序设计竞赛暨第十七届上海大学程序设计春季联赛----D-CSL的字符串,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

首先发出题目链接:
链接:https://ac.nowcoder.com/acm/contest/551/D
来源:牛客网

题目如下:
题目内容
描述
所输出的字符串应该是属于原字符串的字序典最小的子串,并且每个字符有且只能重复一次。

我们可以看做从字符串的首位出发,一个个选取符合条件的字符并排序,形成一个满足条件的子串。

我们可以从原字符串的末尾从头往前遍历,直到遍历到新的满足条件的子串的最后一个字符所选取它的的位置的下一个位置为止(子串最后那个字符你从原字符串哪里选取了它,你就从前往后遍历到它的位置的下一个位置为止),每次遍历更新所选择的字符,直到遍历结束,最后选择的字符即为子字符串下一个位置的字符。

对于每次选取的字符(更新选择)应该符合以下条件:

(1)这个字符没选择过,即在新的满足条件的子串中没有出现过(子串每个字符只能出现一次);

(2)这个字符在在原字符串的后面再未出现过(比如字符串zaac,‘z’这个字符在它出现位置之后再没有出现过);

或者

(3)这个字符比当前已选择字符的ASCII码要小或者等于。

其中,条件(1)是首要条件,条件(2)(3)是次要条件(其中一个成立,就可以更新选择)。

对于每次更新选择,我们可以将每次选择的字符进行标记,避免后来的更新选择到了与之前选择相同的字符。


ps:所有可能出现字符ACSII(0x21~0x7E),共94个字符。

举个例子:

比如说字符串cbacbacba

首先遍历一遍字符串,得到次字符串的字符种类cnt

(此情况下cnt=3

下面开始进行cnt次循环遍历

粗体代表目前选择的字符,删除线 代表被标记过的字符)

由于子字符串为空,所以我们第一次遍历原字符串应该是从末尾到首位。

1.第一次遍历:

c b a c b a c b a
(1)选择字符‘a’并标记(字符‘a’未在子串中出现过,也没有选择过(标记过,或者说在字符串此位置之后没有出现过))。

c b a c b a c b a
(2)字符‘b’满足条件,将选择更新为字符‘b’并标记(字符‘b’未在子串中出现过,也没有选择过)。

c b a c b a c b a
(3)字符‘c’满足条件,将选择更新为字符‘c’并标记(字符‘c’未在子串中出现过,也没有选择过)。

c b a c b a c b a
(4)字符‘a’满足条件,将选择更新为字符‘a’并标记(字符‘a’未在子串中出现过,但是比当前选择的字符‘c’要小)。

c b a c b a c b a
(5)字符‘b’不满足条件,不进行更新,进行标记(字符‘b’未在子串中出现过,但是被选择过(标记过,在字符串此位置之后出现过))。

c b a c b a c b a
(6)字符‘c’不满足条件,不进行更新,进行标记(字符‘c’未在子串中出现过,但是被选择过)。

c b a c b a c b a
(7)字符‘a’满足条件,将选择更新为字符‘c’并标记(字符串未在子串中出现过,但是和当前选择的字符‘a’相等)

c b a c b a c b a
(8)字符‘b’不满足条件,不进行更新,进行标记(字符‘b’未在子串中出现过,但是被选择过)

c b a c b a c b a
(9)字符‘c’不满足条件,不进行更新,进行标记(字符‘c’未在子串中出现过,但是被选择过)

第一次遍历完毕,选择字符‘a’为子串的第一个字符,故字符‘a’之前的字符在不能被选入子串,所以下次遍历到字符‘a’下一个位置的字符‘c’为止,并且将所有字符标记全部删除。

2.第二次遍历

c b a c b a c b a
(1)字符‘a’不满足条件,不进行更新,进行标记(字符‘a’在子串中出现过)。

c b a c b a c b a
(2)选择并标记字符‘b’(字符‘b’未在子串中出现过,也没有选择过)。

c b a c b a c b a
(3)字符‘c’满足条件,将选择更新为字符‘c’并标记(字符‘c’未在子串中出现过,也没有选择过)。

c b a c b a c b a
(4)字符‘a’不满足条件,不进行更新,进行标记(字符‘a’在子串中出现过)。

c b a c b a c b a
(5)字符‘b’满足条件,将选择更新为字符‘b’并标记(字符‘b’未在子串中出现过,但是比当前选择的字符‘c’要小)。

c b a c b a c b a
(6)字符‘c’不满足条件,不进行更新,进行标记(字符‘c’未在子串中出现过,但是被选择过)。

第二次遍历完毕,选择字符‘b’为子串的第二个字符,故字符‘b’之前的字符在不能被选入子串,所以下次遍历到字符‘b’下一个位置的字符‘a’为止,并且将所有字符标记全部删除。

3.第三次遍历

c b a c b a c b a
(1)字符‘a’不满足条件,不进行更新,进行标记(字符‘a’在子串中出现过)。

c b a c b a c b a
(2)字符‘b’不满足条件,不进行更新,进行标记(字符‘b’在子串中出现过)。

c b a c b a c b a
(3)字符‘c’满足条件,将选择更新为字符‘c’并标记(字符‘c’未在子串中出现过,也没有选择过)。

c b a c b a c b a
(4)字符‘a’不满足条件,不进行更新,进行标记(字符‘a’在子串中出现过)。

第三次遍历完毕,选择字符‘c’为子串的第三个字符,cnt次循环结束,此时子串拥有所有的cnt的字符,且字序典最小,子串为“abc”。


下面放出代码:

#include <string>
#include <iostream>
#include <string.h>
using namespace std;
string s;//原字符串s
bool check[95]={false},exist[95]={false};
//check数组用来表示字符是否被标记,exist数组用来表示选择的字符是否在子串里出现过。
int len,pos=0,head,dpos,cnt=0;
//len是原字符串的长度,pos是子字符串最后一个字符在原字符串的位置,dpos是pos的下一个位置,用来表示从尾往前遍历的终点。
//head表示当前选择的字符,cnt是原字符串字符的种类。
int main(){cin>>s;len=s.size();int i;for(int i=0;i<len;i++){//计算原字符串字符种类if(!check[s[i]-0x21])cnt++;check[s[i]-0x21]=true;}for(i=1;i<=cnt;i++){//有多少种字符,就遍历多少次memset(check,false,sizeof(check));//每次遍历开始要清楚所有标记head=0x7E;//初始化选择的字符串,更好的实现第(3)个条件int j;for(j=len-1;j>=pos;j--){//从尾到pos进行遍历原字符串if(!exist[s[j]-0x21] && (s[j]<=head || !check[s[j]-0x21])){//if(字符在子字符串未出现过 && (这个字符比当前选择的字符小 || 这个字符没有被选择过(标记过)))check[s[j]-0x21]=true;//标记字符head=s[j];//更新选择的字符dpos=j;//更新选择的字符在原字符串的位置}}pos=dpos+1;//更新下一次遍历的终点exist[head-0x21]=true;//表示这个字符成为子字符串下一个位置的字符printf("%c",head);//输出这个字符}
} 

这篇关于“新智认知”杯上海高校程序设计竞赛暨第十七届上海大学程序设计春季联赛----D-CSL的字符串的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/586585

相关文章

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数

认知杂谈52

今天分享 有人说的一段争议性的话 I I 1拓展人脉很重要** 咱们活在这世上啊,得明白一件事儿,知识、逻辑能力和实战经验虽然重要,但确实都不是最关键的。真正关键的是要懂得怎么和那些手里有资源的人打交道。人脉那可真是一笔无形的大财富呢。你想想看,有时候一个有影响力的人帮你一把,那效果可比你累死累活干一年都强得多。 I I 就比如说,你要是认识个行业里的大牛,他可能给你介绍个特别好的工

C语言程序设计(数据类型、运算符与表达式)

一、C的数据类型 C语言提供的数据类型: 二、常量和变量 2.1常量和符号常量 在程序运行过程中,其值不能被改变的量称为常量。 常量区分为不同的类型: 程序中用#define(预处理器指令)命令行定义变量将代表常量,用一个标识符代表一个常量,称为符合常量。 2.2变量 变量代表内存中具有特定属性的一个存储单元,用来存放数据,在程序运行期间,这些值是可以 改变的。 变

C和指针:字符串

字符串、字符和字节 字符串基础 字符串就是一串零个或多个字符,并且以一个位模式为全0的NUL字节结尾。 字符串长度就是字符串中字符数。 size_t strlen( char const *string ); string为指针常量(const修饰string),指向的string是常量不能修改。size_t是无符号数,定义在stddef.h。 #include <stddef.h>

PHP字符串全排列

方法一: $str = 'abc';$a =str_split($str);perm($a, 0, count($a)-1);function perm(&$ar, $k, $m) {if($k == $m){ echo join('',$ar), PHP_EOL;}else {for($i=$k; $i<=$m; $i++) {swap($ar[$k], $ar[$i]);perm($ar

C语言程序设计(选择结构程序设计)

一、关系运算符和关系表达式 1.1关系运算符及其优先次序 ①<(小于) ②<=(小于或等于) ③>(大于) ④>=(大于或等于 ) ⑤==(等于) ⑥!=(不等于) 说明: 前4个优先级相同,后2个优先级相同,关系运算符的优先级低于算术运算符,关系运算符的优先级高于赋值运算符 1.2关系表达式 用关系运算符将两个表达式(可以是算术表达式或关系表达式,逻辑表达式,赋值表达式,字符

PHP7扩展开发之字符串处理

前言 这次,我们来看看字符串在PHP扩展里面如何处理。 示例代码如下: <?phpfunction str_concat($prefix, $string) {$len = strlen($prefix);$substr = substr($string, 0, $len);if ($substr != $prefix) {return $prefix." ".$string;} else

十一、C语言:字符串函数

目录 一、strlen 二、strcpy 三、strcat  四、strcmp 五、strstr 六、strtok 七、strerror 一、strlen 注意:strlen()函数的返回值是size_t,两个size_t相减仍为无符号数 int main(){char arr[10] = "abc";char brr[10] = "abc123";if (strl

基于SSM+Vue+MySQL的可视化高校公寓管理系统

系统展示 管理员界面 宿管界面 学生界面 系统背景   当前社会各行业领域竞争压力非常大,随着当前时代的信息化,科学化发展,让社会各行业领域都争相使用新的信息技术,对行业内的各种相关数据进行科学化,规范化管理。这样的大环境让那些止步不前,不接受信息改革带来的信息技术的企业随时面临被淘汰,被取代的风险。所以当今,各个行业领域,不管是传统的教育行业