Leetcode每日一题打卡 3170

题意

       给你一个字符串 s 。它可能包含任意数量的 '*' 字符。你的任务是删除所有的 '*' 字符。当字符串还存在至少一个 '*' 字符时,你可以执行以下操作:删除最左边的 '*' 字符,同时删除该星号字符左边一个字典序最小的字符。如果有多个字典序最小的字符,你可以删除它们中的任意一个。

       请你返回删除所有 '*' 字符以后,剩余字符连接而成的字典序最小的字符串。

思考

        这个题一眼看上去就记起了双指针删除元素,但本题限制更多。最初的想法是根据每个*符号的位置进行分割,然后对子字符串操作,最后再拼接得到答案。想法美好,但是倒在了第一步:C++没有split...。用stringstream也可以做到和split同样的效果,但博主不太熟悉只好作罢。即便能做到分割,但接下来更加麻烦,需要记录星号位置、最小字符位置,对每个子字符串需要考虑在遇到相同字符时,如何删除字典序最小字符才能使字典序最小。

        拿s=aaba*举例,如果删除前两个a中的一个,结果为 aba;删除最后一个a,结果为 aab。很明显aab的字典序最小。因此,我们的想法应该是每次操作去掉相同最小字母中靠后的字母。这样才使得较大字典序的字母不会移动到前面。

        想到这里,题目也很清晰了,可惜博主死脑筋只会卡在指针和循环上,没有AC。

题解

        看了灵神的题解,醍醐灌顶,照镜子仿佛在看傻子。

        从左到右遍历 s,用 26 个栈记录遍历过的每种字母的下标。遇到 *,弹出最小字母栈(第一个非空栈)的栈顶。最后把剩余下标对应的字母按顺序串起来,即为答案

        26个栈也是惊为天人了,这辈子没想过用这么多.....

Leetcode 2434解题代码

class Solution {
public:
    string clearStars(string s) 
    {
        stack<int> stk[26];
        for (int i = 0; i < s.size(); i++) 
        {
            if (s[i] != '*') 
            {
                stk[s[i] - 'a'].push(i);
                continue;
            }
            for (auto& st : stk) 
            {
                if (!st.empty()) 
                {
                    s[st.top()] = '*';
                    st.pop();
                    break;
                }
            }
        }
        s.erase(ranges::remove(s, '*').begin(), s.end());
        return s;
    }
};

致谢

        感谢灵神的题解使我豁然开朗。

PS

        另,在钻死胡同时也学到了一些其他操作

#include <algorithm>
#include <iostream>
using namespace std;
int main()
{
    vector<double> v {1.0, 2.0, 3.0, 4.0, 5.0, 1.0, 2.0, 3.0, 4.0, 5.0};
 
    vector<double>::iterator biggest = std::max_element(std::begin(v), std::end(v));
    //or vector<double>::iterator biggest = std::max_element(v.begin(), v.end);
 
    
    cout << "Max element is " << *biggest<< " at position " <<distance(std::begin(v), biggest) << endl;

    //另一方面,取最大位置也可以这样来写:
    //int nPos = (int)(max_element(v.begin(), v.end()) - (v.begin());
    //效果和采用distance(...)函数效果一致
    //说明:max_element(v.begin(), v.end()) 返回的是vector<double>::iterator, 
    //相当于指针的位置,减去初始指针的位置结果即为最大值得索引。
 
    auto smallest = min_element(std::begin(v), std::end(v));
    cout << "min element is " << *smallest<< " at position " <<distance(std::begin(v), smallest) << endl;
}
 
 
输出:
 
Max element is 5 at position 4
min element is 1 at position 0

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值