go语言实现全排列的示例代码

 更新时间:2023年03月07日 10:32:12   作者:大熊的饲养员  
本文主要介绍了go语言实现全排列的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

思路:

  • 首先画出全排列的树形结构,以123为例,一开始排列为空列表,第一个位置有三种可能,分别是1、2、3,画出三个分支;
  • 由于第一个位置已经被占用,那么第二个位置可选择的就只有两个,所以又可以展开两个分支,如:1下的1,2和1,3;
  • 选出两个数字之后,最后就只剩下一个数字了,所以最后一个位置上的数就是唯一确定的了。 之后这个树的所有叶子结点就是全排列的结果。

回溯过程:

  • 先选择1,之后按顺序选择2,最后没有可选数字就得到了1,2,3;为了得到所有的排列,这时候就要进行回溯。
  • 最后一步选择的是3那么回退的时候就要撤回3,回到1,2结点
  • 由于1,2阶段3已经被选择过了,所以继续撤销2,回退到1结点,这个阶段本可以选择2或者3,但是2已经选择过了,所以下一步就要选择3,得到1,3结点,之后再进行刚才的选择回退操作

这个树除了叶子结点以外,其他结点做的事情都是一样的,也就是在已经选了数 的前提下需要在剩下还没有选择的数里,按照顺序选择一棵树,所以这就是一个递归。那么递归终止的条件就是数字的个数已经选完了。所以我们需要一个变量来记录已经选了多少个数字,其实这个变量等价递归到了第几层depth,当遍历的层数和输入数组的个数相等的时候,所有的元素就都被考虑完了,就可以退出递归。

将已经选择的数放进一个列表里temp,这个其实就是树的路径,因为要不断地添加删除所以这个应该是个栈。在设置一个布尔数组used表示当前已经考虑的数字是否在之前

已经选择过,也就是判断是否在path变量里,初始化都为FALSE,表示都未被选择。

代码:

package main

func main() {
}
func permute(nums []int) [][]int  {
    //保存输入数组的长度
    nlen := len(nums)
    //初始化,用来存放结果
    var result [][]int
    //如果传入长度为0,那就直接返回空数组(要对空列表进行初始化)
    if nlen==0{
        return result
    }
    //创建中间变量,存放临时结果
    var temp []int
    //创建bool值,判断该位置数字是否用过
    used := make([]bool, nlen)
    //回溯函数
    BackTrack(used, temp, nums, &result,nlen,0)
    return result
}
func BackTrack(used []bool, temp []int, nums []int, result *[][]int,nlen int,depth int) {
    //判断回溯函数结束条件
    //当临时temp长度和所给的数字长度相等时(也就是递归到了第几层),将该temp加入结果
    if depth == nlen {
        //由于go语言的特性如果不特别说明创建的切片本质上都是指向同一个内存空间
        //如果想要循环赋值的切片与原来切片不相关,需要另外开辟空间,这里用到copy函数,开辟独立空间
        current := make([]int, depth)
        copy(current, temp)
        *result = append(*result, current)
    }
    //遍历数组中的数字,进行排列组合
    for i := 0; i < nlen; i++ {
        //减枝,当该位置数字使用过时则跳过
        if used[i] {
            continue
        }
        //没有使用过就添加数字
        temp = append(temp, nums[i])
        //将该位置数字设置为访问过的状态
        used[i] = true
        //递归继续搜索该支线
        BackTrack(used, temp, nums, result,nlen,depth+1)
        //回溯,恢复到之前的状态
        temp = temp[:len(temp)-1]
        used[i] = false
    }
}

到此这篇关于go语言实现全排列的示例代码的文章就介绍到这了,更多相关go语言全排列内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • golang变量uint、int大小溢出后的结果方式

    golang变量uint、int大小溢出后的结果方式

    在Go语言中,变量的大小溢出后,`uint`类型会回绕到最小值,而`int`类型会回绕到最大值的相反数,例如,`uint8`溢出后会变成0,`int64`溢出后会变成最小的负数
    2024-12-12
  • go中的参数传递是值传递还是引用传递的实现

    go中的参数传递是值传递还是引用传递的实现

    参数传递机制是一个重要的概念,它决定了函数内部对参数的修改是否会影响到原始数据,本文主要介绍了go中的参数传递是值传递还是引用传递的实现,感兴趣的可以了解一下
    2024-12-12
  • Golang分布式锁简单案例实现流程

    Golang分布式锁简单案例实现流程

    分布式锁是控制分布式系统之间同步访问共享资源的一种方式。如果不同的系统或是同一个系统的不同主机之间共享了一个或一组资源,那么访问这些资源时,需要通过一些互斥手段来防止彼此之间的干扰以保证一致性,在这种情况下,就需要使用分布式锁了
    2022-12-12
  • Go语言共享内存读写实例分析

    Go语言共享内存读写实例分析

    这篇文章主要介绍了Go语言共享内存读写方法,实例分析了共享内存的原理与读写技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-02-02
  • Go语言入门之函数的定义与使用

    Go语言入门之函数的定义与使用

    函数是一段代码的片段,包含连续的执行语句,它可以将零个或多个输入参数映射到零个或多个参数输出。本文将通过示例和大家详细聊聊Go语言中函数的定义与使用,感兴趣的可以了解一下
    2022-11-11
  • Golang结合ip2region实现ip归属地查询

    Golang结合ip2region实现ip归属地查询

    ip2region - 是一个离线IP地址定位库和IP定位数据管理框架,提供了众多主流编程语言的 xdb 数据生成和查询客户端实现,下面我们就来看看Golang如何结合ip2region实现ip归属地查询吧
    2024-03-03
  • Go并发编程sync.Cond的具体使用

    Go并发编程sync.Cond的具体使用

    Go 标准库提供 Cond 原语的目的是,为等待 / 通知场景下的并发问题提供支持,本文主要介绍了Go并发编程sync.Cond的具体使用,具有一定的参考价值,感兴趣的可以了解一下
    2022-05-05
  • Go语言基础单元测试与性能测试示例详解

    Go语言基础单元测试与性能测试示例详解

    这篇文章主要为大家介绍了Go语言基础单元测试与性能测试示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助祝大家多多进步
    2021-11-11
  • Golang使用lua脚本实现redis原子操作

    Golang使用lua脚本实现redis原子操作

    这篇文章主要介绍了Golang使用lua脚本实现redis原子操作,本文通过实例代码给大家介绍的非常详细,对大家的工作或学习具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-03-03
  • Golang利用Template模板动态生成文本

    Golang利用Template模板动态生成文本

    Go语言中的Go Template是一种用于生成文本输出的简单而强大的模板引擎,它提供了一种灵活的方式来生成各种格式的文本,下面我们就来看看具体如何使用Template实现动态文本生成吧
    2023-09-09

最新评论