特殊的二进制字符串 是具有以下两个性质的二进制序列:
0的数量与1的数量相等。- 二进制序列的每一个前缀码中
1的数量要大于等于0的数量。
给定一个特殊的二进制字符串 s。
一次移动操作包括选择字符串 s 中的两个连续的、非空的、特殊子串,并交换它们。两个字符串是连续的,如果第一个字符串的最后一个字符与第二个字符串的第一个字符的索引相差正好为 1。
返回在字符串上应用任意次操作后可能得到的字典序最大的字符串。
示例 1:
输入: S = "11011000"
输出: "11100100"
解释:
将子串 "10" (在 s[1] 出现) 和 "1100" (在 s[3] 出现)进行交换。
这是在进行若干次操作后按字典序排列最大的结果。
示例 2:
输入:s = "10"
输出:"10"
提示:
1 <= s.length <= 50s[i]为'0'或'1'。s是一个特殊的二进制字符串。
分析:可以将 1 看作 (,将 0 看作 ),问题就转化为了左右括号匹配。
合法括号字符串满足:左右括号的个数相等;每个右括号,都在左边有与之匹配的左括号。这意味着,对于括号字符串的每个前缀,左括号的个数不能低于右括号的个数(否则右括号无法找到配对的左括号)。移动操作可看作选择 s 中的一对相邻的合法括号子串,再交换。
合法括号字符串有两种组合方式:
拼接。如 ()+(())=()(())。多个合法括号字符串拼接在一起,可以得到更长的合法括号字符串。
嵌套。如 (+()()+)=(()())。在合法括号字符串的外层套一对括号,可以得到更长的合法括号字符串。
对于拼接,把 s 拆分成若干个更短的合法括号子串,这些子串都可以做相邻交换。可以把这些子串按照字典序从大到小排序。
对于嵌套,去掉外层的一对括号,问题变成操作子串 [1,n−2] 能得到的最大字典序。这是一个规模更小(长为 n−2)的子问题,可以递归解决。
cpp
class Solution {
public:
string makeLargestSpecial(string s) {
if(s.size()<=2)return s;
vector<string>substring;
int cnt=0,index=0;
for(int i=0;i<s.size();++i)
{
if(s[i]=='1')cnt++;
else if(--cnt==0)
{
substring.push_back("1"+makeLargestSpecial(s.substr(index+1,i-index-1))+"0");
index=i+1;
}
}
sort(substring.begin(),substring.end());
string ret="";
for(int i=substring.size()-1;i>=0;--i)
ret+=substring[i];
return ret;
}
};