思路一

可以用递推的思想,观察S=[], S =[1], S = [1, 2] 时解的变化。

可以发现S=[1, 2] 的解就是 把S = [1]的所有解末尾添上2,然后再并上S = [1]里面的原有解。因此可以定义vector<vector<int> > 作为返回结果res, 开始时res里什么都没有,第一步放入一个空的vecotr<int>,然后这样迭代n次,每次更新res 内容,最后返回res。

#include<iostream>
#include<vector>
using namespace std; void subset1(int *a, int n)
{
vector<vector<int>> sub(1);// (pow(2, n) + 1);
//这种初始化方式是直接插入了1个为空的元素。此时sub.size()是1。
//扩展容量用resize,提高性能
sub.reserve(pow(2, n) + 1);
for (int i = 0; i < n; ++i)
{
int len = sub.size();
for (int j = 0; j < len; ++j)
{
auto temp = sub[j];
temp.push_back(a[i]);
sub.push_back(temp);
}
}
for (int i = 0; i < sub.size(); ++i)
{
for (int j = 0; j < sub[i].size(); ++j)
{
cout << sub[i][j];
}
cout << endl;
}
}
int main()
{
int a[] = { 1,2,3,4 };
subset1(a, 4);
}

  结果   第一个空行代表空集。

1
2
12
3
13
23
123
4
14
24
124
34
134
234
1234
请按任意键继续. . .

  

思路二

所谓子集,就是包含原集合中的一些元素,不包含另一些元素。如果单独看某一个元素,它都有两种选择:"被包含在子集中"和"不被包含在子集中",对于元素个数为n、且不含重复元素的S,子集总数是2n。因此我们可以遍历S的所有元素,然后用递归考虑每一个元素包含和不包含的两种情况。

代码,这种思路需要用到递归

#include<iostream>
#include<vector>
using namespace std; static vector<vector<int>> sub2;
void subset2(int *a, int n, int t, vector<int> temp)
{
if (t == n)
{
sub2.push_back(temp);
return;
}
subset2(a, n, t + 1, temp);
temp.push_back(a[t]);
subset2(a, n, t + 1, temp);
} int main()
{
int a[] = { 1,2,3,4 };
vector<int> temp; subset2(a, 4, 0, temp);
for (int i = 0; i < sub2.size(); ++i)
{
for (int j = 0; j < sub2[i].size(); ++j)
cout << sub2[i][j];
cout << endl;
}
}

  结果  第一个空行代表空集。

4
3
34
2
24
23
234
1
14
13
134
12
124
123
1234
请按任意键继续. . .

  

思路三

用一个unsigned int 的每个bit位来表示子集的每一个元素的两种情况。前提 原集合中不含重复字符。(含有的话,会出现重复)

#include<iostream>
#include<vector>
using namespace std; static vector<vector<int>> sub2;
void subset3(int *a, int n)
{
unsigned int num = 1 << n;
cout << num << endl;
for (unsigned int i = 0; i < num; ++i)
{
unsigned int q = i;
for (int j = 0; j < n; ++j,q>>=1)
{
if (q & unsigned int ( 1))
{
cout <<a[j];
}
}
cout << endl;
}
} int main()
{
int a[] = { 1,2,3,4 };
subset3(a, 4);
}

  结果  第二个空行代表空集。

16

1
2
12
3
13
23
123
4
14
24
124
34
134
234
1234
请按任意键继续. . .

  

	vector<int> temp2 = temp;
subset2(a, n, sum, t, f + 1, temp);//当前的不包含
temp.push_back(a[f]);
subset2(a, n, sum, t +a[f], f,temp);//可重复的重点
temp2.push_back(a[f + 1]);
subset2(a, n, sum, t + a[f+1],f+1, temp2);//当前的包含

元素可重复,求子集的元素和为一个常数 

#include<iostream>
#include<vector>
#include<set>
using namespace std; static set<vector<int>> sub2;//去重
//a是数组,n数组个数,sum一开始就是0,t是要求的和,f一开始是0
void subset2(int *a, int n,int sum, int t, int f,vector<int> temp)
{
if (t > sum)
return;
if (t == sum)
{
sub2.insert(temp);
return;
}
if (f == n)return;
vector<int> temp2 = temp;
subset2(a, n, sum, t, f + 1, temp);
temp.push_back(a[f]);
subset2(a, n, sum, t +a[f], f,temp);
temp2.push_back(a[f + 1]);
subset2(a, n, sum, t + a[f+1],f+1, temp2); } int main()
{
int a[] = { 4,1,3,2};
vector<int> temp;
subset2(a, 4,7,0,0, temp);
for (auto p = sub2.begin(); p != sub2.end(); ++p)
{
for (int j = 0; j < (*p).size(); ++j)
cout << (*p)[j];
cout << endl;
}
}  

