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

开发 前端
给定一个长度为 的由小写字母构成的字符串 。请你构造一个长度为 的由小写字母构成的字符串 。本篇就来回答这个问题。

[[417678]]

题目

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

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

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

要求,字符串 需满足:

  • 字符串 在字典序上大于字符串 。
  • 字符串 的字母集是字符串 的字母集的子集。一个字符串的字母集是指该字符串包含的所有不同字母的集合,例如 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 <iostream> 
  2. #include <cstring> 
  3. #include <algorithm> 
  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 <iostream> 
  2. #include <cstring> 
  3. #include <algorithm> 
  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/

 

责任编辑:姜华 来源: Piper蛋窝
相关推荐

2021-04-29 21:06:49

有序数组算法

2022-01-19 11:39:15

数据治理大数据数据

2018-03-13 16:04:45

Promise执行顺序

2009-08-11 10:12:07

C#算法

2017-11-21 12:15:27

数据库面试题SQL

2024-03-18 13:32:11

2009-08-11 14:59:57

一道面试题C#算法

2009-08-11 15:09:44

一道面试题C#算法

2024-04-28 09:26:40

RustRTTI二进制

2021-03-02 11:29:50

算法算法分析前端

2009-01-08 21:21:45

程序员笔记

2022-07-26 01:11:09

AMD芯片Intel

2011-08-18 09:33:23

2024-10-11 17:09:27

2014-04-29 14:58:24

笔试题微软笔试题

2024-12-30 09:55:00

AI数据模型

2010-01-21 16:18:06

C++语言

2021-04-13 19:05:06

Go闭包面试

2018-03-14 07:42:48

2011-05-23 11:27:32

面试题面试java
点赞
收藏

51CTO技术栈公众号