2023-06-28:你想要用小写字母组成一个目标字符串 target。

开始的时候,序列由 target.length 个 '?' 记号组成

而你有一个小写字母印章 stamp。

在每个回合,你可以将印章放在序列上,并将序列中的每个字母替换为印章上的相应字母

你最多可以进行 10 * target.length 个回合

举个例子,如果初始序列为 "?????",而你的印章 stamp 是 "abc"

那么在第一回合,你可以得到 "abc??"、"?abc?"、"??abc"

(请注意,印章必须完全包含在序列的边界内才能盖下去。)

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

如果不能印出序列,就返回一个空数组。

例如,如果序列是 "ababc",印章是 "abc"

那么我们就可以返回与操作 "?????" -> "abc??" -> "ababc" 相对应的答案 [0, 2]

另外,如果可以印出序列,那么需要保证可以在 10 * target.length 个回合内完成

任何超过此数字的答案将不被接受。

输入:stamp = "abc", target = "ababc"。

输出:[0,2]。

答案2023-06-28:

大体步骤如下:

1.创建变量s和t,分别存储印章stamp和目标字符串target的字节数组表示。

2.创建变量m和n,分别存储印章长度和目标字符串长度。

3.创建数组inDegrees,长度为n-m+1,初始化每个元素为m。该数组表示每个位置需要匹配的印章字符数量。

4.创建二维数组graph,长度为n,每个位置是一个空的整数数组。该数组表示目标字符串每个位置对应的可能的匹配位置。

5.创建队列queue,长度为n-m+1,用于存储已经匹配完所有印章字符的位置。

6.创建变量l和r,分别表示队列queue的左右指针,初始时都为0。

7.遍历目标字符串,从0到n-m,依次处理每个位置:

7.1.在当前位置i,遍历印章的每个字符:

7.1.1.若目标字符串t的第i+j个字符与印章字符相等,表示匹配成功,更新inDegrees数组,将对应位置的值减1。

7.1.1.1.如果经过减1操作后,该位置上印章字符匹配数量变为0,将该位置加入队列queue,并将右指针r向右移动。

7.1.2. 若目标字符串t的第i+j个字符与印章字符不相等,表示匹配失败,将该位置加入graph[i+j]数组中,表示可以在该位置之后的某个位置尝试匹配印章。

8.创建bool类型的数组visited,长度为n,用于标记目标字符串的位置是否被访问过。

9.创建数组path,长度为n-m+1,用于记录完成印章替换的顺序。

10.创建变量size,初始为0,表示已经完成替换的印章的数量。

11.当左指针l小于右指针r时,执行以下循环:

11.1.取出队列queue中的当前位置cur,并将左指针l右移。

11.2.将当前位置cur加入数组path中,并增加size的值。

11.3.遍历印章的每个字符:

11.3.1.若当前位置cur+i未被访问过,表示可以尝试在该位置继续匹配印章:

11.3.1.1.将该位置标记为已访问visited[cur+i] = true。

11.3.1.2.遍历当前位置cur+i对应的graph数组中的每个位置next:

11.3.1.2.1.更新inDegrees数组,将对应位置的值减1。

11.3.1.2.1.1.如果经过减1操作后,该位置上印章字符匹配数量变为0,将该位置加入队列queue,并将右指针r向右移动。

12.检查完成替换的印章数量是否等于n-m+1,如果不相等,返回空数组[]。

13.将数组path中的元素按照首尾对称的顺序重新排列,即交换元素path[i]和path[j],其中i从0遍历到size-1,j从size-1遍历到0。

14.返回数组path作为结果。

该程序的总时间复杂度和总空间复杂度为:

总时间复杂度:O((n - m + 1) * m),其中n是target字符串的长度,m是stamp字符串的长度。

总空间复杂度:O(n),其中n是target字符串的长度。

go完整代码如下:

package main

import (
"fmt"
) func movesToStamp(stamp string, target string) []int {
s := []byte(stamp)
t := []byte(target)
m := len(s)
n := len(t)
inDegrees := make([]int, n-m+1)
for i := range inDegrees {
inDegrees[i] = m
}
graph := make([][]int, n)
for i := range graph {
graph[i] = []int{}
}
queue := make([]int, n-m+1)
l, r := 0, 0
for i := 0; i <= n-m; i++ {
for j := 0; j < m; j++ {
if t[i+j] == s[j] {
if inDegrees[i]--; inDegrees[i] == 0 {
queue[r] = i
r++
}
} else {
graph[i+j] = append(graph[i+j], i)
}
}
}
visited := make([]bool, n)
path := make([]int, n-m+1)
size := 0
for l < r {
cur := queue[l]
l++
path[size] = cur
size++
for i := 0; i < m; i++ {
if !visited[cur+i] {
visited[cur+i] = true
for _, next := range graph[cur+i] {
if inDegrees[next]--; inDegrees[next] == 0 {
queue[r] = next
r++
}
}
}
}
}
if size != n-m+1 {
return []int{}
}
for i, j := 0, size-1; i < j; i, j = i+1, j-1 {
path[i], path[j] = path[j], path[i]
}
return path
} func main() {
stamp := "abc"
target := "ababc"
result := movesToStamp(stamp, target)
fmt.Println(result)
}

