2023-05-21:给定一个字符串 s 和一个整数 k 。你可以从 s 的前 k 个字母中选择一个,

并把它加到字符串的末尾。

返回 在应用上述步骤的任意数量的移动后,字典上最小的字符串。

输入:s = "baaca", k = 3。

输出:"aaabc"。

答案2023-05-21:

大体过程如下:

1.当 k 大于 1 时,直接将字符串 s 中的字符按照字典序排序,得到排序后的字符串 s',返回 s'。

2.当 k 等于 1 时,需要使用 DC3 算法对字符串 s 进行处理,得到其所有后缀排名,并找到排名最小的后缀起始位置 minRankIndex。

3.将字符串 s 的前 minRankIndex 个字符移动到字符串末尾,得到新的字符串 s',返回 s'。

值得注意的是,DC3 算法是一种用于求解后缀数组的算法,可以在 O(n) 的复杂度内计算一个字符串的后缀数组。在本题中,我们需要用到 DC3 算法来寻找字符串 s 所有后缀的排名,以便找到排名最小的后缀起始位置。

对于给定字符串 s 和整数 k,orderlyQueue 函数的时间复杂度和空间复杂度分别如下:

1.当 k > 1 时,时间复杂度为 O(nlogn),其中 n 是字符串 s 的长度。主要耗时在排序操作中,使用快速排序等算法可以达到 O(nlogn) 的复杂度。空间复杂度也为 O(nlogn),主要用于存储字符串数组的副本和排序结果。

2.当 k = 1 时,时间复杂度为 O(n),其中 n 是字符串 s 的长度。时间复杂度主要来自 DC3 算法的实现,该算法可以在 O(n) 的时间复杂度内计算一个字符串的后缀数组。空间复杂度为 O(n),主要用于存储后缀数组、排名和其他中间变量。

综上所述,orderlyQueue 函数的时间复杂度为 O(nlogn) 或 O(n),空间复杂度为 O(nlogn) 或 O(n),具体取决于 k 的值。

go完整代码如下:

package main

