C++算法:戳印序列原理及实现二

简介: C++算法:戳印序列原理及实现二

题目

你想要用小写字母组成一个目标字符串 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 只包含小写字母。

分析

倒着处理,把target变成???,某个字符系列除已经变成???问号的部分,其它部分和印章全部相同。

代码

class Solution {
public:
vector movesToStamp(string stamp, string target) {
int m = stamp.size();
int n = target.size();
//vCanLast[i]为x表示。target[i]到target[i+m-1] 共有x个字符和stamp不同。完全相同,为0表示,在i除及后续盖章,
//可以保证target[i]到target[i+m-1] 可以达成目标
//利用拓扑排序解决
m_iWindowNum = n - m + 1;
vector vNeedModify(m_iWindowNum, m);
vector<vector> vDirect(n);
std::stack sta;
for (int i = 0; i < m_iWindowNum; i++)
{
for (int j = 0; j < m; j++)
{
if (target[i + j] == stamp[j])
{
vNeedModify[i]–;
if (0 == vNeedModify[i])
{
sta.push(i);
}
}
else
{
vDirect[i + j].push_back(i);
}
}
}
vector<int> vHasModify(n);
   vector<int> ret;
   while (sta.size())
   {
     int iCur = sta.top();
     sta.pop();
     ret.push_back(iCur);
     for (int j = 0; j < m; j++)
     {
       if (vHasModify[iCur + j])
       {
         continue;
       }
       vHasModify[iCur + j] = 1;
       for (auto& dd : vDirect[iCur + j])
       {
         vNeedModify[dd]--;
         if (0 == vNeedModify[dd])
         {
           sta.push(dd);
         }
       }
     }
   }
   if (n == std::accumulate(vHasModify.begin(), vHasModify.end(), 0))
   {
     return vector<int>(ret.rbegin(), ret.rend());
   }
   return vector<int>();
 }
 int m_iWindowNum;//滑动窗口数量

};

其它

视频课程

如果你觉得复杂,想从简单的算法开始,可以学习我的视频课程。

https://edu.csdn.net/course/detail/38771

我的其它课程

[https://edu.csdn.net/lecturer/6176]

(https://edu.csdn.net/lecturer/6176)

测试环境

win7 VS2019 C++17

相关下载

doc版文档,排版好

https://download.csdn.net/download/he_zhidan/88348653


相关文章
|
1月前
|
算法
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
这篇文章介绍了动态规划算法中解决最大上升子序列问题(LIS)的方法,包括问题的描述、动态规划的步骤、状态表示、递推方程、计算最优值以及优化方法,如非动态规划的二分法。
65 0
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
|
1月前
|
存储 算法 C++
高精度算法(加、减、乘、除,使用c++实现)
高精度算法(加、减、乘、除,使用c++实现)
406 0
高精度算法(加、减、乘、除,使用c++实现)
|
1月前
|
C++
C++番外篇——虚拟继承解决数据冗余和二义性的原理
C++番外篇——虚拟继承解决数据冗余和二义性的原理
39 1
|
1月前
|
算法 数据处理 C++
c++ STL划分算法;partition()、partition_copy()、stable_partition()、partition_point()详解
这些算法是C++ STL中处理和组织数据的强大工具,能够高效地实现复杂的数据处理逻辑。理解它们的差异和应用场景,将有助于编写更加高效和清晰的C++代码。
22 0
|
1月前
|
存储 算法 决策智能
【算法】博弈论(C/C++)
【算法】博弈论(C/C++)
|
1月前
|
存储 算法 C++
【算法】哈希映射(C/C++)
【算法】哈希映射(C/C++)
|
1月前
|
机器学习/深度学习 人工智能 算法
【算法】最长公共子序列(C/C++)
【算法】最长公共子序列(C/C++)
|
24天前
|
算法 安全 数据安全/隐私保护
基于game-based算法的动态频谱访问matlab仿真
本算法展示了在认知无线电网络中,通过游戏理论优化动态频谱访问,提高频谱利用率和物理层安全性。程序运行效果包括负载因子、传输功率、信噪比对用户效用和保密率的影响分析。软件版本:Matlab 2022a。完整代码包含详细中文注释和操作视频。
|
9天前
|
算法 数据挖掘 数据安全/隐私保护
基于FCM模糊聚类算法的图像分割matlab仿真
本项目展示了基于模糊C均值(FCM)算法的图像分割技术。算法运行效果良好,无水印。使用MATLAB 2022a开发,提供完整代码及中文注释,附带操作步骤视频。FCM算法通过隶属度矩阵和聚类中心矩阵实现图像分割,适用于灰度和彩色图像,广泛应用于医学影像、遥感图像等领域。
|
10天前
|
算法 调度
基于遗传模拟退火混合优化算法的车间作业最优调度matlab仿真,输出甘特图
车间作业调度问题(JSSP)通过遗传算法(GA)和模拟退火算法(SA)优化多个作业在并行工作中心上的加工顺序和时间,以最小化总完成时间和机器闲置时间。MATLAB2022a版本运行测试,展示了有效性和可行性。核心程序采用作业列表表示法,结合遗传操作和模拟退火过程,提高算法性能。