从n个元素中选择k个的所有组合(包含重复元素)
LeetCode:Combinations这篇博客中给出了不包含重复元素求组合的5种解法。我们在这些解法的基础上修改以支持包含重复元素的情况。对于这种情况,首先肯定要对数组排序,以下不再强调
修改算法1:按照求包含重复元素集合子集的方法LeetCode:Subsets II算法1的解释,我们知道:若当前处理的元素如果在前面出现过m次,那么只有当前组合中包含m个该元素时,才把当前元素加入组合
class Solution {
public:
void combine(vector<int> &vec, int k) {
if(k > vec.size())return;
sort(vec.begin(), vec.end());
vector<int>tmpres;
helper(vec, 0, k, 0, tmpres);
}
//从vec的[start,vec.size()-1]范围内选取k个数,tmpres是当前组合
//times是上一个元素出现的次数
void helper(vector<int> &vec, int start, int k, int times, vector<int> &tmpres)
{
if(vec.size()-start < k)return;
if(k == 0)
{
for(int i = 0; i < tmpres.size(); i++)
cout<<tmpres[i]<<" ";
cout<<endl;
return;
}
if(start == 0 || vec[start] != vec[start-1])//当前元素前面没有出现过
{
//选择vec[start]
tmpres.push_back(vec[start]);
helper(vec, start+1, k-1, 1, tmpres);
tmpres.pop_back();
//不选择vec[start]
helper(vec, start+1, k, 1, tmpres);
}
else//当前元素前面出现过
{
if(tmpres.size() >= times && tmpres[tmpres.size()-times] == vec[start])
{
//只有当tmpres中包含times个vec[start]时,才选择vec[start]
tmpres.push_back(vec[start]);
helper(vec, start+1, k-1, times+1, tmpres);
tmpres.pop_back();
}
helper(vec, start+1, k, times+1, tmpres);
}
}
};
从[1,2,2,3,3,4,5]中选3个的结果如下:

