• 戳印序列原理及C++实现方法一


    LeetCode936题目

    你想要用小写字母组成一个目标字符串 target。

    开始的时候,序列由 target.length 个 '?' 记号组成。而你有一个小写字母印章 stamp。

    在每个回合,你可以将印章放在序列上,并将序列中的每个字母替换为印章上的相应字母。你最多可以进行 10 * target.length  个回合。

    举个例子,如果初始序列为 "?????",而你的印章 stamp 是 "abc",那么在第一回合,你可以得到 "abc??"、"?abc?"、"??abc"。(请注意,印章必须完全包含在序列的边界内才能盖下去。)

    如果可以印出序列,那么返回一个数组,该数组由每个回合中被印下的最左边字母的索引组成。如果不能印出序列,就返回一个空数组。

    例如,如果序列是 "ababc",印章是 "abc",那么我们就可以返回与操作 "?????" -> "abc??" -> "ababc" 相对应的答案 [0, 2];

    另外,如果可以印出序列,那么需要保证可以在 10 * target.length 个回合内完成。任何超过此数字的答案将不被接受。

    1 <= stamp.length <= target.length <= 1000

    stamp 和 target 只包含小写字母。

    分析

    自己想出来的笨办法,时间复杂度较低。公认的方法是拓扑排序。

    代码

    class Solution {
    public:
        vector movesToStamp(string stamp, string target) {
            int iWindowCount = target.length() - stamp.length() + 1;
            m_strDebug = target;
            unordered_map> vNotSame;
            std::unordered_multiset setNeed;
            for (int i = 0; i < target.size(); i++)
            {
                setNeed.emplace(i);
            }
            vector> vDirect(target.length());
            std::vector vQue;
            for (int i = 0; i < iWindowCount; i++)
            {
                for (int j = 0; j < stamp.size(); j++)
                {
                    if (stamp[j] != target[i + j])
                    {
                        vNotSame[i].emplace(i + j);
                        vDirect[i + j].emplace_back(i);
                    }
                }
                if (!vNotSame.count(i))
                {
                    vQue.emplace_back(i);
                }
            }

            int index = 0;
            while (setNeed.size() && (index < vQue.size()))
            {
                const auto cur = vQue[index];
                index++;
                for (int i = 0; i < stamp.length(); i++)
                {
                    setNeed.erase(i + cur);
                    assert(('?' == m_strDebug[i + cur]) || (m_strDebug[i + cur] == stamp[i]));
                    m_strDebug[i + cur] = '?';
                    for (const auto& n : vDirect[i + cur])
                    {
                        if (!vNotSame.count(n))
                        {
                            continue;
                        }
                        vNotSame[n].erase(i + cur);
                        if (0 == vNotSame[n].size())
                        {
                            vQue.emplace_back(n);
                            vNotSame.erase(n);
                        }
                    }
                }
                //    CConsole::Out(m_strDebug);
            }

            vector vRet(vQue.begin(), vQue.begin() + index);
            return setNeed.size() ? vector{} : vector(vRet.rbegin(), vRet.rend());
        }
        string m_strDebug;
    };

    其它

    视频课程

    如果你觉得复杂,想从简单的算法开始,可以学习我的视频课程。
    https://edu.csdn.net/course/detail/38771
    我的其它课程
    https://edu.csdn.net/lecturer/6176

    测试环境

    win7 VS2019 C++17

    相关下载


    doc版文档,排版好
    https://download.csdn.net/download/he_zhidan/88348653
     

  • 相关阅读:
    前端收藏夹 ,以及他们的github地址:GitHub - w3ctrain/w3ctrain.github.io: w3ctrian前端收藏夹
    Prism 实现依赖注入
    ffmpeg 之ffmpeg 整理流程分析
    ES7,ES8
    java毕业设计毕业设计管理系统Mybatis+系统+数据库+调试部署
    CesiumJS 2022^ 原理[3] 渲染原理之从 Entity 看 DataSource 架构 - 生成 Primitive 的过程
    JAVA微服务知识概述
    束带机安全使用须知
    大健康产业商业供应链管理系统数字化提升产业链运作效率推动供应链标准化建设
    小程序的各种手机屏幕媒体查询
  • 原文地址:https://blog.csdn.net/he_zhidan/article/details/133650811