 重构字符串Java
重构字符串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
# 总结
- 分析出几种情况,然后分别对各个情况实现
 
           
       
    