修改算法2:同理,可以得到代码如下 本文地址
class Solution {
public:
void combine(vector<int> &vec, int k) {
if(k > vec.size())return;
sort(vec.begin(), vec.end());
vector<int>tmpres;
helper(vec, 0, k, 0, tmpres);
}
//从vec的[start,vec.size()-1]范围内选取k个数,tmpres是当前组合
//times是上一个元素出现的次数
void helper(vector<int> &vec, int start, int k, int times, vector<int> &tmpres)
{
if(vec.size()-start < k)return;
if(k == 0)
{
for(int i = 0; i < tmpres.size(); i++)
cout<<tmpres[i]<<" ";
cout<<endl;
return;
}
for(int i = start; i <= vec.size()-k; i++)
{
if(i == 0 || vec[i] != vec[i-1])//当前元素前面没有出现过
{
times = 1;
//选择vec[i]
tmpres.push_back(vec[i]);
helper(vec, i+1, k-1, 1, tmpres);
tmpres.pop_back();
}
else//当前元素前面出现过
{
times++;
//vec[i]前面已经出现过times-1次
if(tmpres.size() >= times-1 && tmpres[tmpres.size()-times+1] == vec[i])
{
//只有当tmpres中包含times-1个vec[i]时,才选择vec[i]
tmpres.push_back(vec[i]);
helper(vec, i+1, k-1, times, tmpres);
tmpres.pop_back();
}
}
}
}
};
修改算法3:算法3是根据LeetCode:Subsets 算法2修改未来,同理我们也修改LeetCode:SubsetsII 算法2
class Solution {
public:
void combine(vector<int> &vec, int k) {
if(k > vec.size())return;
sort(vec.begin(), vec.end());
vector<vector<int> > res(1);//开始加入一个空集
int last = vec[0], opResNum = 1;//上一个数字、即将要进行操作的子集数量
for(int i = 0; i < vec.size(); ++i)
{
if(vec[i] != last)
{
last = vec[i];
opResNum = res.size();
}
//如果有重复数字,即将操作的子集的数目和上次相同
int resSize = res.size();
for(int j = resSize-1; j >= resSize - opResNum; j--)
{
res.push_back(res[j]);
res.back().push_back(vec[i]);
if(res.back().size() == k)//找到一个大小为k的组合
{
for(int i = 0; i < res.back().size(); i++)
cout<<res.back()[i]<<" ";
cout<<endl;
}
}
}
}
};
对于算法4和算法5,都是基于二进制思想,这种解法不适用与包含重复元素的情况
【版权声明】转载请注明出处:http://www.cnblogs.com/TenosDoIt/p/3695463.html
从n个元素中选择k个的所有组合(包含重复元素)的更多相关文章
- 【算法30】从数组中选择k组长度为m的子数组,要求其和最小
原题链接:codeforce 267 Div2 C 问题描述: 给定长度为n的数组a[],从中选择k个长度为m的子数组,要求和最大. 形式描述为:选择$k$个子数组[$l_1$, $r_1$], [$ ...
- JS 验证数组中是否包含重复元素
验证JS中是否包含重复元素,有重复返回true:否则返回false 方案一. function isRepeat(data) { var hash = {}; for (var i in data) ...
- [LeetCode] 219. Contains Duplicate II 包含重复元素 II
Given an array of integers and an integer k, find out whether there are two distinct indices i and j ...
- [LeetCode] 220. Contains Duplicate III 包含重复元素 III
Given an array of integers, find out whether there are two distinct indices i and j in the array suc ...
- [LeetCode] 217. Contains Duplicate 包含重复元素
Given an array of integers, find if the array contains any duplicates. Your function should return t ...
- leetcode-219-Contains Duplicate II(使用set来判断长度为k+1的闭区间中有没有重复元素)
题目描述: Given an array of integers and an integer k, find out whether there are two distinct indices i ...
- Java思考——HashSet集合如何保证元素的唯一性也就是不包含重复元素?
首先将源码逐级找出来1.HashSet<String> hs=new HashSet<String>(); hs.add("hello"); ...
- 从N个元素中抽取K个不重复元素(抽奖问题)
核心就是 把N数组抽中的元素给K数组 把N数组最后一位给N数组被抽走的那一位(这时候N数组最后一位元素和被抽走的那位元素值相等) 把N数组长度减一,去除最后一位
- 在程序中对ArrayList进行排序,并剔除重复元素
import java.util.*; class sortDemo { public static void main(String[] args) { ArrayList<Object> ...
随机推荐
- python3 + selenium 使用 JS操作页面滚动条
js2 = "window.scrollTo(0,0);" #括号中为坐标 当不知道需要的滚动的坐标大小时: weizhi2 = driver.find_element_by_id ...
- 【C++ Primer | 15】虚继承
虚基类 一.虚基类介绍 多继承时很容易产生命名冲突,即使我们很小心地将所有类中的成员变量和成员函数都命名为不同的名字,命名冲突依然有可能发生,比如非常经典的菱形继承层次.如下图所示: 类A派生出类B和 ...
- 用两个int值实现读写锁
private int readcount = 0; private int writecount = 0; public void lockread() throws InterruptedExce ...
- hdu 2112 map+Dijkstra
无向图 用map 起点和终点可能一样 数组不能开太大 WA了好多发 Sample Input6xiasha westlake //起点 终点xiasha station 60xiasha Shoppi ...
- #8 //HDU 5730 Shell Necklace(CDQ分治+FFT)
Description 给出长度分别为1~n的珠子,长度为i的珠子有a[i]种,每种珠子有无限个,问用这些珠子串成长度为n的链有多少种方案 题解: dp[i]表示组合成包含i个贝壳的项链的总方案数 转 ...
- 矩阵优化dp
链接:https://www.luogu.org/problemnew/show/P1939 题解: 矩阵优化dp模板题 搞清楚矩阵是怎么乘的构造一下矩阵就很简单了 代码: #include < ...
- Ubuntu/Linux网络配置常用命令
配置ip 打开配置文件 sudo vim /etc/network/interfaces # This file describes the network interfaces available ...
- In Action HDU3339
这是最短路问题和01背包问题的相结合 第一次用01背包 把j打成了i检查了半个小时 下次要注意! 使用的油耗相当于容量 而power相当于价值 先用dijkstra把从基地到所有路的最短情况算出来 ...
- Java Swing 实时刷新JTextArea,以显示不断append的内容?
方法一: 在代码中执行完textArea.append("message")后,如果你想让这个更新立刻显示在界面上而不是等swing的主线程返回后刷新,我们一般会在该语句后调用te ...
- 全排列-hdu1716
题目描述: 题目意思很简单,就是要我们输出全排列后的数据组成,但是要注意组成的数据是一个实数,并且千位数字相同的处在同一行中. 代码实现: #include<stdio.h> #inclu ...