Java求子集问题

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


前言

提示:这里可以添加本文要记录的大概内容:

个人记录参加蓝桥杯的算法学习


提示:以下是本篇文章正文内容,下面案例可供参考

一、子集的概念

给出一个集合,集合里有n个元素,则有 2^n 子集(含空)

二、这里给出三个方法

1.递归

代码如下(示例):

/**
	 * 
	 * @param A 数组
	 * @param cur 当前元素位置的下标
	 * @return 返回一个嵌套的集合
	 */
	private Set<Set<Integer>> getsubset3(int[] A, int cur) {
		Set<Set<Integer>> newres = new HashSet<>();
		if (cur == 0) {
			Set<Integer> nil = new HashSet<>();
			Set<Integer> first = new HashSet<>();
			first.add(A[0]);
			newres.add(nil);
			newres.add(first);
			return newres;
		}
		Set<Set<Integer>> oldres = getsubset3(A, n, cur-1);
		for (Set<Integer> set : oldres) {
			newres.add(set);
			Set<Integer> copy = (Set<Integer>)(((HashSet)set).clone());
			copy.add(A[cur-1]);
			newres.add(copy);
			
		}		
		return newres;
	}

2.迭代

代码如下(示例):

/**
	 * 
	 * @param A 数组
	 * @return
	 */
	private Set<Set<Integer>> getsubset2(int[] A) {
		Set<Set<Integer>> res = new HashSet<>();
		Set<Integer> nil = new HashSet<>();
		res.add(nil);
		for (int i = 0; i < A.length; i++) {
			
			Set<Set<Integer>> help = new HashSet<>();
			help.addAll(res);
			for (Set<Integer> set : res) {
				Set a = (HashSet)(((HashSet)set).clone());
				a.add(A[i]);
				help.add(a);
			}
			res = help;
		}
		return res;	
	}

3.二进制法

代码如下(示例):

/**
	 * 
	 * @param A 数组
	 * @return
	 */
	private ArrayList<ArrayList<Integer>> getsub1(int[] A) {
		ArrayList<ArrayList<Integer>> res = new ArrayList<>();
		int maxnum = cifang(2, A.length)-1;
		for (int i = maxnum; i > 0; i--) {
			ArrayList<Integer> res_new = new ArrayList<>();
			String bit = new StringBuilder((Integer.toBinaryString(i))).reverse().toString();
			for (int j = 0; j <bit.length(); j++) {
				if (bit.charAt(j) == '1') {
					res_new.add(A[j]);
				}
			}
			res.add(res_new);
		}	
		return res;
		
	}
	//生成次方的方法
	private int cifang(int i,int n) {
		if (n == 0) {
			return 1;
		} else {
			int de = i;
			for (int j = 1; j < n; j++) {
				de=de<<1;
			}
			return de;
		}
		
		
	}

完整代码总结

package 子集生成;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.Set;

public class Demo {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
//		Set<Set<Integer>> as = new Demo().getsubset3(new int[] {1,2,3,4},4,4);
//		System.out.println(as);
		Set<Set<Integer>> as1 = new Demo().getsubset2(new int[] {1,2,3,4});
		System.out.println(as1);
		ArrayList<ArrayList<Integer>> as2 = new Demo().getsub1(new int[] {1,2,3,4});
		
		System.out.println(as2);
		
	}
	
	//递归方法
	/**
	 * 
	 * @param A 数组
	 * @param cur 当前元素位置的下标
	 * @return 返回一个嵌套的集合
	 */
	private Set<Set<Integer>> getsubset3(int[] A, int cur) {
		Set<Set<Integer>> newres = new HashSet<>();
		if (cur == 0) {
			Set<Integer> nil = new HashSet<>();
			Set<Integer> first = new HashSet<>();
			first.add(A[0]);
			newres.add(nil);
			newres.add(first);
			return newres;
		}
		Set<Set<Integer>> oldres = getsubset3(A, n, cur-1);
		for (Set<Integer> set : oldres) {
			newres.add(set);
			Set<Integer> copy = (Set<Integer>)(((HashSet)set).clone());
			copy.add(A[cur-1]);
			newres.add(copy);
			
		}		
		return newres;
	}
	
	//迭代
	/**
	 * 
	 * @param A 数组
	 * @return
	 */
	private Set<Set<Integer>> getsubset2(int[] A) {
		Set<Set<Integer>> res = new HashSet<>();
		Set<Integer> nil = new HashSet<>();
		res.add(nil);
		for (int i = 0; i < A.length; i++) {
			
			Set<Set<Integer>> help = new HashSet<>();
			help.addAll(res);
			for (Set<Integer> set : res) {
				Set a = (HashSet)(((HashSet)set).clone());
				a.add(A[i]);
				help.add(a);
			}
			res = help;
		}
		return res;
		

		
	}
	
	//二进制法
	/**
	 * 
	 * @param A 数组
	 * @return
	 */
	private ArrayList<ArrayList<Integer>> getsub1(int[] A) {
		ArrayList<ArrayList<Integer>> res = new ArrayList<>();
		int maxnum = cifang(2, A.length)-1;
		for (int i = maxnum; i > 0; i--) {
			ArrayList<Integer> res_new = new ArrayList<>();
			String bit = new StringBuilder((Integer.toBinaryString(i))).reverse().toString();
			for (int j = 0; j <bit.length(); j++) {
				if (bit.charAt(j) == '1') {
					res_new.add(A[j]);
				}
			}
			res.add(res_new);
		}	
		return res;
		
	}
	//生成次方的方法
	private int cifang(int i,int n) {
		if (n == 0) {
			return 1;
		} else {
			int de = i;
			for (int j = 1; j < n; j++) {
				de=de<<1;
			}
			return de;
		}		
	}
}

你的一键三连是对我最大的鼓励

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

这猪能飞

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值