非负整数可重集去重&排序+获得可重集的全排列的几种方法
非负整数可重集O(n)去重并排序
可重集是指元素可重复的集合,对于在一定区间内的正整数集,比如[1,n],我们可以在不不使用任何额外空间(包括不使用O(1)的空间)的情况下,用O(n)的时间复杂度完成集合的去重并排序,这种O(n)的算法,是理想的联机算法。
思路:本质上和桶排序类似,用数组下标来表示存在的元素,数组中的元素作为flag。
对于正整数可重集来说,打标记的方法可以是将元素变负(思考,为什么不是随便赋一个规定的值),负整数依次类推。
对于元素属于[1,n]的集合(n为元素个数),我们可以用下面的代码完成上述操作并取出元素,总时间是O(2n)
void removeDuplicates(int *a,int len){
for(int i=;i<len;i++)
a[abs(a[i])-] = -abs(a[abs(a[i])-]);//如果元素a[i]存在,则将a[abs(a[i])-1] 变负,下标0 ~ n-1 对应1 ~ n
for(int i=;i<len;i++)if(a[i]<)//如果a[i]<0 ,则说明i+1存在 ,取出
printf("%d ",i+);
}
int main(){
int seq[] = {,,,,,,,,};
removeDuplicates(seq,sizeof(seq)/sizeof(int));
return ;
}
获得可重集全排列:
自己玩:
可重集是指元素可重复的集合,可重集的全排列通常可以递归地进行求解。
对于n个元素不重复的集合来说,我们可以递归为:
- 将第k个元素(k=1,2...n)放到集合首部
- 求解剩下n-k个元素的集合的全排列
- 重复1和2,直到集合的元素为空时,打印整个集合
实现的代码(此处是以字符串为例),其中len表示字符串s的长度。注意,这里s必须定义为数组,如果定义为指针,将会引发错误,具体请看我的另一篇博客:C++指针和数组的区别中的情况2
void swap(char &i,char &j){
char t=i;i=j;j=t;
}
void permutation(char s[],int left,int len){
if(left==len)printf("%s\n",s);
else{
for(int k=left;k<len;k++){
if(s[left]!=s[k])swap(s[left],s[k]);
//递归求解n-k个元素的全排列
permutation(s,left+,len);
if(s[left]!=s[k])swap(s[left],s[k]);
}
}
}
需要注意:这种实现不是遵从字典序的实现
当我们需要打印可重集的全排列时,我们只需对递归调用的部分稍作改动
- 重复的情况要保证出现,所以,当left==k的时候,代表第一次递归,此时,应当保留
- 除了1之外,如果s[left]和s[k]仍有相等情况,则不应交换和递归,因为此时若递归,会造成重复
简单修改上述代码,实现如下:
void swap(char &i,char &j){
char t=i;i=j;j=t;
}
void permutation(char s[],int left,int len){
if(left==len)printf("%s\n",s);
else{
for(int k=left;k<len;k++){
//增加了上文的两个判定条件
if(k==left||s[left]!=s[k]){
swap(s[left],s[k]);
permutation(s,left+,len);
swap(s[left],s[k]);
}
}
}
}
同样,这种实现不是遵从字典序的实现。
当然,我们很多时候都需要按照字典序进行排列,说实话,我觉得我是很讨厌手写这个的,毕竟相当的麻烦,所以,就有了下面这个:
黑科技:STL中的next_permutation(s,s+n)
#include<algorithm>
using namespace std;
void permutation(char s[],int len){
sort(s,s+len);//一定要先排序
do{
puts(s);
}while(next_permutation(s,s+len));
}
这是货真价实的字典序的全排列,今天就到这,拜拜~
非负整数可重集去重&排序+获得可重集的全排列的几种方法的更多相关文章
- java中的排序除了冒泡以来, 再给出一种方法, 举例说明
9.5 排序: 有一种排序的方法,非常好理解,详见本题的步骤,先找出最大值和最小值,把最小值打印出来后,把它存在另一个数组b当中,再删除此最小值,之后再来一次找出最小值,打印出最小值以后,再把它存 ...
- 并查集+拓扑排序 赛码 1009 Exploration
题目传送门 /* 题意:无向图和有向图的混合图判环: 官方题解:首先对于所有的无向边,我们使用并查集将两边的点并起来,若一条边未合并之前, 两端的点已经处于同一个集合了,那么说明必定存在可行的环(因为 ...
- List集合对象去重及按属性去重的8种方法-java基础总结系列第六篇
最近在写一些关于java基础的文章,但是我又不想按照教科书的方式去写知识点的文章,因为意义不大.基础知识太多了,如何将这些知识归纳总结,总结出优缺点或者是使用场景才是对知识的升华.所以我更想把java ...
- 分享一种容易理解的js去重排序方法
<script> var arr=[1,8,6,4,88,22,99,4,6,86,5,58,89,5]; //先使用sort()函数去重 var a=arr.sort(function ...
- linux下批量修改存有超大数据量IP文件中的IP内容以及去重排序
作为一个linux的学徒,分享一下自己解决这个小问题的心得,在处理这个问题时使用了一个小技巧感觉很适用,个人发觉linux的终端真滴是非常强大,下面就详细地介绍这个问题以及解决办法吧 问题描述:由于要 ...
- onethink对二维数组结果集进行排序
<?php /** * 对查询结果集进行排序 * @access public * @param array $list 查询结果 * @param string $field 排序的字段名 * ...
- Kettle的集群排序 2——(基于Windows)
5.使用kettle集群模式对相关的数据进行排序 既然,基于Carte服务程序所搭建的集群已经在Spoon中设定好了, 可以首先,先来启动四个节点: "以管理员身份运行"打开 四个 ...
- 在使用Kettle的集群排序中 Carte的设定——(基于Windows)
本片文章主要是关于使用Kettle的UI界面: Spoon来实现基于集群的对数据库中的数据表数据进行排序的试验. 以及在实验过程中所要开启的Carte服务的一些配置文件的设置, 还有基于Windows ...
- for循环去重排序
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
随机推荐
- P1268 树的重量
题目描述 树可以用来表示物种之间的进化关系.一棵“进化树”是一个带边权的树,其叶节点表示一个物种,两个叶节点之间的距离表示两个物种的差异.现在,一个重要的问题是,根据物种之间的距离,重构相应的“进化树 ...
- BZOJ5340 [Ctsc2018]假面 【概率dp】
题目链接 BZOJ5340 题解 我们能很容易维护每个人当前各种血量的概率 设\(p[u][i]\)表示\(u\)号人血量为\(i\)的概率 每次攻击的时候,讨论一下击中不击中即可转移 是\(O(Qm ...
- Elasticsearch 5.2.1Cluster 搭建
1.安装java cd ~ wget --no-cookies --no-check-certificate --header "Cookie: gpw_e24=http%3A%2F%2Fw ...
- 关于GitHub学习的地方,很明了
地址: http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000
- (转)tableview的索引设置
.感觉tableview的索引条将表视图往左边挤了一点?别担心,只是颜色问题.只要如此设置即可 //索引条背景的颜色(清空颜色就不会感觉索引条将tableview往左边挤) [_tableView s ...
- shell分发文件脚本
配置文件scp.conf ssh_hosts=("IP") #需要分发机器的所有IP ssh_ports=("22") ssh_users=("roo ...
- 【poj1743-Musical Theme】不可重叠最长重复子串-后缀数组
http://poj.org/problem?id=1743 这题是一道后缀数组的经典例题:求不可重叠最长重复子串. 题意: 有N(1 <= N <=20000)个音符的序列来表示一首乐曲 ...
- loj6029 「雅礼集训 2017 Day1」市场
传送门:https://loj.ac/problem/6029 [题解] 考虑如果有一些近似连续的段 比如 2 2 2 3 3 3,考虑在除3意义下,变成0 0 0 1 1 1,相当于整体-2 又:区 ...
- 标签 JLable 类
标签JLable上可以添加图像,当鼠标停留在标签上时,可以显示一段提示文字. package first; import javax.swing.*; import java.awt.*; impor ...
- mongoDB文档操作【增删改】
MongoDB 插入文档 文档的数据结构和JSON基本一样. 所有存储在集合中的数据都是BSON格式. BSON是一种类json的一种二进制形式的存储格式,简称Binary JSON. 插入文档 Mo ...