2023-07-13:如果你熟悉 Shell 编程,那么一定了解过花括号展开,它可以用来生成任意字符串。 花括号展开的表达式可以看作一个由 花括号、逗号 和 小写英文字母 组成的字符串 定义下面几条语
2023-07-13:如果你熟悉 Shell 编程,那么一定了解过花括号展开,它可以用来生成任意字符串。
花括号展开的表达式可以看作一个由 花括号、逗号 和 小写英文字母 组成的字符串
定义下面几条语法规则:
如果只给出单一的元素 x,那么表达式表示的字符串就只有 "x"。R(x) = {x}
例如,表达式 "a" 表示字符串 "a"。
而表达式 "w" 就表示字符串 "w"。
当两个或多个表达式并列,以逗号分隔,我们取这些表达式中元素的并集
R({e_1,e_2,...}) = R(e_1) ∪ R(e_2) ∪ ...
例如,表达式 "{a,b,c}" 表示字符串 "a","b","c"。
而表达式 "{{a,b},{b,c}}" 也可以表示字符串 "a","b","c"。
要是两个或多个表达式相接,中间没有隔开时,
我们从这些表达式中各取一个元素依次连接形成字符串
R(e_1 + e_2) = {a + b for (a, b) in R(e_1) × R(e_2)}
例如,表达式 "{a,b}{c,d}" 表示字符串 "ac","ad","bc","bd"。
表达式之间允许嵌套,单一元素与表达式的连接也是允许的。
例如,表达式 "a{b,c,d}" 表示字符串 "ab","ac","ad"。
例如,表达式 "a{b,c}{d,e}f{g,h}"
可以表示字符串 :
"abdfg", "abdfh", "abefg", "abefh",
"acdfg", "acdfh", "acefg", "acefh"。
给出表示基于给定语法规则的表达式 expression。
返回它所表示的所有字符串组成的有序列表。
输入:expression = "{a,b}{c,{d,e}}"。
输出:["ac","ad","ae","bc","bd","be"]。
答案2023-07-13:
大体步骤如下:
1.定义了一个结构体 Info
,其中包含一个 treeset.Set
类型的指针 ans
和一个整数 end
。
2.定义了一个 NewInfo
函数,用于初始化 Info
对象。
3.定义了 braceExpansionII
函数,接收一个表示表达式的字符串,并返回展开后的字符串列表。
4.process
函数是实际处理展开过程的核心函数,它接收一个表示表达式的字符数组 exp
和一个起始索引 start
,返回一个 Info
对象。
5.在 process
函数中,创建了一个空的 treeset.Set
对象 ans
和一个空的 []*treeset.Set
切片 parts
。
6.使用 strings.Builder
创建了一个字符串构建器 builder
。
7.在循环中,依次遍历 exp
中的字符,直到遇到 }
或到达字符串末尾为止。
8.如果当前字符为 {
,则调用 addStringToParts
函数将构建器中的字符串添加到 parts
中,并递归调用 process
函数处理 {}
内部的表达式,将返回的 ans
添加到 parts
中,并更新起始索引 start
。
9.如果当前字符为 ,
,则调用 addStringToParts
函数将构建器中的字符串添加到 parts
中,并将 parts
中的所有集合添加到 ans
中,然后清空 parts
,并更新起始索引 start
。
10.如果当前字符为小写英文字母,则将其添加到构建器中。
11.循环结束后,调用 addStringToParts
函数将构建器中的最后一个字符串添加到 parts
中。
12.调用 addPartsToSet
函数将 parts
中的所有集合添加到 ans
中。
13.返回包含 ans
和起始索引 start
的 Info
对象。
14.addStringToParts
函数将构建器中的字符串添加到 parts
中,如果构建器不为空,则创建一个新的 treeset.Set
对象,并将字符串添加到集合中,再将集合添加到 parts
中。
15.addPartsToSet
函数将 parts
中的所有集合进行组合并添加到 ans
中。
16.processParts
函数是递归处理 parts
切片的核心函数。如果索引 i
等于 parts
的长度,则表示已经处理完所有集合,将连接后的字符串添加到 ans
中。否则,取出当前集合,遍历集合中的每个元素,与 path
进行连接,并递归调用 processParts
处理下一个集合。
17.toSlice
函数将 ans
中的元素转换为有序字符串切片,并返回该切片。
18.在 main
函数中,定义了一个表达式字符串 expression
,并调用 braceExpansionII
函数对表达式进行展开,并将结果打印输出。
该代码的时间复杂度为$O(N^M)$,其中N为表达式中的字符数,M为展开括号的深度。具体来说,代码中的核心函数process
通过遍历表达式字符并进行递归处理,每次递归都会将问题规模缩小,直到达到展开括号的最深层级。因此,时间复杂度取决于表达式中字符的数量以及展开括号的深度。
空间复杂度是$O(NM)$,其中N为表达式中的字符数,M为展开括号的深度。在代码执行过程中,会创建一些辅助数据结构,如字符串构建器和集合。对于集合这种动态数据结构,其占用的内存空间与展开括号的深度呈指数关系。而字符串构建器的空间复杂度与表达式中字符的数量成线性关系。因此,最终的空间复杂度取决于展开括号的深度和表达式中字符的数量,即$O(NM)$。
go完整代码如下:
package main
import (
"fmt"
"strings"
"github.com/emirpasic/gods/sets/treeset"
)
type Info struct {
ans *treeset.Set
end int
}
func NewInfo(a *treeset.Set, e int) *Info {
ans := &Info{}
ans.ans = a
ans.end = e
return ans
}
func braceExpansionII(expression string) []string {
ans := process([]rune(expression), 0).ans
return toSlice(ans)
}
func process(exp []rune, start int) *Info {
ans := treeset.NewWith(func(a, b interface{}) int {
aa := a.(string)
bb := b.(string)
if aa < bb {
return -1
} else if aa == bb {
return 0
} else {
return 1
}
})
parts := make([]*treeset.Set, 0)
builder := strings.Builder{}
for start < len(exp) && exp[start] != '}' {
if exp[start] == '{' {
addStringToParts(&builder, &parts)
next := process(exp, start+1)
parts = append(parts, next.ans)
start = next.end + 1
} else if exp[start] == ',' {
addStringToParts(&builder, &parts)
addPartsToSet(ans, &parts)
start++
parts = make([]*treeset.Set, 0)
} else {
builder.WriteRune(exp[start])
start++
}
}
addStringToParts(&builder, &parts)
addPartsToSet(ans, &parts)
return &Info{ans, start}
}
func addStringToParts(builder *strings.Builder, parts *[]*treeset.Set) {
if builder.Len() != 0 {
s := treeset.NewWithStringComparator()
s.Add(builder.String())
*parts = append(*parts, s)
builder.Reset()
}
}
func addPartsToSet(ans *treeset.Set, parts *[]*treeset.Set) {
processParts(parts, 0, "", ans)
}
func processParts(parts *[]*treeset.Set, i int, path string, ans *treeset.Set) {
if i == len(*parts) {
if path != "" {
ans.Add(path)
}
} else {
part := (*parts)[i]
it := part.Iterator()
for it.Next() {
cur := it.Value().(string)
processParts(parts, i+1, path+cur, ans)
}
}
}
func toSlice(set *treeset.Set) []string {
slice := make([]string, 0)
it := set.Iterator()
for it.Next() {
slice = append(slice, it.Value().(string))
}
return slice
}
func main() {
expression := "{a,b}{c,{d,e}}"
result := braceExpansionII(expression)
fmt.Println(result)
}
2023-07-13:如果你熟悉 Shell 编程,那么一定了解过花括号展开,它可以用来生成任意字符串。 花括号展开的表达式可以看作一个由 花括号、逗号 和 小写英文字母 组成的字符串 定义下面几条语的更多相关文章
- 随机生成长度为len的密码,且包括大写、小写英文字母和数字
一道华三面试题,随机生成长度为len的密码,且包括大写.小写英文字母和数字,主要Random类的使用,random.nextInt(len)表示生成[0,len)整数.具体实现见下面代码,已经很详细了 ...
- 末学者笔记--shell编程上 1 玄
Shell 编程 [内容简列]: 1.shell简介 2. shell分类 3. 查看shell 4. 第一个shell脚本 5. shell编程常用命令 5.1 grep 5.2 cut 5.3 s ...
- [转]Shell编程
原文链接 Shell编程其实真的很简单(一) 如今,不会Linux的程序员都不意思说自己是程序员,而不会shell编程就不能说自己会Linux.说起来似乎shell编程很屌啊,然而不用担心,其实she ...
- shell编程快速入门及实战
shell编程:对于hadoop程序员,通常需要熟悉shell编程,因为shell可以非常方便的运行程序代码. 1.shell文件格式:xxx.sh #!/bin/sh ---shell文件第一行必须 ...
- shell编程其实真的很简单(三)
通过前两篇文章,我们掌握了shell的一些基本写法和变量的使用,以及基本数据类型的运算.那么,本次就将要学习shell的结构化命令了,也就是我们其它编程语言中的条件选择语句及循环语句. 不过,在学习s ...
- shell编程/字库裁剪(2)——编程过程
版权申明:本文为博主窗户(Colin Cai)原创,欢迎转帖.如要转贴,必须注明原文网址 http://www.cnblogs.com/Colin-Cai/p/7704085.html 作者:窗户 Q ...
- 【shell】shell编程(三)-if,select,case语句
通过前两篇文章,我们掌握了shell的一些基本写法和变量的使用,以及基本数据类型的运算.那么,本次就将要学习shell的结构化命令了,也就是我们其它编程语言中的条件选择语句及循环语句. 不过,在学习s ...
- 查找常用字符(给定仅有小写字母组成的字符串数组 A,返回列表中的每个字符串中都显示的全部字符(包括重复字符)组成的列表。例如,如果一个字符在每个字符串中出现 3 次,但不是 4 次,则需要在最终答案中包含该字符 3 次。)
给定仅有小写字母组成的字符串数组 A,返回列表中的每个字符串中都显示的全部字符(包括重复字符)组成的列表. 例如,如果一个字符在每个字符串中出现 3 次,但不是 4 次,则需要在最终答案中包含该字符 ...
- 【Python】【demo实验18】【练习实例】【统计输入字符串中,数字的个数、英文字母的个数及其他符号的个数】
原题: 输入一行字符,分别统计出其中英文字母.空格.数字和其它字符的个数. (本题暂时不支持中文字符及汉字) 我的代码: #!/usr/bin/python # encoding=utf-8 # -* ...
- Day_09【常用API】扩展案例5_获取长度为5的随机字符串,字符串由随机的4个大写英文字母和1个0-9之间(包含0和9)的整数组成
分析以下需求,并用代码实现 1.定义String getStr(char[] chs)方法 功能描述:获取长度为5的随机字符串,字符串由随机的4个大写英文字母和1个0-9之间(包含0和9)的整数组成 ...
随机推荐
- golang 必会之 pprof 监控系列(5) —— cpu 占用率 统计原理
golang pprof 监控系列(5) -- cpu 占用率 统计原理 大家好,我是蓝胖子. 经过前面的几节对pprof的介绍,对pprof统计的原理算是掌握了七八十了,我们对memory,bloc ...
- 会使用ChatGPT写作业找工作会不会加分啊!!!代码问题直接问ChatGPT不比CSDN好多了吗
1.使用reactnative写一个数字游戏 2.MySQL问题
- Pytorch实践模型训练(损失函数)
一个深度学习项目包括了: 模型设计.损失函数设计.梯度更新方法.模型保存和加载和模型训练,其中损失函数就像一把衡量模型学习效果的尺子,训练模型的过程就是优化损失函数的过程,模型拿到数据之后有一个非常重 ...
- 【FAQ】关于JavaScript版本的华为地图服务Map的点击事件与Marker的点击事件存在冲突的解决方案
一. 问题描述 创建地图对象,并添加marker标记,对map和marker均添加了点击事件: <body> <script> function initMap() { // ...
- 基于pyinstaller的python打包工具
以下是软件链接:https://mysecreat.lanzoub.com/iZPGf0swgtbc 软件功能:可以对py文件进行打包,功能基于pyinstaller模块,因此需要安装python环境 ...
- 「学习笔记」2-SAT问题
SAT 是适定性 (Satisfiability) 问题的简称.一般形式为 k - 适定性问题,简称 k-SAT.而当 \(k>2\) 时该问题为 NP 完全的.所以我们只研究 \(k=2\) ...
- Go函数基础
在Go语言中,函数是一种基本的代码组织方式.函数能够接受输入参数并返回结果.Go语言中的函数有以下特点: 函数定义使用关键字func,后跟函数名.参数列表和返回值类型. 如果函数有返回值,则在函数定义 ...
- linux随心记
linux前言 1.计算机有哪两部分组成? 硬件和软件 2.常见的操作系统有哪些? pc端:window ,linux,MacOS 移动端:Android,ios,鸿蒙 3.什么是Linux系统内核 ...
- css设置单行多行超出显示省略号
单行: width: 200px; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; 多行: text-overflow: ...
- express服务器框架
Express 为了提高开发效率,我们在开发过程中,都会尽量使用别人已经开发好的第三方模块,而我们想要快速实现服务器端的开发,推荐一个当下比较流行的框架:Express Express 作为开发框架, ...