2023-04-11:给你下标从 0 开始、长度为 n 的字符串 pattern ,
它包含两种字符,‘I’ 表示 上升 ,‘D’ 表示 下降 。
你需要构造一个下标从 0 开始长度为 n + 1 的字符串,且它要满足以下条件:
num 包含数字 ‘1’ 到 ‘9’ ,其中每个数字 至多 使用一次。
如果 pattern[i] == ‘I’ ,那么 num[i] < num[i + 1] 。
如果 pattern[i] == ‘D’ ,那么 num[i] > num[i + 1] 。
请你返回满足上述条件字典序 最小 的字符串 num。
输入:pattern = “IIIDIDDD”,
输出:“123549876”。

答案2023-04-11:

解题思路

这是一道比较有趣的贪心题目。我们可以根据给定的 pattern 字符串来决定数字串中相邻两个数的关系。

步骤1:定义 next 函数

首先,我们需要定义一个函数 next(status, num),用来查找在状态 status 中没有使用过的最小数字(大于 num)。该函数通过遍历数字 1 到 9,判断哪些数字在 status 中未被使用,且大于 num,然后返回其中最小的数字。

fn next(status: usize, num: u8) -> Option<u8> {
for i in (num + 1)..=9 {
if (status & (1 << i)) == 0 {
return Some(i as u8);
}
}
None
}

步骤2:定义 create 函数

接着,我们需要定义另一个函数 create(pattern, index, status, number),用来递归生成数字串,并判断是否符合要求。在递归过程中,我们需要判断当前位应该填入哪个数字,并根据数字的大小关系更新 status、number 和 index 的值。如果生成的数字串不符合要求,则需要回溯并重新选择数字。

fn create(pattern: &[char], index: usize, status: &mut usize, number: &mut u32) -> bool {
if index == pattern.len() + 1 {
return true;
} let mut cur = 0;
while let Some(next_cur) = next(*status, cur) {
cur = next_cur; // cur == 0 , 当前位,1 X
// cur == 1 , 当前位,2 X
// cur == 2, 当前位,4
// pattern I D >
// 0 1 2 3
// ? ? ? ?
// D
// 0 1
// 5 ?
if index == 0
|| (pattern[index - 1] == 'I' && *number % 10 < cur as u32)
|| (pattern[index - 1] == 'D' && *number % 10 > cur as u32)
{
*status |= 1 << cur;
*number = *number * 10 + cur as u32; if create(pattern, index + 1, status, number) {
return true;
} *number /= 10;
*status &= !(1 << cur);
}
} false
}

步骤3:定义 smallest_number 函数

最后,我们需要定义 smallest_number 函数,调用 create 函数来生成数字串,并将其转化为字符串类型返回。

fn smallest_number(pattern: &str) -> String {
let chars = pattern.chars().collect::<Vec<_>>();
let mut status = 0usize;
let mut number = 0; create(&chars, 0, &mut status, &mut number); number.to_string()
}

时间复杂度

对于这个解法,最坏情况下需要枚举所有可能的数字串,因此时间复杂度为 O(n * 9!),其中 n 是 pattern 字符串的长度。在实际测试中,由于存在大量剪枝操作,实际运行时间要比这个上界要小得多。

空间复杂度

主要的存储空间是用来记录数字是否被使用过的 status 变量和已经生成的数字串 number 变量,以及递归调用栈所占用的空间。其中,status 和 number 变量的大小均为常数级别,因此空间复杂度为 O(1)。递归调用栈的深度最多为 n + 1,因此空间复杂度为 O(n)。

rust完整代码如下:

fn smallest_number(pattern: &str) -> String {
let chars = pattern.chars().collect::<Vec<_>>();
let mut status = 0usize;
let mut number = 0; create(&chars, 0, &mut status, &mut number); number.to_string()
} /// 返回 i... 所有数字都决定了,并且不破坏pattern,
/// 并且1~9每个数字最多用一次能出来的最小值是啥,返回
fn create(pattern: &[char], index: usize, status: &mut usize, number: &mut u32) -> bool {
if index == pattern.len() + 1 {
return true;
} let mut cur = 0;
while let Some(next_cur) = next(*status, cur) {
cur = next_cur; // cur == 0 , 当前位,1 X
// cur == 1 , 当前位,2 X
// cur == 2, 当前位,4
// pattern I D >
// 0 1 2 3
// ? ? ? ?
// D
// 0 1
// 5 ?
if index == 0
|| (pattern[index - 1] == 'I' && *number % 10 < cur as u32)
|| (pattern[index - 1] == 'D' && *number % 10 > cur as u32)
{
*status |= 1 << cur;
*number = *number * 10 + cur as u32; if create(pattern, index + 1, status, number) {
return true;
} *number /= 10;
*status &= !(1 << cur);
}
} false
} /// 返回没有使用,且 > num, 最小的数字
fn next(status: usize, num: u8) -> Option<u8> {
for i in (num + 1)..=9 {
if (status & (1 << i)) == 0 {
return Some(i as u8);
}
}
None
} fn main() {
let pattern = "IIIDIDDD";
let result = smallest_number(pattern);
println!("{}", result); // 输出:123549876
}