rust完整代码如下:

fn moves_to_stamp(stamp: String, target: String) -> Vec<i32> {
let s: Vec<char> = stamp.chars().collect();
let t: Vec<char> = target.chars().collect();
let m = s.len();
let n = t.len();
let mut in_degrees: Vec<i32> = vec![m as i32; n - m + 1];
let mut graph: Vec<Vec<i32>> = vec![Vec::new(); n];
let mut queue: Vec<i32> = vec![0; n - m + 1];
let mut l = 0;
let mut r = 0; for i in 0..=n - m {
for j in 0..m {
if t[i + j] == s[j] {
if in_degrees[i] > 0 {
in_degrees[i] -= 1;
} if in_degrees[i] == 0 {
queue[r] = i as i32;
r += 1;
}
} else {
graph[i + j].push(i as i32);
}
}
} let mut visited: Vec<bool> = vec![false; n];
let mut path: Vec<i32> = vec![0; n - m + 1];
let mut size = 0; while l < r {
let cur = queue[l];
l += 1;
path[size] = cur;
size += 1; for i in 0..m {
let cur_i = cur + i as i32;
if !visited[cur_i as usize] {
visited[cur_i as usize] = true;
for &next in &graph[cur_i as usize] {
if in_degrees[next as usize] > 0 {
in_degrees[next as usize] -= 1;
} if in_degrees[next as usize] == 0 {
queue[r] = next;
r += 1;
}
}
}
}
} if size != n - m + 1 {
return Vec::new();
} for i in 0..size / 2 {
let tmp = path[i];
path[i] = path[size - 1 - i];
path[size - 1 - i] = tmp;
} path
} fn main() {
let stamp = String::from("abc");
let target = String::from("ababc");
let result = moves_to_stamp(stamp, target);
println!("{:?}", result);
}

c++完整代码如下:

#include <iostream>
#include <vector>
#include <algorithm> using namespace std; vector<int> movesToStamp(string stamp, string target) {
int m = stamp.length();
int n = target.length();
vector<int> inDegrees(n - m + 1, m);
vector<vector<int>> graph(n, vector<int>());
vector<int> queue(n - m + 1, 0);
int l = 0, r = 0; for (int i = 0; i <= n - m; i++) {
for (int j = 0; j < m; j++) {
if (target[i + j] == stamp[j]) {
if (--inDegrees[i] == 0) {
queue[r++] = i;
}
}
else {
graph[i + j].push_back(i);
}
}
} vector<bool> visited(n, false);
vector<int> path(n - m + 1, 0);
int size = 0; while (l < r) {
int cur = queue[l++];
path[size++] = cur; for (int i = 0; i < m; i++) {
if (!visited[cur + i]) {
visited[cur + i] = true; for (int next : graph[cur + i]) {
if (--inDegrees[next] == 0) {
queue[r++] = next;
}
}
}
}
} if (size != n - m + 1) {
return vector<int>();
} reverse(path.begin(), path.begin() + size);
return path;
} int main() {
string stamp = "abc";
string target = "ababc"; vector<int> result = movesToStamp(stamp, target); cout << "Result: ";
for (int num : result) {
cout << num << " ";
}
cout << endl; return 0;
}

