一道简单题看y总C++代码风格优于我自己的地方

[[417678]]

题目

原题:AcWing 3805. 环形数组[1]

创新互联专注为客户提供全方位的互联网综合服务,包含不限于成都做网站、网站制作、房山网络推广、小程序开发、房山网络营销、房山企业策划、房山品牌公关、搜索引擎seo、人物专访、企业宣传片、企业代运营等,从售前售中售后,我们都将竭诚为您服务,您的肯定,是我们最大的嘉奖;创新互联为所有大学生创业者提供房山建站搭建服务,24小时服务热线:13518219792,官方网址:www.cdcxhl.com

给定一个长度为 的由小写字母构成的字符串 。

请你构造一个长度为 的由小写字母构成的字符串 。

要求,字符串 需满足:

  • 字符串 在字典序上大于字符串 。
  • 字符串 的字母集是字符串 的字母集的子集。一个字符串的字母集是指该字符串包含的所有不同字母的集合,例如 abadaba 的字母集为 。
  • 字符串 在字典序上尽可能小。

保证答案存在。

输入格式

第一行包含整数 ,表示共有 组测试数据。

每组数据第一行包含两个整数 和 。

第二行包含一个长度为 的字符串表示 。

输出格式

每组数据输出一行满足所有条件的字符串 。

数据范围

  • 前三个测试点满足 。
  • 所有测试点满足 ,。
  • 同一测试点内,所有 的和不超过 ,所有 的和不超过 。

输入样例:

 
 
 
  1. 3 3 
  2. abc 
  3. 3 2 
  4. abc 
  5. 3 3 
  6. ayy 
  7. 2 3 
  8. ba 

输出样例:

 
 
 
  1. aca 
  2. ac 
  3. yaa 
  4. baa 

思路:分情况讨论

  • 当 k 大于 n 时,前 n 位不变,我们让 n 位开始填补出现过的最小字符就行
  • 当 k 小于等于 n 时,我们从原字符串 k - 1 位开始往前找,如果当前字符还有变小的可能,那么就让其变小,寻找停止,输出新字符串

代码

 
 
 
  1. #include  
  2. #include  
  3. #include  
  4. using namespace std; 
  5.  
  6. int n, k; 
  7. bool used[26]; 
  8. string s, t; 
  9.  
  10. string tail() 
  11.     int i = 0; 
  12.     for (; i < 26; ++ i) 
  13.         if (used[i]) break; 
  14.     char a = 'a' + i; 
  15.     string res(k - n, a); 
  16.     return res; 
  17.  
  18. string get() 
  19.     char max_char; 
  20.     for (int i = 25; i >= 0; -- i) 
  21.     { 
  22.         if (used[i]) 
  23.         { 
  24.             max_char = 'a' + i; 
  25.             break; 
  26.         } 
  27.     } 
  28.     char min_char; 
  29.     for (int i = 0; i < 26; ++ i) 
  30.     { 
  31.         if (used[i]) 
  32.         { 
  33.             min_char = 'a' + i; 
  34.             break; 
  35.         } 
  36.     } 
  37.      
  38.     int i = k - 1; 
  39.     for (; i >= 0; -- i) 
  40.     { 
  41.         if (s[i] != max_char) break; 
  42.     } 
  43.      
  44.     string res1 = s.substr(0, i); 
  45.     string res2; 
  46.     for (int j = s[i] - 'a' + 1; j < 26; ++ j) 
  47.     { 
  48.         if (used[j]) 
  49.         { 
  50.             res2 = (char) 'a' + j; 
  51.             break; 
  52.         } 
  53.     } 
  54.     string res3(k - i - 1, min_char); 
  55.  
  56.     return res1 + res2 + res3; 
  57.  
  58. int main() 
  59.     int T; 
  60.     cin >> T; 
  61.     while (T --) 
  62.     { 
  63.         cin >> n >> k; 
  64.         cin >> s; 
  65.         memset(used, 0, sizeof used); 
  66.         for (int i = 0; i < s.size(); ++ i) used[s[i] - 'a'] = true; 
  67.         if (k > n) 
  68.         { 
  69.             t = s + tail(); 
  70.         } 
  71.         else 
  72.         { 
  73.             t = get(); 
  74.         } 
  75.         cout << t << endl; 
  76.     } 

可以看出我的代码思路很清晰,但是写得有一点冗余。

y 总代码

看看 y 总的代码。

 
 
 
  1. #include  
  2. #include  
  3. #include  
  4.  
  5. using namespace std; 
  6.  
  7. const int N = 100010; 
  8.  
  9. int n, k; 
  10. char s1[N], s2[N]; 
  11. bool st[26]; 
  12.  
  13. char get_min() 
  14.     for (int i = 0; i < 26; i ++ ) 
  15.         if (st[i]) 
  16.             return i + 'a'; 
  17.     return -1; 
  18.  
  19. char get_next(int t) 
  20.     for (int i = t + 1; i < 26; i ++ ) 
  21.         if (st[i]) 
  22.             return i + 'a'; 
  23.     return -1; 
  24.  
  25. int main() 
  26.     int T; 
  27.     scanf("%d", &T); 
  28.     while (T -- ) 
  29.     { 
  30.         scanf("%d%d", &n, &k); 
  31.         scanf("%s", s1); 
  32.         memset(st, 0, sizeof st); 
  33.         for (int i = 0; i < n; i ++ ) st[s1[i] - 'a'] = true; 
  34.         if (k > n) 
  35.         { 
  36.             printf("%s", s1); 
  37.             char c = get_min(); 
  38.             for (int i = n; i < k; i ++ ) printf("%c", c); 
  39.             puts(""); 
  40.         } 
  41.         else 
  42.         { 
  43.             s2[k] = 0; 
  44.             for (int i = k - 1; i >= 0; i -- ) 
  45.             { 
  46.                 char c = get_next(s1[i] - 'a'); 
  47.                 if (c != -1) 
  48.                 { 
  49.                     s2[i] = c; 
  50.                     for (int j = 0; j < i; j ++ ) s2[j] = s1[j]; 
  51.                     break; 
  52.                 } 
  53.                 s2[i] = get_min(); 
  54.             } 
  55.             puts(s2); 
  56.         } 
  57.     } 
  58.  
  59.     return 0; 
  60.  
  61. // 作者:yxc 
  62. // 链接:https://www.acwing.com/activity/content/code/content/1634481/ 
  63. // 来源:AcWing 
  64. // 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。 

很简洁。

经验:

  • char s[]; puts(s); 中, puts 遇到 \0 注意是 char s[k] = 0 而不是 char s[k] = '0' 字符串停止输出。

参考资料

[1]AcWing 3805. 环形数组:

https://www.acwing.com/activity/content/problem/content/5457/

 

网站题目:一道简单题看y总C++代码风格优于我自己的地方
当前URL:http://www.csdahua.cn/qtweb/news11/121361.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网