本文涉及知识点
栈
LeetCode 1096. 花括号展开 II
如果你熟悉 Shell 编程,那么一定了解过花括号展开,它可以用来生成任意字符串。
花括号展开的表达式可以看作一个由 花括号、逗号 和 小写英文字母 组成的字符串,定义下面几条语法规则:
如果只给出单一的元素 x,那么表达式表示的字符串就只有 “x”。R(x) = {x}
例如,表达式 “a” 表示字符串 “a”。
而表达式 “w” 就表示字符串 “w”。
当两个或多个表达式并列,以逗号分隔,我们取这些表达式中元素的并集。R({e_1,e_2,…}) = R(e_1) ∪ R(e_2) ∪ …
例如,表达式 “{a,b,c}” 表示字符串 “a”,“b”,“c”。
而表达式 “{{a,b},{b,c}}” 也可以表示字符串 “a”,“b”,“c”。
要是两个或多个表达式相接,中间没有隔开时,我们从这些表达式中各取一个元素依次连接形成字符串。R(e_1 + e_2) = {a + b for (a, b) in R(e_1) × R(e_2)}
例如,表达式 “{a,b}{c,d}” 表示字符串 “ac”,“ad”,“bc”,“bd”。
表达式之间允许嵌套,单一元素与表达式的连接也是允许的。
例如,表达式 “a{b,c,d}” 表示字符串 “ab”,“ac”,"ad"。
例如,表达式 “a{b,c}{d,e}f{g,h}” 可以表示字符串 “abdfg”, “abdfh”, “abefg”, “abefh”, “acdfg”, “acdfh”, “acefg”, “acefh”。
给出表示基于给定语法规则的表达式 expression,返回它所表示的所有字符串组成的有序列表。
示例 1:
输入:expression = “{a,b}{c,{d,e}}”
输出:[“ac”,“ad”,“ae”,“bc”,“bd”,“be”]
示例 2:
输入:expression = “{{a,z},a{b,c},{ab,z}}”
输出:[“a”,“ab”,“ac”,“z”]
解释:输出中 不应 出现重复的组合结果。
提示:
1 <= expression.length <= 60
expression[i] 由 ‘{’,‘}’,‘,’ 或小写英文字母组成
给出的表达式 expression 用以表示一组基于题目描述中语法构造的字符串
栈
结果不能有重复元素,且有序,故用set。
运算只有两种:逗号表示+,省略表示乘。
为了方便,我们补上乘号。以下三种情况补上*:
}{
{之前是字母
}之后是字母。
代码
核心代码
class Solution {
public:
vector<string> braceExpansionII(string expression) {
string exp;
exp = expression[0];
for (int i = 1; i < expression.length(); i++) {
if ('{' == expression[i]) {
if (('}' == expression[i - 1]) || isalpha(expression[i - 1])) {
exp += '*';
}
}else if (isalpha(expression[i])) {
if ('}' == expression[i - 1]) {
exp += '*';
}
}
exp += expression[i];
}
for (int i = 0; i < exp.length(); i++) {
if ('{' == exp[i]) {
m_staOpe.emplace('(');
}else if (',' == exp[i]) {
m_staOpe.emplace('+');
}
else if ('*' == exp[i]) {
m_staOpe.emplace('*');
}
else if ('}' == exp[i]) {
CalAdd();
}
else {
string strName;
while (isalnum(exp[i])) {
strName += exp[i++];
}
i--;
m_staNum.emplace(set<string>{ strName });
CalMul();
}
}
CalAdd();
return vector<string>(m_staNum.top().begin(), m_staNum.top().end());
}
void CalMul() {
if (m_staOpe.size() && ('*' == m_staOpe.top())) {
auto num2 = m_staNum.top();
m_staNum.pop();
m_staOpe.pop();
set<string> ret;
for (const auto& s1 : m_staNum.top()) {
for (const auto& s2 : num2) {
ret.emplace(s1 + s2);
}
}
m_staNum.top().swap(ret);
}
}
void CalAdd() {
set<string> ret;
while (m_staOpe.size() && ('(' != m_staOpe.top())) {
ret.insert(m_staNum.top().begin(), m_staNum.top().end());
m_staOpe.pop();
m_staNum.pop();
}
m_staNum.top().insert(ret.begin(), ret.end());
if (m_staOpe.size() && ('(' == m_staOpe.top())) {
m_staOpe.pop();
}
CalMul();
}
stack<set<string>> m_staNum;
stack<char> m_staOpe;
};
单元测试
template<class T1,class T2>
void AssertEx(const T1& t1, const T2& t2)
{
Assert::AreEqual(t1 , t2);
}
template<class T>
void AssertEx(const vector<T>& v1, const vector<T>& v2)
{
Assert::AreEqual(v1.size(), v2.size());
for (int i = 0; i < v1.size(); i++)
{
Assert::AreEqual(v1[i], v2[i]);
}
}
template<class T>
void AssertV2(vector<vector<T>> vv1, vector<vector<T>> vv2)
{
sort(vv1.begin(), vv1.end());
sort(vv2.begin(), vv2.end());
Assert::AreEqual(vv1.size(), vv2.size());
for (int i = 0; i < vv1.size(); i++)
{
AssertEx(vv1[i], vv2[i]);
}
}
namespace UnitTest
{
string expression;
TEST_CLASS(UnitTest)
{
public:
TEST_METHOD(TestMethod0)
{
expression = "{a,b}{c,{d,e}}";
auto res = Solution().braceExpansionII(expression);
AssertEx({ "ac","ad","ae","bc","bd","be" },res);
}
TEST_METHOD(TestMethod1)
{
expression = "{{a,z},a{b,c},{ab,z}}";
auto res = Solution().braceExpansionII(expression);
AssertEx({ "a","ab","ac","z" }, res);
}
};
}
扩展阅读
视频课程
有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771
如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176
相关下载
想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653
我想对大家说的话 |
---|
《喜缺全书算法册》以原理、正确性证明、总结为主。 |
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。 |
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。 |
如果程序是一条龙,那算法就是他的是睛 |
测试环境
操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。