2023-04-11:给你下标从 0 开始、长度为 n 的字符串 pattern , 它包含两种字符,‘I‘ 表示 上升 ,‘D‘ 表示 下降 。 你需要构造一个下标从 0 开始长度为 n + 1 的的更多相关文章

  1. zabbix3.0.4使用shell脚本和zabbix自带模板两种方法添加对指定进程和端口的监控

    zabbix3.0.4添加对进程的监控: 方法一:通过自定义命令进行监控 主要思路: 通过 ps -ef|grep sdk-push-1.0.0.jar |grep -v grep|wc -l 这个命 ...

  2. SQL 按月统计(两种方式) 分类: SQL Server 2014-08-04 15:36 154人阅读 评论(0) 收藏

    (1)Convert 函数 select Convert ( VARCHAR(7),ComeDate,120) as Date ,Count(In_code) as 单数,Sum(SumTrueNum ...

  3. 转载:Flash AS3.0 加载外部资源(图片,MP3,SWF)的两种方式

    Flash AS3.0 加载外部资源(图片,MP3,SWF)的两种方式 出自:http://www.cnblogs.com/top5/archive/2012/08/04/2623464.html 关 ...

  4. 4.写一个控制台应用程序,接收一个长度大于3的字符串,完成下列功能: 1)输出字符串的长度。 2)输出字符串中第一个出现字母a的位置。 3)在字符串的第3个字符后面插入子串“hello”,输出新字符串。 4)将字符串“hello”替换为“me”,输出新字符串。 5)以字符“m”为分隔符,将字符串分离,并输出分离后的字符串。 */

    namespace test4 {/* 4.写一个控制台应用程序,接收一个长度大于3的字符串,完成下列功能: 1)输出字符串的长度. 2)输出字符串中第一个出现字母a的位置. 3)在字符串的第3个字符 ...

  5. C#返回字符串的字节长度,一个中文算两个字符的代码

    如下代码段是关于C#返回字符串的字节长度,一个中文算两个字符的代码. public static int GetLength(string str) { if (str.Length == 0) re ...

  6. Shell脚本中字符串判空:使用-z 字符串长度为0时,为真,-n字符串长度不为0,为真。这两个都不靠谱【转】

    最近发现使用  -z   和  -n  来判断字符串判空,或不空时,很不靠谱. 使用下面的方法最可靠: if [ "x${value}" == "x" ]    ...

  7. php在数字前面补0得到固定长度数字的两种方法

    比較基础,事实上两个内置函数都能实现. 1  sprintf 语法: string sprintf(string format, mixed [args]...); 返回值: 字符串 函数种类: 资料 ...

  8. 20172319 2018.04.11 《Java程序设计教程》第7周课堂测验(补写博客)

    20172319 2018.04.11 <Java程序设计教程>第7周课堂测验 课程:<程序设计与数据结构> 班级:1723 学生:唐才铭 学号:20172319 指导老师:王 ...

  9. LeetCode 周赛 342(2023/04/23)容斥原理、计数排序、滑动窗口、子数组 GCB

    本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 提问. 大家好,我是小彭. 前天刚举办 2023 年力扣杯个人 SOLO 赛,昨天周赛就出了一场 Easy - Ea ...

  10. 山东理工大学第七届ACM校赛-字符的变化 分类: 比赛 2015-06-26 10:32 46人阅读 评论(0) 收藏

    字符的变化 Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描述 一个长度为n(1<=n<=1000)的字符串(只包含小写字 ...

随机推荐

  1. ubuntu下删除U盘文件到回收站无法清空问题的解决

    Ubuntu可以自动加载U盘 每当,拷贝新的文件,而空间不足的时候,就会删除原有的文件. 可是,它不是彻底删除,而是放在垃圾箱中(/home/mrc/.local/share/Trash/files) ...

  2. finereport连接mysql8.0

    1.java包更新 2.下载地址:https://dev.mysql.com/downloads/connector/j/ 3.替换文件为8.0删除5.1版本 4.驱动器手动输入com.mysql.c ...

  3. 《MySQL是怎样运行的》第四章小结

  4. 文件上传靶场 upload-labs搭建 Pass 1-4

    upload-labs是一个练习文件上传的靶场 我们需要先安装中间件和PHP,推荐使用小皮面板,如何安装使用见sqli-labs搭建前部分 upload-labs下载:https://gitcode. ...

  5. Thingsboard3.2.2本地部署

    Thingboard3.2.2本地安装编译详细教程!!! 一:拉取源码. 创建一个空的文件夹 在此处使用git拉取源码. git clone https://github.com/thingsboar ...

  6. Linux & 标准C语言学习 <DAY8_1>

    一.进制转换     1.为什么要使用二进制.八进制.十六进制         因为目前的CPU只能识别高低两种电平,只能对二进制数据进行计算         二进制虽然能够直接被计算机识别,但是不方 ...

  7. 【AI 全栈 SOTA 综述 】这些你都不知道,怎么敢说会 AI?【语音识别原理 + 实战】

    章目录 前言语音识别原理   信号处理,声学特征提取   识别字符,组成文本   声学模型   语言模型   词汇模型语音声学特征提取:MFCC和LogFBank算法的原理实战一 ASR语音识别模型 ...

  8. 用Python基于Google Bard做一个交互式的聊天机器人

    用Python基于Google Bard做一个交互式的聊天机器人 之前已经通过浏览器试过了 Google Bard ,更多细节请看: Try out Google Bard, Will Google ...

  9. 如何提取 x64 程序那些易失的方法参数

    一:背景 1. 讲故事 最近经常遇到有朋友反馈,在 x64 环境下如何提取线程栈中的方法参数,熟悉 x64 调用协定的朋友应该知道,这种协定范围下,方法的前四个参数都是用寄存器传递的,比如rcx,rd ...

  10. Spring Boot 中的 ApplicationRunner 和 CommandLineRunner

    前言 一般项目中的初始化操作,初次遇见,妙不可言.如果你还有哪些方式可用于初始化操作,欢迎在评论中分享出来~ ApplicationRunner 和 CommandLineRunner Spring ...