结果:和为7

1111111
111112
11113
11122
1132
1222
133
322
4111
412
43
请按任意键继续. . .

  

subset子集全排序问题的更多相关文章

  1. R语言︱排序问题

    每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 数据排序 1.sort(),rank(),or ...

  2. 第K大子集-LH

    题解:搜索+二分 对于每个数有选与不选两种情况.然后我们先搜前一半的状态,每个数选还是不选. 有2^17种,然后我将每种状态拍一个序先存着.然后我再搜后一半的状态,2^18种. 假设后一半某一种情况的 ...

  3. CSDN-Markdown语法集锦

    前言: 使用Markdown近一个来月.越来越认为不舒爽. 改字体.改字号.改颜色.改样式,全不会!想加个数学公式.得,仅仅会截图.把图片传上去了还不会控制大小.也不会控制文字与图片的排版,写出来的博 ...

  4. Player Settings 导出设置

    Player Settings is where you define various parameters (platform specific) for the final game that y ...

  5. 播放器设置 Player Settings

    原地址:http://game.ceeger.com/Manual/class-PlayerSettings.html#Android Player Settings is where you def ...

  6. Unity导包配置详解

    Player Settings is where you define various parameters (platform specific) for the final game that y ...

  7. words2

    餐具:coffee pot 咖啡壶coffee cup 咖啡杯paper towel 纸巾napkin 餐巾table cloth 桌布tea -pot 茶壶tea set 茶具tea tray 茶盘 ...

  8. 2019-8-31-Latex-公式速查

    title author date CreateTime categories Latex 公式速查 lindexi 2019-08-31 16:55:58 +0800 2018-05-25 16:5 ...

  9. Latex 公式速查

    本文记录了一些常用的数学公式对应的 Latex 字符,用于快速查找需要的字符 所有的在 Latex 使用的字符公式,都需要放在\(和\),$ 和 $,\begin{math} 和\end{math}之 ...

随机推荐

  1. 010. windows10下安装kivy 1.9.1版

    Microsoft Windows [版本 10.0.14393] 以管理员权限打开cmd (c) 2016 Microsoft Corporation. 保留所有权利. 1. C:\Users\LG ...

  2. SQL Server 2008怎么自动备份数据库

    在SQL Server 2008数据库中.为了防止数据的丢失我们就需要按时的来备份数据库了.要是每天都要备份的话,人工备份会很麻烦的,自动备份的话就不需要那么麻烦了,只 要设置好了,数据库就会自动在你 ...

  3. WebSocket实战之JavaScript例子

    一.详细代码案例 详细解读一个简单html5 WebSocket的Js实例教程,附带完整的javascript websocket实例源码,以及实例代码效果演示页面,并对本实例的核心代码进行了深入解读 ...

  4. CentOS7上elasticsearch5.5启动报错

    ERROR: [2] bootstrap checks failed [1]: max file descriptors [4096] for elasticsearch process is too ...

  5. 本博文将一步步带领你实现抽屉官网的各种功能:包括登陆、注册、发送邮箱验证码、登陆验证码、页面登陆验证、发布文章、上传图片、form验证、点赞、评论、文章分页处理以及基于tronado的后端和ajax的前端数据处理。

    本博文将一步步带领你实现抽屉官网的各种功能:包括登陆.注册.发送邮箱验证码.登陆验证码.页面登陆验证.发布文章.上传图片.form验证.点赞.评论.文章分页处理以及基于tronado的后端和ajax的 ...

  6. 无锁的同步策略——CAS操作详解

    目录 1. 从乐观锁和悲观锁谈起 2. CAS详解 2.1 CAS指令 2.3 Java中的CAS指令 2.4 CAS结合失败重试机制进行并发控制 3. CAS操作的优势和劣势 3.1 CAS相比独占 ...

  7. libevent源码深度剖析二

    libevent源码深度剖析二 ——Reactor模式 张亮 前面讲到,整个libevent本身就是一个Reactor,因此本节将专门对Reactor模式进行必要的介绍,并列出libevnet中的几个 ...

  8. mongo_2 $in 和 $all 区别

    in 只需满足( )内的某一个值即可, 而$all 必须满足[ ]内的所有值, > db.table1.find({}); { "_id" : ObjectId(" ...

  9. 【微信公众平台开发】微信JS-SDK开发,信公众平台js-sdk

    根据微信开发文档步骤如下: 1.先登录微信公众平台进入“公众号设置”的“功能设置”里填写“JS接口安全域名”. JS接口安全域名设置 mi.com(前面不用带www/http,域名必须备案过)   2 ...

  10. 13-爬取百度贴吧中的图片(python+xpath)

    通过xpath分析页面,爬取页面中的图片: #_*_ coding: utf-8 _*_ ''' Created on 2018年7月15日 @author: sss function: 使用xpat ...