非负整数可重集去重&排序+获得可重集的全排列的几种方法
非负整数可重集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/ ...
随机推荐
- [CF15C]Industrial Nim
题目大意:有$n$个采石场,每行一个$m_i$一个$x_i$,表示第$i$个采石场有$m_i$辆车,这个采石场中车中的石子为从$x_i$开始的自然数.Nim游戏若先手赢输出"tolik&qu ...
- 01、BUCK电路的参数计算
案例:设计一个Buck电路,满足如下性能指标要求:一.性能指标要求 1.输入电压 2.输出电压 3.输出电压纹波 4.电流纹波 5.开关频率 二.需要计算的参数 三.BUCK电路拓扑 ...
- [Leetcode] container with most water 最大水容器
Given n non-negative integers a1 , a2 , ..., an , where each represents a point at coordinate (i, ai ...
- number 解题报告
number 题目描述 给定整数 \(m,k\),求出正整数 \(n\) 使得 \(n+1,n+2,-,2n\) 中恰好有 \(m\) 个数在二进制下恰好有 \(k\) 个 \(1\). 有多组数据. ...
- Walk 解题报告
Walk 题目描述 给定一棵 \(n\) 个节点的树,每条边的长度为 \(1\),同时有一个权值\(w\).定义一条路径的权值为路径上所有边的权值的最大公约数.现在对于任意 \(i \in [1,n] ...
- 【POJ 1201 Intervals】
Time Limit: 2000MSMeamory Limit: 65536K Total Submissions: 27949Accepted: 10764 Description You are ...
- 东北育才冲刺noip(day9)
这十天来呢,感觉自己进步很大,(虽然被碾压的很惨),看到了自己以前完全没见过,也没想过的算法,打开新世界的大门. 同时呢,也感觉自己太弱了,于是就注册了这个博客. 为了促进进步,在这里立下flag,我 ...
- MySQL使用笔记(三)表的操作
By francis_hao Dec 11,2016 表的操作 表的操作有创建表.查看表.删除表和修改表 创建表 创建表之前要在某个数据库中. mysql> create table ta ...
- [hdu 4734]数位dp例题
通过这个题目更加深入了解到了数位dp在记忆化搜索的过程中就是实现了没有限制条件的n位数的状态复用. #include<bits/stdc++.h> using namespace std; ...
- codeforces 792CDivide by Three(两种方法:模拟、动态规划
传送门:https://codeforces.com/problemset/problem/792/C 题意:给你一个字符串,要求让你删除最少个数的元素,使得最终答案是没有前导0并且是3的倍数. 题解 ...