打印全排列和stl::next_permutation
打印全排列是个有点挑战的编程问题。STL提供了stl::next_permutation完美的攻克了这个问题。
可是,假设不看stl::next_permutation,尝试自己解决,怎么做?
非常自然地,使用递归的办法:
1. 单个元素的排列仅仅有1个。
2. 多个元素的排列能够转化为:
以每一个元素为排列的首个元素,加上其它元素的排列。
有了思路,就能够编码了。
第一个版本号:
prefix, int set[], int n)
i<n; ++i)
= set[i];
== 1)
" ;
shift set[0,i) to right by 1
j>=0; --j)
shift set[0,i) to left by 1
j<i; ++j)
測试:
myints, 4);
通过。
这样的方法的缺点是产生了大量的string对象。
怎么避免呢?
第二个版本号:
i<n; ++i)
= set[i];
== n-1)
it is possible use callback here instead of printing a permutation
j<n; ++j)
' ;
shift set[from,i) to right by 1
j>=from; --j)
shift set[from,i) to left by 1
j<i; ++j)
測试:
通过。
第二个版本号相比第一个版本号的还有一个改进是能够非常easy地改变成回调函数的形式,扩展函数的用途。而不不过打印排列。
似乎非常不错了。
可是和stl::next_permutation相比,以上的方案就太逊了。
1. stl::next_permutation支持部分排列,而不必是全排列。你能够从不论什么一个排列開始,能够随时退出next_permutation循环。
2. stl::next_permutation支持多重集的排列。比如:
= {1,2,2,2};
' << myints[1] << ' ' <<
myints[2] << ' ' << myints[3]
<< '\n';
std::next_permutation(myints,myints+4) );
输出:
1 2 2 2
2 1 2 2
2 2 1 2
2 2 2 1
没有反复的排列。
stl::next_permutation这么强大,非常值得看看它到底是怎么实现的。
_First, _BidIt _Last)
and test for pure ascending, using operator<
== _Last || _First == --_Next)
; )
find rightmost element smaller than successor
*_Next1))
swap with rightmost element that's smaller, flip suffix
!_DEBUG_LT(*_Next, *--_Mid); )
== _First)
pure descending, flip all
_First, _BidIt _Last)
and test for pure ascending, using operator<
_CHECKED_BASE(_Last));
代码不长,但须要研究才干理解。
非常多算法都是这种。
这个算法能够概括为:
假设仅仅有零个或一个元素,返回false,表示回到全排列的起点。
否则。从右边開始。找到第一个不是递减的元素,即E(i) < E(i+1),从E(i+1)一直到E(n)都是不增的。
假设找到。从右边開始。找到大于E(i)的那个元素E(x)【一定会找到】,交换E(i)和E(x),然后把E[i+1, n]范围内的元素反转。
返回true。
假设找不到,把整个范围内的元素反转,返回false,表示回到全排列的起点。
为什么这个算法可行呢?看以下1 2 3 4的全排列。
能够非常easy地看到,
假设把每一个排列看成一个数,那么下一个排列大于上一个排列。
由上可知,第一个排列是最小排列【不减排列】。最后一个排列是最大排列【不增排列】。
最小排列和最大排列是反序的关系。
算法的关键:从E(i+1)一直到E(n)都是不增的。
这个特性说明,这一范围的元素的排列是一个最大排列,下一个排列必然是找到这一范围内大于这一范围的前一元素的元素,交换这两个元素,交换后E[i+1, n]仍为不增排列【最大排列】。反转之后,变成最小排列。这样处理后得到的排列正好是E[0,n]的下一个排列。
1 2 3 4
1 2 4 3
1 3 2 4
1 2
1 4 2 3
1 4 3 2
2 1 3 4
2 1 4 3
2 3 1 4
2 3 4 1
2 4 1 3
2 4 3 1
3 1 2 4
3 1 4 2
3 2 1 4
3 2 4 1
3 4 1 2
3 4 2 1
4 1 2 3
4 1 3 2
4 2 1 3
4 2 3 1
4 3 1 2
4 3 2 1
打印全排列和stl::next_permutation的更多相关文章
- STL - next_permutation 全排列函数
学习: http://blog.sina.com.cn/s/blog_9f7ea4390101101u.html http://blog.csdn.net/ac_gibson/article/deta ...
- STL next_permutation(a,a+n) 生成一个序列的全排列。满足可重集。
/** 题目: 链接: 题意: 思路: */ #include <iostream> #include <cstdio> #include <vector> #in ...
- STL next_permutation 全排列
调用方法: ]={,,,}; )){ ;i<;i++) printf("%d ",arr[i]); puts(""); } 测试效果: 注:可以看到1 2 ...
- STL next_permutation和prev_permutation函数
利用next_permutation实现全排列升序输出,从尾到头找到第一个可以交换的位置, 直接求到第一个不按升序排列的序列. #include <iostream> #include & ...
- STL next_permutation 算法原理和自行实现
目标 STL中的next_permutation 函数和 prev_permutation 两个函数提供了对于一个特定排列P,求出其后一个排列P+1和前一个排列P-1的功能. 这里我们以next_pe ...
- STL next_permutation 算法原理和实现
转载自:https://www.cnblogs.com/luruiyuan/p/5914909.html 目标 STL中的next_permutation 函数和 prev_permutation 两 ...
- C++ 全排列函数 std::next_permutation与std::prev_permutation
C++ STL中提供了std::next_permutation与std::prev_permutation可以获取数字或者是字符的全排列,其中std::next_permutation提供升序.st ...
- 全排列(STL)
输入一个整数n,输出1~n的全排列(是不是很水) 在此记录stl做法 #include<bits/stdc++.h> using namespace std; ]; int main(){ ...
- C++中全排列算法函数next_permutation的使用方法
首先,先看对next_permutation函数的解释: http://www.cplusplus.com/reference/algorithm/next_permutation/?kw=next_ ...
随机推荐
- 构建基于Javascript的移动CMS——生成博客(二).路由
在有了上部分的基础之后.我们就能够生成一个博客的内容--BlogPosts Detail.这样就完毕了我们这个移动CMS的差点儿基本的功能了,有了上节想必对于我们来说要获取一个文章已经不是一件难的事情 ...
- <LeetCode OJ> 100. Same Tree
100. Same Tree Total Accepted: 100129 Total Submissions: 236623 Difficulty: Easy Given two binary tr ...
- unity3D游戏开发实战原创视频讲座系列9之塔防类游戏开发第一季
解说文件夹 塔防游戏0基础篇... 第一讲 游戏演示和资源介绍... 第二讲 游戏场景的完毕... 第三讲 预制体的制作... 第四讲 敌人的随机产生和按路径行走... 第五讲 塔防工具的产 ...
- 最小生成树基础 (Kruskal)
最小生成树 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit Status ...
- Spark任务调度
不多说,直接上干货! Spark任务调度 DAGScheduler 构建Stage—碰到shuffle就split 记录哪个RDD 或者Stage 输出被物化 重新提交shuffle 输出丢失的sta ...
- jQuery中文学习站点
jQuery是一个快速.简单的Javascript library,它简化了HTML文件的traversing,事件处理.动画.Ajax互动,从而方便了网页制作的快速发展.jQuery是为改变你编写J ...
- Java基础——增强for循环
java1.5版本引入了一个增强for循环,基本原理和for循环类似. 语法声明:for(表达式:条件表达式) 举例:for (String str : set) 解释:set代表set集合,str代 ...
- 动态库连接器–动态库链接信息(Mach-O文件格式和程序从加载到执行过程)
section cmd 说明 举例 __text 主程序代码 __stubs 用于动态库链接的桩 __stub_helper 用于动态库链接的桩 __cstring 常亮字符串符号表描述信 ...
- javascript: 基于原型的面向对象编程
Douglas Crockford指出javascript是世界上最被误解的编程语言.由于javascript缺少常见的面向对象概念,许多程序猿认为javascript不是一个合适的语言.我在做第一个 ...
- OCR文字识别软件FineReader系列产品双十一特惠!
17年的双十一,似乎比以往来的更早一些,说是双十一,这不,从十月里就开始了各种宣传,各种造势,说到底还是影响力太大,关注人群太多,优惠力度太劲爆,简直让人不注意都不行啊!对于ABBYY来说,给用户来点 ...