删除额外的括号
remove extra parenthesis
问题:
从字符串中删除额外的括号。
例如
((a+b))*c => (a+b)*c
(a+b)+c => (a+b)+c
((a+b)/(c+d)) => ((a+b)/(c+d))
(a+(((b-c)))*d) => (a+(b-c)*d) and so on.
我提出了以下解决方案:
方法:我通过字符串扫描,并记住(使用地图)开口括号的索引以及是否额外(默认情况下它是额外的)。如果我找到一个闭合括号,我会从MAP中检查相应的打开括号,如果是额外的,则两者都会删除。
void removeExtraParentheses(string& S){
map<int, bool> pmap;
for(int i = 0; i < S.size(); i++){
map<int, bool>::iterator it;
if(S.at(i) == '('){
pmap[i] = true;
}
else if(S.at(i) == ')'){
it = pmap.end();
it--;
if(!(*it).second){
pmap.erase(it);
}
else{
S.erase(S.begin() + i);
S.erase(S.begin() + (*it).first);
pmap.erase(it);
i = i - 2;
}
}
else{
if(!pmap.empty()){
it = pmap.end();
it--;
(*it).second= false;
}
}
}
}
时间复杂性:O(n2)
空间:o(n)
我对解决方案不太满意,因为我正在使用额外的存储空间并在二次时间内进行。
我们可以在o(n)时间和o(1)空间中执行此操作吗?如果不是,最好的办法可以做什么?
构建一个表达树,然后以最低限度再生表达式括号。原始的任何括号都不在生成是不必要的。
一个简单,几乎正确的解决方案是将优先级分配给每个操作员。然后,您需要任何直接节点的括号在您正在使用的节点下的是一个操作员,其具有较低的操作员比节点的优先级;例如,如果您在*
(乘法)节点,两个子节点之一是+
(加法)节点。加上一些逻辑要处理左或右装订:如果您在+
节点上,右手节点也是+
节点,您需要括号。
这只是部分正确的,因为C 实际上无法真正映射到优先语法:其中一些想到类型的铸造构造或三元操作员。在至少在三元操作员的情况下,特殊处理这不是那么困难。
编辑:
关于您的big-o问题:这显然不是O(1)空间,因为您需要在内存中构造整个表达式。我不要以为可以进行内存的o(1),因为有可能,您可以有无限的嵌套,您无法判断括号是否是直到不确定的时间之后或不需要。我实际上不是分析了它,但我认为及时是o(n)。上限节点的数量是n
(字符串的长度),您需要访问每个节点一次。
或多或少在网络上找到...
给定输入:((a b)*c)预期输出:( a b)*c
假设:
- peek(队列)只是在没有删除的情况下告诉队列前端的元素。
- preceDence()是一个函数,看起来是运算符的优先表
伪代码下面:
-
将infix表达式转换为rpn(例如,shunting-yard algo o(n))
AB+C*
-
仅在队列
中插入操作员Q
(前) -------- *(后方)
- parse Postfix表达式
- 如果操作数,请推到stack's'
- 如果操作员
- y = delete(q)
- 如果优先级(y)>优先(peek(q)),则按(s," pop(s)y pop(s)")
- 如果优先(y)&lt;优先(peek(q)),然后推(s,"(pop(s)y pop(s))")
-
S
顶部的最终结果
全部应该是o(n)。
我以为我会做一个刺。这是解决我的问题的解决方案。请注意,这是伪代码,不是直接运行。
(实际上,这是C - ISH,但是自从我上次写真正的C 以来已经有一段时间了,当我打算通过算法时,我不想付出努力来使一切正确。)
queue<tuple<int, bool> > q = new queue<tuple<int, bool> >();
for (int i = 0; i < str.length; i++)
{
char a = str.charAt(i);
if (a == '(')
{
q.push(new tuple<int, bool>(i, false));
}
else if (a == ')')
{
if (q.top().bool)
{
// remove top.int and i from string
}
q.pop();
q.top().bool = true;
}
else
{
q.top().bool = false;
}
}
它在O(n)
中完成工作并使用O(n)
空间(实际上,所使用的空间量实际上是基于字符串中最深的嵌套级别,但保证它低于n
)
(请注意,// remove top.int and i from string
实际上无法在O(1)
中完成。但是,如果您获得一些创意,则可以在O(1)
中执行类似的操作。例如,您实际上可以为输出构建字符列表并存储迭代器而不是int,然后可以删除O(1)中的两个字符。最后,您可以通过在o(n)中的列表上迭代来构建最终字符串。另一种解决方案是实际上在假char虫的阵列(或向量)上工作,这些阵列(或矢量)要么是虚拟的,而且不包含或包含一个字符。再次,您可以通过O(1)
中的虚拟替换字符。再一次,您将迭代结构并在O(n)
中构建输出字符串)
- 将数组的地址分配给变量并删除
- 删除一个线程上有数百万个字符串的大型哈希映射会影响另一个线程的性能
- C/C++编译器通常会删除重复的库吗
- 从链接列表c++中删除一个项目
- C++如何通过用户输入删除列表元素
- 为什么在C++中使用私有复制构造函数与删除复制构造函数
- 是否需要删除包含对象的"pair"?
- 如何在自删除后将对象设置为nullptr
- 迭代时从向量和内存中删除对象
- 使用函数"remove"删除重复元素
- 如何从多映射中删除特定的重复项
- 运算符C++ "delete []"仅删除 2 个前值
- 删除指向指针的指针是运行时错误吗
- 将指针设置为"nullptr"并不能防止双重删除?
- 为什么示例代码访问IUnknown中已删除的内存
- 如何通过 getter 函数删除矢量的元素?
- 从控制台中删除最后打印的元素
- C++中的线程安全删除
- 如何从存储在std::映射中的std::集中删除元素
- 是否有C++编译器选项允许激进地删除所有函数调用,并将参数传递给具有空体的函数