https://blog.csdn.net/so_geili/article/details/71078945
全排列算法难度适中,既可以递归实现,又能用非递归的实现,可以考察解决问题的基本能力。在校园招聘的笔试和面试中经常出现。
1.排列问题描述
有nn个元素,分别编号为1、2、3...、n1、2、3...、n。排列问题就是要生成这nn个元素的全排列。假定这nn个元素已按编号次序由小到大存放着数组AA中,可以认为数组中存放的就是元素的编号。
例如,123的全排列有123、132、213、231、312、321这六种。
例如,abc 的全排列有abc、acb、bca、dac、 cab、cba这六种。
2.算法的分析 和 设计
只有找到生成元素全排列的规律,才能找到递归关系。分析递归关系,找到递归出口和递归关系式,最终才能设计出递归的函数实现。
递归算法设计如下:
【1】将规模为nn的全排列问题转化为规模为n−1n−1的全排列问题。
基本思想就是:任意选一个数(一般从小到大或者从左到右)打头,对后面的n-1个数进行全排列。
例如:(A、B、C、D)的全排列为
A后面跟(B、C、D)的全排列
B后面跟(A、C、D)的全排列
C后面跟(A、B、D)的全排列
D后面跟(A、B、C)的全排列
【2】将规模为n−1n−1的全排列问题转化为规模为n−2n−2的全排列问题。实现方法类似【1】。
【3】将规模为n−2n−2的全排列问题转化为规模为n−3n−3的全排列问题。实现方法类似【1】。
以此类推….
最终,问题的规模一级一级降至1,11,1个元素的全排列就是它本身,这也就是递归出口。
3.递归算法描述和C实现
假定全排列算法为:void permutation(char *A,int k,int n)。生成数组A[n]A[n]后面kk个元素的全排列。
当k==1k==1时,是递归出口,仅对最后一个元素全排列,最后一个元素的全排列就是它本身,此时仅需打印整个数组A[n]A[n]。
当1<k<=n1<k<=n时,按照上述的分析,应减小算法的规模,调用permutation(A, k-1, n),生成后面k−1k−1个元素的全排列。然后需要将数组A[n−k]A[n−k]元素和数组A[n−k,n−1]A[n−k,n−1]中的元素逐一互换,互换后执行permutation(A, k-1, n)。
#include <stdio.h>
#include <string.h>
int cnt=0;//计数器
void swap(char *a,char *b)
{
char tmp;
tmp=*a;
*a=*b;
*b=tmp;
}
void permutation(char *A,int k,int n)
{
int i;
if(k==1)//递归出口,此时打印整个数组。
{
cnt++;
printf("第%2d种排列:",cnt);
for(i=0;i<n;i++)
printf("%c ",A[i]);
printf("\n");
}
else
{
for(int j=n-k;j<n;j++)
{
swap(&A[n-k],&A[j]);
permutation(A,k-1,n);
swap(&A[n-k],&A[j]);
}
}
}
int main()
{
char ch[] = "ABCD";
permutation(ch,strlen(ch),strlen(ch));
return 0;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
执行结果如下:
这样的方法没有考虑到重复元素的情况,如生成”ABB”全排列时:
======================
4.带重复元素的全排列的递归实现
由于上述的全排列,在递归时就是从第一个元素起分别与它后面的元素逐一交换(没考虑重复元素)。所以当带重复元素时,这种方法就行不通了。
例如:对122全排列时,第一个元素1与第二个元素2交换得到212;然后第一个元素1与第三个元素2交换,同样得到212,这种情况应该避免发生。所以在第一个元素与第三个元素交换之前,一定要检查第三个元素之前是否出现过,如果没有出现则执行交换;如果出现过则不进行交换。
所以在全排列算法中,在递归实现时,应从第一个元素起分别与它后面的非重复元素交换。在编程实现时,在交换第i个元素与第j个元素之前,要求数组的[i+1,j)[i+1,j)区间中的元素没有与第j个元素重复。下面给出完整代码:
#include <stdio.h>
#include <string.h>
typedef char bool;
#define false 0
#define true 1
int cnt=0;//计数器
void swap(char *a,char *b)
{
char tmp;
tmp=*a;
*a=*b;
*b=tmp;
}
//检查[from,to)之间的元素和第to号元素是否相同
bool isRepeat(char *A,int from,int to)
{
bool flag=false;//初始为false,代表不重复元素
for(int i=from;i<to;i++)
{
if(A[to]==A[i])
{
flag=true;
return flag;
}
}
return flag;
}
void permutation(char *A,int k,int n)
{
int i;
if(k==1)//递归出口,此时打印整个数组。
{
cnt++;
printf("第%2d种排列:",cnt);
for(i=0;i<n;i++)
printf("%c ",A[i]);
printf("\n");
}
else
{
for(int j=n-k;j<n;j++)
{
if(!isRepeat(A,n-k,j))
{
swap(&A[n-k],&A[j]);
permutation(A,k-1,n);
swap(&A[n-k],&A[j]);
}
}
}
}
int main()
{
char ch[] = "122";
permutation(ch,strlen(ch),strlen(ch));
return 0;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
执行结果如下:
======================
5.全排列的非递归实现(字典序)
非递归实现,也就是通常所说的字典序全排列算法。
首先,必须要理解字典序(dictionary order)。
第一种理解方式,按照纸质版字典的方式理解。第二种按照C语言中strcmp()函数的比较原理理解字典序。
例如:“abc”<“abcd”<“abde”<“afab”。
其次,如何求某一个排列紧邻着的后一个字典序。
设P是1~n的一个全排列:PP=p1p2......pnp1p2......pn=p1p2......pj−1pjpj+1......pk−1pkpk+1......pnp1p2......pj−1pjpj+1......pk−1pkpk+1......pn,求该排列的紧邻着的后一个字典序的步骤如下:
从排列的右端开始,找出第一个比右边数字小的数字的序号jj,即 j=max(i|pi<pi+1)j=max(i|pi<pi+1)
在pjpj的右边的数字中,找出所有比pjpj大的数中最小的数字pkpk
对换pjpj和pkpk
再将pj+1......pk−1pkpk+1...pnpj+1......pk−1pkpk+1...pn倒转得到排列p’=p1.....pjpn.....pk+1pkpk−1.....pj+1p1.....pjpn.....pk+1pkpk−1.....pj+1,这就是排列p的下一个排列。
例如:p=839647521是数字1~9的一个排列。下面生成下一个排列的步骤如下:
自右至左找出排列中第一个比右边数字小的数字4
在该数字后的数字中找出比4大的数中最小的一个5
将5与4交换,得到839657421
将7421反转,得到839651247。这就是排列p的下一个排列。
以上这4步,可以归纳为:一找、二找、三交换、四翻转。和C++的 STL库中的next_permutation()函数(#include<algorithm>#include<algorithm>)原理相同。
C代码实现如下:
#include <stdio.h>
#include <string.h>
typedef char bool;
#define false 0
#define true 1
int cnt=0;//计数器
void swap(char *a,char *b) {
char tmp;
tmp=*a;
*a=*b;
*b=tmp;
}
//冒泡升序排序
void BubbleSort(int arr[],int length) {
int tmp;
int i,j;
for(i=length-1; i>0; i--) {
for(j=0; j<i; j++) {
if(arr[j] > arr[j+1]) {
tmp=arr[j];
arr[j]=arr[j+1];
arr[j+1]=tmp;
}
}
}
}
void PrintArray(int arr[],int length) { //打印数组,用于查看排序效果
printf("第%2d种排列为:[",++cnt);
for(int i=0; i<length; i++) {
if(i==length-1)
printf("%d]\n",arr[i]);
else
printf("%d ,",arr[i]);
}
}
void reverse(int arr[],int from, int to) {//反转[from,to]
while(from < to) {
char tmp = arr[from];
arr[from]= arr[to];
arr[to] = tmp;
from ++;
to --;
}
}
bool Next_Permutation(int A[], int n) {
int i,j,m;
for(i = n-2; i >= 0; i--) { //一、找:
if(A[i+1] > A[i])
break;
}
if(i < 0)
return false;//找不到,返回false
m = i;
i++;
for( j=n-1; j>i; j--) { //二、找
if(A[j] >= A[m])
break;
}
swap(&A[j],&A[m]); //三、交换
reverse(A,m+1,n-1); //四、反转
return true;
}
void Full_Permutation(int A[],int n) {
if(A==NULL || n<=0) return; //健壮性检测
BubbleSort(A,n); //冒泡排序(升序)
do {
PrintArray(A,n);
} while(Next_Permutation(A,n));//检测是否还有下一个字典序排列
}
int main() {
int arr[]= {3,2,4,1};
Full_Permutation(arr,sizeof(arr)/sizeof(int));
return 0;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
执行的结果为:
当测试数据中,含有重复元素时,仍然可以得到正确的全排列:
======================
6.递归算法时间复杂度分析
全排列的递推关系式是:
T(n)={O(1),nT(n−1),当n=1当n>=2
T(n)={O(1),当n=1nT(n−1),当n>=2
所以不难算出,递归方式的全排列算法时间复杂度为O(n!)O(n!)。
注意:因为全排列一共有n!n!种,所以无论是递归实现,还是非递归实现,时间复杂度都是O(n!)O(n!)。如果要对全排列进行输出,那么输出的时间要O(n∗n!)O(n∗n!)。
---------------------
作者:GNG
来源:CSDN
原文:https://blog.csdn.net/so_geili/article/details/71078945
版权声明:本文为博主原创文章,转载请附上博文链接!
- 程序优化
- vtune
- linux性能监控软件Perf
- 系统级性能分析工具perf的介绍与使用
- perf的二级命令
- 全局性概况
- 全局细节
- 最常用功能perf record
- 可视化工具perf timechart
- perf引入的overhead
- perf stat
- gprof
- 三种Linux性能分析工具的比较
- perf+gprof+gprof2dot+graphviz进行性能分析热点
- 英特尔多核平台编程优化大赛报告
- 内存操作
- mmap
- mmap的分类
- 深入理解内存映射mmap
- 计算机底层知识拾遗(九)深入理解内存映射mmap
- 内核驱动mmap Handler利用技术(一)
- Windows内存管理机制及C++内存分配实例
- Linux内存管理初探
- Windows CPU信息查看
- Linux CPU信息查看
- 预留大内存
- Linux下试验大页面映射
- /dev/mem
- Linux中通过/dev/mem操控物理地址
- /dev/mem分析
- 用法举例
- Linux下直接读写物理地址内存
- 查看内存信息
- Cache Memory
- 页面缓存
- 查看各级cache信息的方法
- dmidecode命令查看cache size
- CPU Cache 机制以及 Cache miss
- ARM体系关闭mmu和cache
- CR0-4寄存器介绍
- 查看CR0,CR2,CR3的值
- Linux 下如何禁用CPU cache
- 7个示例科普CPU Cache
- 第一个例子的C代码
- 其中之一
- Linux 从虚拟地址到物理地址
- 内存测试例子
- 每个程序员都应该了解的内存
- Part 1
- 程序员能够做什么
- 3 CPU caches
- 6 What Programmers Can Do
- VirtualAlloc
- Large-Page Support
- Some remarks on VirtualAlloc and MEM_LARGE_PAGES
- DMA
- MOV和MOVS的效率问题?如何高效的拷贝内存 中的数据
- how to use movntdqa to avoid cache pollution
- 计算机底层知识拾遗(一)理解虚拟内存机制
- How to access the control registers cr0,cr2,cr3 from a program
- 细说Cache-L1/L2/L3/TLB
- what-is-the-meaning-of-non-temporal-memory-accesses-in-x86
- How can the L1, L2, L3 CPU caches be turned off on modern x86/amd64 chips?
- UA list
- GDB
- 程序运行参数
- Linux下GDB的多线程调试
- CMake
- CMake快速入门教程:实战
- cmake打印变量值
- function
- source_group
- cmake_parse_arguments
- 编译.S文件
- add_definitions
- CMake添加-g编译选项
- Debug模式下启动
- Mysql
- Mysql联合查询union和union all的使用介绍
- MySQL数据库导入错误:ERROR 1064 (42000) 和 ERROR at line xx: Unknown command '\Z'.
- 解决MYSQL数据库 Table ‘xxx’ is marked as crashed and should be repaired 145错误
- C/C++
- c语言中static的作用
- strlen和sizeof有什么区别?
- printf
- Libuv中文文档之线程
- RapidJSON
- gcc/g++ 实战之编译的四个过程
- __thread
- TARGET_LINK_LIBRARIES
- MAP_HUGETLB
- 使用Intel格式的汇编
- __m128i
- emmintrin.h
- _mm_stream_si128
- _mm_stream_load_si128
- _mm_load_si128
- _mm_xor_si128
- _mm_store_si128
- _mm_cvtsi128_si64
- Intel SSE指令集
- _mm_set_epi64x
- _mm_aesenc_si128
- _umul128
- _mm_malloc
- reinterpret_cast
- strlen
- 读取UTF-8的txt文件发现开头的多三个字节的问题
- PHP
- php计算函数执行时间的方法
- 框架
- Json Rpc远程调用框架
- PHP多进程
- PHP CLI模式下的多进程应用
- php多进程总结
- 优化
- PHP7 优化
- 让你的PHP7更快(GCC PGO)
- PHP的性能演进(从PHP5.0到PHP7.1的性能全评测)
- PHP字符串全排列算法
- 获取服务器基本信息
- cookie
- phpstudy2018 安装xdebug扩展
- 软件下载
- PHP mysqli_error() 函数
- PHP Session 变量
- curl
- curl_getinfo
- 获取请求头
- PHP使用CURL获取302跳转后的地址实例
- PHP基于cURL实现自动模拟登录
- PHP获取远程图片大小(CURL实现)
- CURL模拟登录
- curl模拟登录提交(从目录中获取文件)
- CURL HTTPS
- curl帮v
- rename
- copy
- JSON
- json_encode
- json_decode
- json_last_error_msg
- json_last_error
- PHP json_encode中文乱码解决方法
- var_dump
- PHPStorm与Xdebug设置
- Xdebug原理以notepad为例
- str_pad
- pack
- PHP二进制与字符串之间的相互转换
- PHP执行系统命令(简介及方法)
- 函数
- 十进制转二进制
- 字符串到ASSCI
- 字符串转二进制
- 合并两个表
- 图像识别
- Tesseract
- 虚拟机
- vmware下Kali 2.0安装VMware Tools
- 安装 VMware tools出现“正在进行简易安装时,无法手动启动VMware tools安装”
- 爬虫
- 有哪些好的数据来源或者大数据平台?
- Cygwin
- Git 常用命令
- 排列组合
- 含重复元素序列的全排列
- 全排列的非递归和递归实现(含重复元素)
- GitBook
- 编辑环境
- visual studio code
- 2名数学家或发现史上最快超大乘法运算法,欲破解困扰人类近半个世纪的问题
- 系统预定义常量
- 指令集
- SSE
- _MSC_VER
- msys2
- 安装cmake
- MSYS2 更新源
- 讲Cmake msys32使用问题解答 CXX CMAKE_C_COMPILER配置详解
- VirtualBox
- 解决virtualbox只能安装32位系统的问题
- Ubuntu
- 使用AES-NI的编译参数
- debian下安装内核源码的方法
- tar.xz结尾的文件的解压方法
- Linux命令
- insmod
- fatal error: openssl/bio.h
- 准备module的编译环境(kali)
- Ubuntu/Debian 之内核模块开发准备
- dmesg的详细用法
- Linux系统开机自动加载驱动module
- linux /Module 浅析(转载)
- Kali
- 找回gpedit
- Enable the Lock Pages in Memory Option (Windows)
- TLA
- 双系统
- 显卡
- 显示no CUDA的解决过程