import (
"fmt"
"sort"
"strings"
) func orderlyQueue(s string, k int) string {
if k > 1 {
sArr := strings.Split(s, "")
sort.Strings(sArr)
return strings.Join(sArr, "")
} else {
s2 := s + s
n := len(s2)
arr := make([]int, n)
for i := 0; i < n; i++ {
arr[i] = int(s2[i] - 'a' + 1)
}
dc3 := NewDC3(arr, 26)
n = n >> 1
minRankIndex := 0
for i := 1; i < n; i++ {
if dc3.rank[i] < dc3.rank[minRankIndex] {
minRankIndex = i
}
}
return s[minRankIndex:] + s[0:minRankIndex]
}
} // DC3算法实现
// 根据原算法Java代码修改
type DC3 struct {
sa []int
rank []int
} // NewDC3 构造函数
func NewDC3(nums []int, max int) *DC3 {
dc3 := &DC3{}
dc3.sa = dc3.sa0(nums, max)
dc3.rank = dc3.rank0()
return dc3
} func (dc3 *DC3) sa0(nums []int, K int) []int {
n := len(nums)
arr := make([]int, n+3)
copy(arr, nums)
return dc3.skew(arr, n, K)
} func (dc3 *DC3) skew(nums []int, n int, K int) []int {
n0 := (n + 2) / 3
n1 := (n + 1) / 3
n2 := n / 3
n02 := n0 + n2
s12 := make([]int, n02+3)
sa12 := make([]int, n02+3)
for i, j := 0, 0; i < n+(n0-n1); i++ {
if i%3 != 0 {
s12[j] = i
j++
}
}
dc3.radixPass(nums, s12, sa12, 2, n02, K)
dc3.radixPass(nums, sa12, s12, 1, n02, K)
dc3.radixPass(nums, s12, sa12, 0, n02, K) name := 0
c0 := -1
c1 := -1
c2 := -1
for i := 0; i < n02; i++ {
if nums[sa12[i]] != c0 || nums[sa12[i]+1] != c1 || nums[sa12[i]+2] != c2 {
name++
c0 = nums[sa12[i]]
c1 = nums[sa12[i]+1]
c2 = nums[sa12[i]+2]
}
if sa12[i]%3 == 1 {
s12[sa12[i]/3] = name
} else {
s12[sa12[i]/3+n0] = name
}
}
if name < n02 {
sa12 = dc3.skew(s12, n02, name)
for i := 0; i < n02; i++ {
s12[sa12[i]] = i + 1
}
} else {
for i := 0; i < n02; i++ {
sa12[s12[i]-1] = i
}
} s0 := make([]int, n0)
sa0 := make([]int, n0)
for i, j := 0, 0; i < n02; i++ {
if sa12[i] < n0 {
s0[j] = 3 * sa12[i]
j++
}
}
dc3.radixPass(nums, s0, sa0, 0, n0, K) sa := make([]int, n)
for p, t, k := 0, n0-n1, 0; k < n; k++ {
i := sa12[t]
if i < n0 {
i = i*3 + 1
} else {
i = (i-n0)*3 + 2
}
j := sa0[p]
if i < n-1 && j < n-1 {
if nums[i] < nums[j] || (nums[i] == nums[j] && nums[i+1] < nums[j+1]) ||
(nums[i] == nums[j] && nums[i+1] == nums[j+1] && nums[i+2] <= nums[j+2]) {
sa[k] = i
t++
if t == n02 {
k++
for ; p < n0; p++ {
sa[k] = sa0[p]
k++
}
}
} else {
sa[k] = j
p++
if p == n0 {
k++
for ; t < n02; t++ {
i := sa12[t]
if i < n0 {
sa[k] = i*3 + 1
} else {
sa[k] = (i-n0)*3 + 2
}
k++
}
}
}
} else {
if nums[i] < nums[j] || (nums[i] == nums[j] && nums[i+1] <= nums[j+1]) {
sa[k] = i
t++
if t == n02 {
k++
for ; p < n0; p++ {
sa[k] = sa0[p]
k++
}
}
} else {
sa[k] = j
p++
if p == n0 {
k++
for ; t < n02; t++ {
i := sa12[t]
if i < n0 {
sa[k] = i*3 + 1
} else {
sa[k] = (i-n0)*3 + 2
}
k++
}
}
}
}
}
return sa
} func (dc3 *DC3) radixPass(nums []int, input []int, output []int, offset int, n int, k int) {
cnt := make([]int, k+1)
for i := 0; i < n; i++ {
cnt[nums[input[i]+offset]]++
}
for i, sum := 0, 0; i < len(cnt); i++ {
t := cnt[i]
cnt[i] = sum
sum += t
}
for i := 0; i < n; i++ {
output[cnt[nums[input[i]+offset]]] = input[i]
cnt[nums[input[i]+offset]]++
}
} func (dc3 *DC3) rank0() []int {
n := len(dc3.sa)
ans := make([]int, n)
for i := 0; i < n; i++ {
ans[dc3.sa[i]] = i
}
return ans
} func main() {
s := "baaca"
k := 3
result := orderlyQueue(s, k)
fmt.Println(result)
}

