
题目描述
一个 「开心字符串」定义为:
- 仅包含小写字母
['a', 'b', 'c']. - 对所有在
1 到 s.length - 1 之间的 i ,满足 s[i] != s[i + 1] (字符串的下标从 1 开始)。
比方说,字符串 "abc","ac","b" 和 "abcbabcbcb" 都是开心字符串,但是 "aa","baa" 和 "ababbc" 都不是开心字符串。
给你两个整数 n 和 k ,你需要将长度为 n 的所有开心字符串按字典序排序。
请你返回排序后的第 k 个开心字符串,如果长度为 n 的开心字符串少于 k 个,那么请你返回 空字符串 。
示例 1:
输入:n = 1, k = 3
输出:"c"
解释:列表 ["a", "b", "c"] 包含了所有长度为 1 的开心字符串。按照字典序排序后第三个字符串为 "c" 。
示例 2:
输入:n = 1, k = 4
输出:""
解释:长度为 1 的开心字符串只有 3 个。
示例 3:
输入:n = 3, k = 9
输出:"cab"
解释:长度为 3 的开心字符串总共有 12 个 ["aba", "abc", "aca", "acb", "bab", "bac", "bca", "bcb", "cab", "cac", "cba", "cbc"] 。第 9 个字符串为 "cab"
示例 4:
输入:n = 2, k = 7
输出:""
示例 5:
输入:n = 10, k = 100
输出:"abacbabacb"
提示:
1 <= n <= 10 1 <= k <= 100
解法
方法一:DFS
我们用一个字符串 \(\textit{s}\) 来记录当前的字符串,初始时为空字符串。然后,我们设计一个函数 \(\text{dfs}\),用来生成所有长度为 \(n\) 的开心字符串。
函数 \(\text{dfs}\) 的具体实现如下:
- 如果当前字符串的长度等于 \(n\),则将当前字符串加入答案数组 \(\textit{ans}\) 中,然后返回;
- 如果答案数组的长度大于等于 \(k\),则直接返回;
- 否则,我们遍历字符集 \(\{a, b, c\}\),对于每个字符 \(c\),如果当前字符串为空,或者当前字符串的最后一个字符不等于 \(c\),则将字符 \(c\) 加入当前字符串,然后递归调用 \(\text{dfs}\),递归结束后,将当前字符串的最后一个字符删除。
最后,我们判断答案数组的长度是否小于 \(k\),如果是,则返回空字符串,否则返回答案数组的第 \(k-1\) 个元素。
时间复杂度 \(O(n \times 2^n)\),空间复杂度 \(O(n)\)。其中 \(n\) 为字符串长度。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18 | class Solution:
def getHappyString(self, n: int, k: int) -> str:
def dfs():
if len(s) == n:
ans.append("".join(s))
return
if len(ans) >= k:
return
for c in "abc":
if not s or s[-1] != c:
s.append(c)
dfs()
s.pop()
ans = []
s = []
dfs()
return "" if len(ans) < k else ans[k - 1]
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29 | class Solution {
private List<String> ans = new ArrayList<>();
private StringBuilder s = new StringBuilder();
private int n, k;
public String getHappyString(int n, int k) {
this.n = n;
this.k = k;
dfs();
return ans.size() < k ? "" : ans.get(k - 1);
}
private void dfs() {
if (s.length() == n) {
ans.add(s.toString());
return;
}
if (ans.size() >= k) {
return;
}
for (char c : "abc".toCharArray()) {
if (s.isEmpty() || s.charAt(s.length() - 1) != c) {
s.append(c);
dfs();
s.deleteCharAt(s.length() - 1);
}
}
}
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25 | class Solution {
public:
string getHappyString(int n, int k) {
vector<string> ans;
string s = "";
auto dfs = [&](this auto&& dfs) -> void {
if (s.size() == n) {
ans.emplace_back(s);
return;
}
if (ans.size() >= k) {
return;
}
for (char c = 'a'; c <= 'c'; ++c) {
if (s.empty() || s.back() != c) {
s.push_back(c);
dfs();
s.pop_back();
}
}
};
dfs();
return ans.size() < k ? "" : ans[k - 1];
}
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28 | func getHappyString(n int, k int) string {
ans := []string{}
var s []byte
var dfs func()
dfs = func() {
if len(s) == n {
ans = append(ans, string(s))
return
}
if len(ans) >= k {
return
}
for c := byte('a'); c <= 'c'; c++ {
if len(s) == 0 || s[len(s)-1] != c {
s = append(s, c)
dfs()
s = s[:len(s)-1]
}
}
}
dfs()
if len(ans) < k {
return ""
}
return ans[k-1]
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22 | function getHappyString(n: number, k: number): string {
const ans: string[] = [];
const s: string[] = [];
const dfs = () => {
if (s.length === n) {
ans.push(s.join(''));
return;
}
if (ans.length >= k) {
return;
}
for (const c of 'abc') {
if (!s.length || s.at(-1)! !== c) {
s.push(c);
dfs();
s.pop();
}
}
};
dfs();
return ans[k - 1] ?? '';
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31 | impl Solution {
pub fn get_happy_string(n: i32, k: i32) -> String {
let mut ans = Vec::new();
let mut s = String::new();
let mut k = k;
fn dfs(n: i32, s: &mut String, ans: &mut Vec<String>, k: &mut i32) {
if s.len() == n as usize {
ans.push(s.clone());
return;
}
if ans.len() >= *k as usize {
return;
}
for c in "abc".chars() {
if s.is_empty() || s.chars().last() != Some(c) {
s.push(c);
dfs(n, s, ans, k);
s.pop();
}
}
}
dfs(n, &mut s, &mut ans, &mut k);
if ans.len() < k as usize {
"".to_string()
} else {
ans[(k - 1) as usize].clone()
}
}
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27 | /**
* @param {number} n
* @param {number} k
* @return {string}
*/
var getHappyString = function (n, k) {
const ans = [];
const s = [];
const dfs = () => {
if (s.length === n) {
ans.push(s.join(''));
return;
}
if (ans.length >= k) {
return;
}
for (const c of 'abc') {
if (!s.length || s.at(-1) !== c) {
s.push(c);
dfs();
s.pop();
}
}
};
dfs();
return ans[k - 1] ?? '';
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26 | public class Solution {
public string GetHappyString(int n, int k) {
List<string> ans = new List<string>();
StringBuilder s = new StringBuilder();
void Dfs() {
if (s.Length == n) {
ans.Add(s.ToString());
return;
}
if (ans.Count >= k) {
return;
}
foreach (char c in "abc") {
if (s.Length == 0 || s[s.Length - 1] != c) {
s.Append(c);
Dfs();
s.Length--;
}
}
}
Dfs();
return ans.Count < k ? "" : ans[k - 1];
}
}
|
方法二:数学
我们可以直接计算出第 \(k\) 个开心字符串是什么,而不需要生成所有的开心字符串。
从长度为 \(n\) 的开心字符串的第一个字符串开始,我们可以计算出每个字符的位置应该是什么。
对于长度为 \(n\) 的开心字符串,第一个字符有 \(3\) 种选择,第二个字符有 \(2\) 种选择(不能和第一个字符相同),第三个字符也有 \(2\) 种选择(不能和第二个字符相同),以此类推,直到第 \(n\) 个字符也有 \(2\) 种选择(不能和第 \(n-1\) 个字符相同)。因此,长度为 \(n\) 的开心字符串的总数为 \(3 \times 2^{n-1}\)。
如果 \(k\) 大于长度为 \(n\) 的开心字符串的总数,那么我们直接返回空字符串。
否则,我们从第一个字符开始,依次计算每个字符的位置应该是什么。对于第 \(i\) 个字符,我们枚举字符集 \(\{a, b, c\}\),如果当前字符串的最后一个字符不等于 \(c\),则计算出剩余的开心字符串的总数,如果 \(k\) 小于等于剩余的开心字符串的总数,那么我们就将字符 \(c\) 加入当前字符串,然后继续计算下一个字符的位置;否则,我们就将 \(k\) 减去剩余的开心字符串的总数,然后继续枚举下一个字符。
时间复杂度 \(O(n)\),空间复杂度 \(O(n)\)。其中 \(n\) 为字符串长度。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 | class Solution:
def getHappyString(self, n: int, k: int) -> str:
if k > 3 * (1 << (n - 1)):
return ""
cs = "abc"
ans = []
for i in range(n):
remain = 1 << (n - i - 1)
for c in cs:
if ans and ans[-1] == c:
continue
if k <= remain:
ans.append(c)
break
k -= remain
return "".join(ans)
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23 | class Solution {
public String getHappyString(int n, int k) {
if (k > 3 * (1 << (n - 1))) {
return "";
}
String cs = "abc";
StringBuilder ans = new StringBuilder();
for (int i = 0; i < n; i++) {
int remain = 1 << (n - i - 1);
for (char c : cs.toCharArray()) {
if (ans.length() > 0 && ans.charAt(ans.length() - 1) == c) {
continue;
}
if (k <= remain) {
ans.append(c);
break;
}
k -= remain;
}
}
return ans.toString();
}
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24 | class Solution {
public:
string getHappyString(int n, int k) {
if (k > 3 * (1 << (n - 1))) {
return "";
}
string cs = "abc";
string ans;
for (int i = 0; i < n; ++i) {
int remain = 1 << (n - i - 1);
for (char c : cs) {
if (!ans.empty() && ans.back() == c) {
continue;
}
if (k <= remain) {
ans.push_back(c);
break;
}
k -= remain;
}
}
return ans;
}
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22 | func getHappyString(n int, k int) string {
if k > 3*(1<<(n-1)) {
return ""
}
cs := "abc"
ans := make([]byte, 0, n)
for i := 0; i < n; i++ {
remain := 1 << (n - i - 1)
for j := 0; j < len(cs); j++ {
c := cs[j]
if len(ans) > 0 && ans[len(ans)-1] == c {
continue
}
if k <= remain {
ans = append(ans, c)
break
}
k -= remain
}
}
return string(ans)
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21 | function getHappyString(n: number, k: number): string {
if (k > 3 * (1 << (n - 1))) {
return '';
}
const cs = 'abc';
const ans: string[] = [];
for (let i = 0; i < n; i++) {
const remain = 1 << (n - i - 1);
for (const c of cs) {
if (ans.at(-1) === c) {
continue;
}
if (k <= remain) {
ans.push(c);
break;
}
k -= remain;
}
}
return ans.join('');
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23 | impl Solution {
pub fn get_happy_string(n: i32, mut k: i32) -> String {
if k > 3 * (1 << (n - 1)) {
return String::new();
}
let cs = ['a', 'b', 'c'];
let mut ans: Vec<char> = Vec::with_capacity(n as usize);
for i in 0..n {
let remain = 1 << (n - i - 1);
for &c in &cs {
if !ans.is_empty() && *ans.last().unwrap() == c {
continue;
}
if k <= remain {
ans.push(c);
break;
}
k -= remain;
}
}
ans.into_iter().collect()
}
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27 | /**
* @param {number} n
* @param {number} k
* @return {string}
*/
var getHappyString = function (n, k) {
if (k > 3 * (1 << (n - 1))) {
return '';
}
const cs = 'abc';
const ans = [];
for (let i = 0; i < n; i++) {
const remain = 1 << (n - i - 1);
for (let j = 0; j < cs.length; j++) {
const c = cs[j];
if (ans.at(-1) === c) {
continue;
}
if (k <= remain) {
ans.push(c);
break;
}
k -= remain;
}
}
return ans.join('');
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23 | public class Solution {
public string GetHappyString(int n, int k) {
if (k > 3 * (1 << (n - 1))) {
return "";
}
string cs = "abc";
var ans = new System.Text.StringBuilder();
for (int i = 0; i < n; i++) {
int remain = 1 << (n - i - 1);
foreach (char c in cs) {
if (ans.Length > 0 && ans[ans.Length - 1] == c) {
continue;
}
if (k <= remain) {
ans.Append(c);
break;
}
k -= remain;
}
}
return ans.ToString();
}
}
|