2023-06-28:你想要用小写字母组成一个目标字符串 target。 开始的时候,序列由 target.length 个 ‘?‘ 记号组成 而你有一个小写字母印章 stamp。 在每个回合,你可的更多相关文章

  1. Insider Dev Tour(2018.06.28)

    时间:2018.06.28地点:北京金茂万丽酒店

  2. http://www.blogjava.net/zJun/archive/2006/06/28/55511.html

    http://www.blogjava.net/zJun/archive/2006/06/28/55511.html http://www.cnblogs.com/alipayhutu/archive ...

  3. 22.编写一个类A,该类创建的对象可以调用方法showA输出小写的英文字母表。然后再编写一个A类的子类B,子类B创建的对象不仅可以调用方法showA输出小写的英文字母表,而且可以调用子类新增的方法showB输出大写的英文字母表。最后编写主类C,在主类的main方法 中测试类A与类B。

    22.编写一个类A,该类创建的对象可以调用方法showA输出小写的英文字母表.然后再编写一个A类的子类B,子类B创建的对象不仅可以调用方法showA输出小写的英文字母表,而且可以调用子类新增的方法sh ...

  4. 2018.06.28 BZOJ1014 [JSOI2008]火星人prefix(非旋treap+hash)

    [JSOI2008]火星人prefix Time Limit: 10 Sec Memory Limit: 162 MB Submit: 8951 Solved: 2860 Description 火星 ...

  5. Murano Weekly Meeting 2016.06.28

    Meeting time: 2016.June.28 1:00~2:00 Chairperson:  Kirill Zaitsev, from Mirantis Meeting summary: 1. ...

  6. Effective C++ -----条款06:若不想使用编译器自动生成的函数,就该明确拒绝

    为驳回编译器自动提供的功能,可将相应的成员函数声明为private并且不予实现. 使用像Uncopyable这样的base class也是一种做法(即先声明一个基类,然后私有继承它).这其实有点像使用 ...

  7. http://www.cnblogs.com/Joyes1989/archive/2013/06/28/3161739.html centos 输入法安装切换

    昨天装了一个centos  安装输入法的时候  让我有点纠结  全英文的 读不懂

  8. 【资料下载区】【GMT43相关代码、资料下载地址】更新日期2017/06/28

    [GMT43相关文档][更新中...] GMT43原理图(PDF)下载GMT43说明书(PDF)下载GMT43机械结构尺寸(PDF)下载 [GMT43相关例程代码][ARM][更新中...] 基于HA ...

  9. DSP builder安装指南(以9.1为例) 转自http://www.cnblogs.com/sleepy/archive/2011/06/28/2092362.html

    DSP Builder在算法友好的开发环境中帮助设计人员生成DSP设计硬件表征,从而缩短了DSP设计周期.已有的MATLAB函数和Simulink模块可以和Altera DSP Builder模块以及 ...

  10. 2019.06.28 MERGE INTO备忘

    --保存主表 MERGE INTO dbo.DeliveryReceiving AS t USING @ReceiveMainDt AS s ON t.Id=s.id WHEN MATCHED THE ...

随机推荐

  1. 企业应用可观测性利器!华为云CodeArts APM发布

    摘要:近日,华为云全链路应用性能管理服务CodeArts APM全新上线,提供端到端的全链路性能管理服务,涵盖前端监控.应用性能监控,全面拥抱开源生态. 本文分享自华为云社区<企业应用可观测性利 ...

  2. springboot-poi ---封装注解式导入导出

    此demo 是基于poi封装对象式注解导入导出,项目框架为springboot项目! 简单的说明一下此demo涉及到的知识点,希望能给初学者带来方便! poi-excel 基本操作(工具) 自定义注解 ...

  3. Albert理论详解:用矩阵分解与跨层参数共享减少参数量

    1.介绍 Albert是Bert的一个变种,它在Bert的基础上减少了参数量,使整个模型更加的"轻量化",同时也保持了Bert的性能,但值得注意的是,Albert虽然显著地减少了参 ...

  4. [Pytorch框架] PyTorch 中文手册

    PyTorch 中文手册 书籍介绍 这是一本开源的书籍,目标是帮助那些希望和使用PyTorch进行深度学习开发和研究的朋友快速入门. 由于本人水平有限,在写此教程的时候参考了一些网上的资料,在这里对他 ...

  5. Android-图片压缩(二)-纯干货

    Android - 图片压缩(一)- 项目中取图片转bitmap Android - 图片压缩(二)- 纯干货 前言:让我们手撸一个图片压缩库,对压缩工具鲁班进行升级改造. 在平常开发当中,我们一般是 ...

  6. laravel框架三级联动,详细代码

    这里运用到省份表中,下面是效果图 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 需要一个省份表,里面的字段要有个pid,name这些字段 下面是控制器代码,(Cit ...

  7. 数据库定时备份winserver2012篇

    目录 1 序言 2 任务计划相关知识点介绍 2.1 任务计划 是什么? 2.2 批处理文件 2.2.1 批处理文件简介 2.2.2 批处理常用命令介绍 3 各个数据库备份脚本 3.1 Oracle数据 ...

  8. 2022-10-17:特殊的二进制序列是具有以下两个性质的二进制序列: 0 的数量与 1 的数量相等。 二进制序列的每一个前缀码中 1 的数量要大于等于 0 的数量。 给定一个特殊的二进制序列 S,以

    2022-10-17:特殊的二进制序列是具有以下两个性质的二进制序列: 0 的数量与 1 的数量相等. 二进制序列的每一个前缀码中 1 的数量要大于等于 0 的数量. 给定一个特殊的二进制序列 S,以 ...

  9. KubeCon EU 2023 落幕,哪些技术趋势值得关注?

    KubeCon+CloudNativeCon 是云原生领域的技术盛会,上个月月末,在荷兰阿姆斯特丹举办的欧洲 KubeCon+CloudNativeCon 刚刚落下帷幕,此次大会吸引了10000多名参 ...

  10. JS中的纯函数

    在JavaScript中,纯函数是指在相同的输入下,始终产生相同的输出,并且没有副作用的函数.纯函数不会修改或依赖于函数之外的状态,也不会对外部环境产生任何可观察的影响. 以下是纯函数的特点: 1. ...