重构字符串Java
文章发布较早,内容可能过时,阅读注意甄别。
# 题目
给定一个字符串 s ,检查是否能重新排布其中的字母,使得两相邻的字符不同。
返回 s 的任意可能的重新排列。若不可行,返回空字符串 "" 。
示例 1:
输入: s = "aab" 输出: "aba" 示例 2:
输入: s = "aaab" 输出: ""
提示:
- 1 <= s.length <= 500
- s 只包含小写字母
# 思路
/**
* 写一个NewChar类,里面包含字母的出现频数,和字母本身。用优先队列PriorityQueue来存储一个一个的NewChar,
* 并自己写一个比较器,通过字母的频数降序排列,即构建一个大顶堆。之后两两输出,输出前两个大的,
* 然后将它们两个对应的count频率-1,再次放入,继续输出……
*
* 这样输出是为了总能有一个字母可以把频率最多的字母隔开,优先队列是为了维持储存NewChar的集合总是可以降序输出。
*
* @param S 定一个字符串S
* @return 若可行,输出任意可行的结果。若不可行,返回空字符串。
*/
优先级队列
PriorityQueue
# 解法
class Solution {
/**
* 写一个NewChar类,里面包含字母的出现频数,和字母本身。用优先队列PriorityQueue来存储一个一个的NewChar,
* 并自己写一个比较器,通过字母的频数降序排列,即构建一个大顶堆。之后两两输出,输出前两个大的,
* 然后将它们两个对应的count频率-1,再次放入,继续输出……
*
* 这样输出是为了总能有一个字母可以把频率最多的字母隔开,优先队列是为了维持储存NewChar的集合总是可以降序输出。
*
* @param S 定一个字符串S
* @return 若可行,输出任意可行的结果。若不可行,返回空字符串。
*/
public String reorganizeString(String S) {
//整理好各个字母对应出现的频率
int[] counts = new int[26];
for (int i = 0; i < S.length(); i++) {
counts[S.charAt(i) - 'a']++;
}
//定义大顶堆规则
PriorityQueue<NewChar> pq = new PriorityQueue<>(26, new Comparator<NewChar>() {
//重写比较规则 (后面对象 - 前面对象)为大顶堆
/**
* 拿过来api说
* @param o1 the first object to be compared.
* @param o2 the second object to be compared.
* @return a negative integer, zero, or a positive integer as the
* first argument is less than, equal to, or greater than the
* second.
*/
@Override
public int compare(NewChar o1, NewChar o2) {
//基于出现频率的比较
//默认是小顶堆,重写为大顶堆
return o2.count - o1.count;
}
});
//构建大顶堆
for (int i = 0; i < 26; i++) {
//判断重构是否可行,counts[i] <= (S.length() + 1) / 2)---某个字母过半就不能重构
if (counts[i] > 0 && counts[i] <= (S.length() + 1) / 2) {
//可以重构,就往大顶堆里面塞对象
pq.add(new NewChar(counts[i], (char) (i + 'a')));
} else if (counts[i] > (S.length() + 1) / 2) {
return "";
}
}
//由大顶堆重构字符串
StringBuilder str = new StringBuilder();
while (pq.size() > 1) {//最后剩下一个字符或者一个不剩,终止
//拿出来频率老大和老二
NewChar c1 = pq.poll();
NewChar c2 = pq.poll();
str.append(c1.letter);
str.append(c2.letter);
if (--c1.count > 0) pq.add(c1);
if (--c2.count > 0) pq.add(c2);
}
//若剩下一个,特殊处理;一个不剩正好,美滋滋
if (pq.size() > 0)
str.append(pq.poll().letter);
return str.toString();
}
/**
* 自己根据数据特点搞个对象
*/
static class NewChar {
int count;//出现的频率
char letter;//字母
NewChar(int count, char letter) {
this.count = count;
this.letter = letter;
}
}
/**
* 创建PriorityQueue的时候一定要写一个比较器Comparator,因为NewChar是自己写的一个类,
* 不写比较器的话程序自己不知道该如何排序,从而会报错:
*
* cannot be cast to java.lang.Comparable at java.util.PriorityQueue.siftUpCom
*/
}
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
# 总结
- 分析出几种情况,然后分别对各个情况实现