2023-05-21:给定一个字符串 s 和一个整数 k 。你可以从 s 的前 k 个字母中选择一个, 并把它加到字符串的末尾。 返回 在应用上述步骤的任意数量的移动后,字典上最小的字符串。 输入:s的更多相关文章

  1. 在某网站的登录页面登录时如果选择“记住用户名”,登录成功后会跳转到一个中间层(页面代码将登录的用户名和密码存在cookie),中间页面中存在一个超链接,单击超链接可以链接到第三个页面查看信息。若选择“

    Response实现登录并记录用户名和密码信息 在某网站的登录页面登录时如果选择"记住用户名",登录成功后会跳转到一个中间层(页面代码将登录的用户名和密码存在cookie),中间页 ...

  2. Java变量命名前俩个字母仅含有一个大写字母的坑

    背景 前几周在做项目fetch切换,即将HttpUtils调用改成使用Feign调用.大概代码如下: // 原代码 String resultJson = HttpUtil.get(url + &qu ...

  3. 现在有一个城市销售经理,需要从公司出发,去拜访市内的商家,已知他的位置以及商家的位置,但是由于城市道路交通的原因,他只能在左右中选择一个方向,在上下中选择一个方向,现在问他有多少种方案到达商家地址。给定一个地图map及它的长宽n和m,其中1代表经理位置,2代表商家位置,-1代表不能经过的地区,0代表可以经过的地区,请返回方案数,保证一定存在合法路径。保证矩阵的长宽都小于等于10。

    include "stdafx.h" #include<iostream> #include<vector> #include<algorithm&g ...

  4. break、continue、return中选择一个,我们结束掉它

      在平时的开发过程中,经常会用到循环,在写循环的过程中会有很多判断条件及逻辑,你知道如何结束一个循环吗?在java中有break.continue.reture三个关键字都可以结束循环,我们看下他们 ...

  5. jquery点击一组按钮中的一个,跳转至对应页面处理策略。(如点击订单列表中的一个订单,跳转至该订单的详情)

    将改组按钮的数据设置一个相同的属性(如class),然后每个按钮设置不同的id 当用户点击属性为class的按钮,根据id属性来判断点击的是哪个按钮,然后进行相关操作. 代码示例: <scrip ...

  6. e769. 在按钮组中选择一个单选按钮

    // To create a radio button and button group, // see e768 创建单选按钮 // Select the radio button; the cur ...

  7. 《Entity Framework 6 Recipes》中文翻译系列 (20) -----第四章 ASP.NET MVC中使用实体框架之在MVC中构建一个CRUD示例

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 第四章  ASP.NET MVC中使用实体框架 ASP.NET是一个免费的Web框架 ...

  8. 在 Confluence 6 中连接一个 LDAP 目录

    希望将 Confluence 连接到一个 LDAP 目录: 在屏幕的右上角单击 控制台按钮 ,然后选择 General Configuration 链接. 在左侧的面板中单击 用户目录(User Di ...

  9. 报错需要选择一个空目录,或者选择的非空目录下存在 app.json 或者 project.config.json解决方案

    前言 小程序的第一个坑就是,创建了一个小程序项目,却在微信web开发者工具无法打开... 报了个错:需要选择一个空目录,或者选择的非空目录下存在 app.json 或者 project.config. ...

  10. 请写一个php函数,可以接受任意数量的参数

    请写一个php函数,可以接受任意数量的参数 这是一道面试题.怎么写这个函数呢? function fun(......) { } ----------------------------------- ...

随机推荐

  1. Cryptanalyzing and Improving a Novel Color Image Encryption Algorithm Using RT-Enhanced Chaotic Tent Maps

    Cryptanalyzing and Improving a Novel ColorImage Encryption Algorithm Using RT-EnhancedChaotic Tent M ...

  2. 推荐一套轻量级的开源图床系统:Light Fast Picture

    如果您跟我一样平时有些博客的习惯,那么图片存储是否有困扰过你呢?今天就给大家推荐一款不错的开源图床系统:Light Fast Picture 它是一个基于koa + vue3.x + typescri ...

  3. Python ArcPy批量拼接长时间序列栅格图像

      本文介绍基于Python中ArcPy模块,对大量不同时相的栅格遥感影像按照其成像时间依次执行批量拼接的方法.   在前期的文章Python arcpy创建栅格.批量拼接栅格中,我们介绍了利用Pyt ...

  4. java网络编程--3 TCP

    java网络编程--3 TCP 1.6.TCP 客户端 连接服务器 Socket 发送消息 package com.ssl.lesson02; import java.io.IOException; ...

  5. MySQL 中 explain关键字

    更多内容,前往IT-BLOG explain关键字可以模拟优化器执行 SQL 查询语句,从而知道 MySQL 是如何处理 SQL 语句的.分析查询语句或表结构的性能瓶颈.执行语句:explain + ...

  6. MySQL 主从复制的问题及解决方案

    更多内容,前往 IT-BLOG 复制功能是构建 MySQL 的大规模.高性能的基础,也就是所谓的 "水平扩展" 架构.我们可以通过为服务器配置一个或多个备库.同时,复制也是高可用性 ...

  7. ChatGPT 设计游戏剧情 | 基于 AI 5 天创建一个农场游戏,完结篇!

    欢迎使用 AI 进行游戏开发! 在本系列中,我们将使用 AI 工具在 5 天内创建一个功能完备的农场游戏.到本系列结束时,您将了解到如何将多种 AI 工具整合到游戏开发流程中.本文将向您展示如何将 A ...

  8. vue3 ts 类式写法的mixins

    vue-property-decorator 混入(mixins) // mixins.ts import { Vue } from 'vue-property-decorator' class Mi ...

  9. 每日复习------main()方法以及对象的初始化顺序

    由于 Java 虚拟机需要调用类的 main()方法,所以该方法的访问权限必须是 public,又因为 Java 虚拟机在执行 main()方法时不必创建对象,所以该方法必须是 static 的,该方 ...

  10. Algorithm参数记录

    一.vector<Point2f> vector是一个存储二维点坐标的容器,其中每个元素都是一个Point2f类型的对象.在OpenCV中,Point2f表示一个由两个单精度浮点数构成的二 ...