特殊的二进制序列是具有以下两个性质的二进制序列:
0 的数量与 1 的数量相等。
二进制序列的每一个前缀码中 1 的数量要大于等于 0 的数量。
给定一个特殊的二进制序列 S
,以字符串形式表示。定义一个 操作 为首先选择 S
的两个连续且非空的特殊的子串,然后将它们交换。(两个子串为连续的当且仅当第一个子串的最后一个字符恰好为第二个子串的第一个字符的前一个字符。)
在任意次数的操作之后,交换后的字符串按照字典序排列的最大的结果是什么?
示例 1:
**输入:** S = "11011000"
**输出:** "11100100"
**解释:**
将子串 "10" (在S[1]出现) 和 "1100" (在S[3]出现)进行交换。
这是在进行若干次操作后按字典序排列最大的结果。
说明:
S
的长度不超过 50
。
S
保证为一个满足上述定义的 特殊 的二进制序列。
方法一:分治 前言
对于本题而言,将 1 看成左括号 (',0 看成右括号
)’,那么一个特殊的二进制序列就可以看成一个合法的括号序列。这种「映射」有助于理解题目中的操作,即交换两个相邻且非空的合法括号序列。但为了与题目保持一致,下面的部分仍然使用 1/0 进行叙述。
思路与算法
对于一个特殊序列而言,它一定以 1 开始,以 0 结束。这是因为:
如果给定的字符串是一个「整体」的特殊序列,也就是说,它无法完整地拆分成多个特殊序列,那么它的首位 1 和末位 0 是不可能在任何交换操作中出现的。这里给出首位 1 的证明,末位 0 的证明是类似的:
如果首位 1 可以在交换操作中出现,那么包含它的子串是给定字符串(特殊序列)的一个前缀,同时这个子串也是一个特殊序列。对于字符串中剩余的后缀部分,0 和 1 的数量相等(因为给定字符串和前缀子串的 0 和 1 数量均相等)并且满足「每一个前缀中 1 的数量大于等于 0 的数量」(因为后缀部分的每一个前缀可以映射为给定字符串在同一位置结束的前缀,再扣掉前缀子串,由于前缀子串中 0 和 1 的数量相等,因此扣除后仍然满足要求),那么后缀部分也是一个特殊序列,这就说明给定字符串可以拆分成两个特殊序列,与假设相矛盾。
因此,我们可以把首位 1 和末位 0 直接移除,进一步考虑剩余的字符串。
如果给定的字符串可以拆分成多个特殊序列(这里规定每一个拆分出来的特殊序列都是一个「整体」,不能继续进行拆分),那么我们可以「分别」进一步考虑每一个特殊序列,即把某个特殊序列的首位 1 和末位 0 移除后,递归地进行相同的拆分操作。
在递归返回后,我们可以进行「合并」操作:将所有的特殊序列按照字典序进行降序排序,再拼接起来,就可以得到字典序最大的字符串。由于每一次我们可以交换两个相邻的特殊序列,因此按照冒泡排序的方法,我们可以将这些特殊序列任意进行的排列,也就一定能得到字典序最大的字符串。
细节
在编写代码时,我们可以使用一个计数器,并从头遍历给定的字符串。当我们遇到 1 时计数器加 1,遇到 0 时计数器减 1。当计数器为 0 时,我们就拆分除了一个「整体」的特殊序列。
当递归到的字符串长度小于等于 2 时,说明字符串要么为空,要么为 10,此时字符串就是字典序最大的结果,可以直接返回。
代码
[sol1-C++] 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 class Solution {public : string makeLargestSpecial (string s) { if (s.size () <= 2 ) { return s; } int cnt = 0 , left = 0 ; vector<string> subs; for (int i = 0 ; i < s.size (); ++i) { if (s[i] == '1' ) { ++cnt; } else { --cnt; if (cnt == 0 ) { subs.push_back ("1" + makeLargestSpecial (s.substr (left + 1 , i - left - 1 )) + "0" ); left = i + 1 ; } } } sort (subs.begin (), subs.end (), greater<string>{}); string ans = accumulate (subs.begin (), subs.end (), "" s); return ans; } };
[sol1-Java] 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 class Solution { public String makeLargestSpecial (String s) { if (s.length() <= 2 ) { return s; } int cnt = 0 , left = 0 ; List<String> subs = new ArrayList <String>(); for (int i = 0 ; i < s.length(); ++i) { if (s.charAt(i) == '1' ) { ++cnt; } else { --cnt; if (cnt == 0 ) { subs.add("1" + makeLargestSpecial(s.substring(left + 1 , i)) + "0" ); left = i + 1 ; } } } Collections.sort(subs, (a, b) -> b.compareTo(a)); StringBuilder ans = new StringBuilder (); for (String sub : subs) { ans.append(sub); } return ans.toString(); } }
[sol1-C#] 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 public class Solution { public string MakeLargestSpecial (string s ) { if (s.Length <= 2 ) { return s; } int cnt = 0 , left = 0 ; List<string > subs = new List<string >(); for (int i = 0 ; i < s.Length; ++i) { if (s[i] == '1' ) { ++cnt; } else { --cnt; if (cnt == 0 ) { subs.Add("1" + MakeLargestSpecial(s.Substring(left + 1 , i - left - 1 )) + "0" ); left = i + 1 ; } } } subs.Sort((a, b) => b.CompareTo(a)); StringBuilder ans = new StringBuilder(); foreach (string sub in subs) { ans.Append(sub); } return ans.ToString(); } }
[sol1-Python3] 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 class Solution : def makeLargestSpecial (self, s: str ) -> str : if len (s) <= 2 : return s cnt = left = 0 subs = list () for i, ch in enumerate (s): if ch == "1" : cnt += 1 else : cnt -= 1 if cnt == 0 : subs.append("1" + self.makeLargestSpecial(s[left+1 :i]) + "0" ) left = i + 1 subs.sort(reverse=True ) return "" .join(subs)
[sol1-C] 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 static inline int cmp (const void * pa, const void * pb) { return strcmp (*(char **)pb, *(char **)pa); } char *helper (char *s, int start, int end) { int len = end - start + 1 ; if (len <= 2 ) { char *ans = (char *)malloc (sizeof (char ) * (len + 1 )); strncpy (ans, s + start, len); ans[len] = '\0' ; return ans; } int cnt = 0 , left = start; char **subs = (char **)malloc (sizeof (char *) * len); int subsSize = 0 ; for (int i = start; i <= end; ++i) { if (s[i] == '1' ) { ++cnt; } else { --cnt; if (cnt == 0 ) { char *res = helper(s, left + 1 , i - 1 ); subs[subsSize] = (char *)malloc (sizeof (char ) * (strlen (res) + 3 )); sprintf (subs[subsSize], "%s%s%s" , "1" , res, "0" ); left = i + 1 ; subsSize++; } } } qsort(subs, subsSize, sizeof (char *), cmp); char *ans = (char *)malloc (sizeof (char ) * (len + 1 )); int pos = 0 ; for (int i = 0 ; i < subsSize; i++) { pos += sprintf (ans + pos, "%s" , subs[i]); free (subs[i]); } ans[pos] = '\0' ; return ans; } char * makeLargestSpecial (char * s) { int len = strlen (s); return helper(s, 0 , len - 1 ); }
[sol1-Golang] 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 func makeLargestSpecial (s string ) string { if len (s) <= 2 { return s } subs := sort.StringSlice{} cnt, left := 0 , 0 for i, ch := range s { if ch == '1' { cnt++ } else if cnt--; cnt == 0 { subs = append (subs, "1" +makeLargestSpecial(s[left+1 :i])+"0" ) left = i + 1 } } sort.Sort(sort.Reverse(subs)) return strings.Join(subs, "" ) }
[sol1-JavaScript] 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 var makeLargestSpecial = function (s ) { if (s.length <= 2 ) { return s; } let cnt = 0 , left = 0 ; const subs = []; for (let i = 0 ; i < s.length ; ++i) { if (s[i] === '1' ) { ++cnt; } else { --cnt; if (cnt === 0 ) { subs.push ("1" + makeLargestSpecial (s.substring (left + 1 , i)) + '0' ); left = i + 1 ; } } } subs.sort ().reverse (); return subs.join ('' ); };
复杂度分析
时间复杂度:O(n^2),其中 n 是字符串 s 的长度。在最坏的情况下,s 由 \dfrac{n}{2 个 1 接着 \dfrac{n}{2 个 0 拼接而成,每次递归仅减少 2 的字符串长度,需要进行 \dfrac{n}{2 次递归。同时每次递归需要 O(n) 的时间进行拼接并返回答案,因此总时间复杂度为 O(n^2)。
空间复杂度:O(n),即为递归需要的栈空间以及存储递归返回的字符串需要的临时空间。