计算一维组合数的java实现
背景很简单,就是从给定的m个不同的元素中选出n个,输出所有的组合情况!
例如:从1到m的自然数中,选择n(n<=m)个数,有多少种选择的组合,将其输出!
本方案的代码实现逻辑是比较成熟的方案:
* 一个bit位(boolean)一维数组中,初始化全为0(false), 然后给左边的n个位初始化为1(true)。
* <> 从左向右找第一个10的位置,将10换位程01,然后将这个01左边的所有的1全都移位到数组的最左边,此时得到的1所在位置下标对应序列即为一个组合数。
* <> 循环重复上面的<>步骤的操作,直到所有的1都移动到最右边为止。
先不多说其他的,直接将代码贴在这里,以供有需要的伙伴借鉴:
/**
* @author "shihuc"
* @date 2016年12月1日
*/ import java.util.ArrayList;
import java.util.Arrays; /**
* @author chengsh05
*
* 组合算法实现,支持产品列表页的筛选模块实现全静态化。
*
* 给定m个不同的数,从中选择出n个数的不同选择方法有多少种?
* 答案:一共有 n!/(n-m)!*m!
*
* 存储计算出来的组合结构,组合元素之间用逗号分隔
* 例如1,2,3的全组合:
* "1,", "2,", "3,","1,2,", "1,3,", "2,3,", "1,2,3,"
*/
public class Combination { /**
* 从m个元素中取出n个元素,获取排序好了的组合数列表,同一个组合中的元素按从小到大排序。
*
* @param m 组合的元素基数
* @param n 组合的被选元素个数
* @return 排序好后的组合列表
* @throws Exception
*/
public ArrayList<String> getCombinations(int m, int n) throws Exception{ if(m < n){
throw new IllegalCombinationInputException();
} ArrayList<String> resCom = calcCombination(m, n); return resCom;
} /**
* 通过移位方式,计算给定m个数中取出n个数的组合列表。
*
* 具体思路:
* 一个bit位(boolean)数组中,初始化全为0(false), 然后给左边的n个位初始化为1(true)。
* <1> 从左向右找第一个10的位置,将10换位程01,然后将这个01左边的所有的1全都 移位到数组的最左边,此时得到的1所在位置下标对应的序列即为一个组合数。
* <2> 循环重复上面的<1>步骤的操作,直到所有的1都移动到最右边为止。
*
* @param m 输入的基数个数
* @param n 组合被选元素格式
* @return 原始组合数列表,即没有排序的组合
*/
private ArrayList<String> calcCombination(int m, int n){
boolean base[] = new boolean[m];
Arrays.fill(base, false);
for(int i=; i<n; i++){
base[i] = true;
}
return combination(base,m,n);
} private ArrayList<String> combination(boolean a[], int m, int n){
ArrayList<String> combination = new ArrayList<String>();
while(!isAllZeroLeft(a, m, n)){
for(int i=; i<m-; i++){
if(a[i] == true && a[i+] == false){
String elem = calcElement(a); //计算出一个组合元素
//System.out.println(elem);
combination.add(elem);
oneZeroSwap(a, i, i+); //完成10/01换位
moveFrontOneToLeft(a, i); //完成剩余左边的1全向最左边搬移操作
break;
}
}
} //最后一个元素也是组合的一个,即所有的1(true)都到了数组的最右边
combination.add(calcElement(a));
return combination;
} /**
* 异或操作实现不开辟新的存储空间完成两个数的位置交换。
*
* @param a 待操作数所在的数组
* @param x 待交换位置的第一个数在数组中的下标
* @param y 待交换位置的第二个数在数组中的下标
*/
private void oneZeroSwap(boolean a[], int x, int y){
a[x] = a[x] ^ a[y];
a[y] = a[y] ^ a[x];
a[x] = a[x] ^ a[y];
} /**
* 判断10作01交换位置后,是否实现了数组a中右端的n个元素全为1(true),这个结果作为10/01换位结束标识
*
* @param a 10/01换位的输入数组
* @param m 计算组合的元数据个数
* @param n 计算组合的被选取元素个数
* @return true表示10/01换位结束,false表示还可以继续
*/
private boolean isAllZeroLeft(boolean a[], int m, int n){
int gap = m - n;
for(int i=; i<gap; i++){
if(a[i]){
return false;
}
}
return true;
} /**
* 将10/01换位之后数组左边的全部1都搬移到数组的最左边。
*
* @param a 待操作的组合数组
* @param end 指明搬移操作的范围,在end数组下标左边的进行搬移, 这个end的值小于数组的长度
*/
private void moveFrontOneToLeft(boolean a[], int end){
int oneCnt = ;
for(int i=; i<end; i++){
if(a[i]){
oneCnt++;
a[i] = false;
}
}
for(int i=; i<oneCnt; i++){
a[i] = true;
}
} /**
* 计算当前数组中的组合元素的值,数组元素为1(true)的对应的下标的全集,即为所需的一个组合元素值
*
* @param a 待操作的组合数组
* @return 一个组合的值, 去掉了最后的一个逗号分隔符
*/
private String calcElement(boolean a[]){
String elem = "";
for(int i=; i<a.length; i++){
if(a[i]){
elem += (i+) + ",";
}
}
return elem.substring(, elem.length() - );
} /**
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
int m = , n = ;
Combination combination = new Combination();
ArrayList<String> coms = combination.getCombinations(m, n);
for(int i = ; i<coms.size(); i++){
System.out.println(coms.get(i));
}
}
}
代码中定义的Exception的类:
/**
* @author "shihuc"
* @date 2016年12月1日
*/ /**
* @author chengsh05
*
*/
public class IllegalCombinationInputException extends Exception{ private static final long serialVersionUID = 678024281707796100L; public IllegalCombinationInputException(){
super("The combination base number should be not less than the selection number.");
} }
此算法思路,在很多场景下还是值得借鉴的!
算法是计算机科学的灵魂,坚持算法学习和应用......
计算一维组合数的java实现的更多相关文章
- 构建一个学生Student,根据类Student的定义,创建五个该类的对象,输出每个学生的信息,计算并输出这五个学生Java语言成绩的平均值,以及计算并输出他们Java语言成绩的最大值和最小值。
定义一个表示学生信息的类Student,要求如下: (1)类Student的成员变量: sNO 表示学号: sName表示姓名: sSex表示性别: sAge表示年龄: sJava:表示Java课程成 ...
- 组合数的计算以及组合数对p取余后结果的计算
前奏:统计 n! 中的所有质因子中pi的个数 普通方法:复杂度O(nlogn), 当n为10的18次方无法承受 // 复杂度O(nlogn), n为10的18次方无法承受 int cal(int n, ...
- SDUT1574组合数的计算(组合数)
http://acm.sdut.edu.cn/sdutoj/problem.php?action=showproblem&problemid=1574 这个题,比较奇怪,是用递推去做的,我试了 ...
- 基于OpenStreetMap计算驾车距离(Java)
最近公司有个项目需要计算6000个点之间的驾车距离,第一时间想到的是利用Google的Distance Matrix API,但是免费Key每天只能计算2500个元素(元素 = 起点数量 * 终点数量 ...
- 算法笔记_044:表达式计算求值(Java)
目录 1 问题描述 2 解决方案 1 问题描述 问题描述 输入一个只包含加减乖除和括号的合法表达式,求表达式的值.其中除表示整除. 输入格式 输入一行,包含一个表达式. 输出格式 输出这个表达式的 ...
- 复利计算C语言转java的相关代码
static void principal()// 计算本金 { int N, m; double i, F, P; System.out.printf("复利终值:"); F = ...
- 根据两点经纬度计算距离和角度——java实现
原理:见上一篇博客 http://blog.csdn.net/xiaobai091220106/article/details/50879365 百度地图拾取经纬度坐标:http://api.ma ...
- [Java]简单计算下一段Java代码段运行了多少秒
long startTime = System.currentTimeMillis(); ...... long endTime = System.currentTimeMillis(); logge ...
- 组合数计算-java
排列组合是计算应用经常使用的算法,通常使用递归的方式计算,但是由于n!的过于大,暴力计算很不明智.一般使用以下两种方式计算. 一,递归的思想:假设m中取n个数计算排列组合数,表示为comb(m,n). ...
随机推荐
- jquery ajax详解
详细参数列表url:发送请求的连接地址type:请求方式 get:获取 post:发送 put和deletetimeout:设置请求超时时间async:默认true为异步请求,false同步请求锁住浏 ...
- Log4j快速使用精简版
Log4j快速使用精简版 1.导入log4j-1.2.17.jar包 2.在src根目录下创建log4j.properties文件 log4j.rootLogger=INFO, CONSOLE, FI ...
- 一维码生成 c# winform GUI
最近看到同事小红在做一维码,感觉挺好玩,于是就在网上找了一个例子来玩玩. 下面的代码均为网上的代码,做了一些整理,但是忘记了出处,原作者看到可以提醒我,谢谢. 首先,一维码的相关知识可以先百度一下:h ...
- UE4 代码编写细节:静态变量
Note:因为在切换关切时,会GC掉所有GameThread线程下的Object类,如果Static是UOBject 请调用AddToRoot函数 当然如果你的UObject子类Object是在自己 ...
- 关于jackson处理数据
/** * 将请求参数封装成Map对象 * * @param json 参数 * @return Object */ public static Map ...
- UE4 4.14 专用服务器没有生成解决办法
简单说一下UE4 专用服务器的生成(网上也有其它版本的但是在4.14.1 上不管用) 1.用源代码编译的引擎(如何获取百度上有很多介绍)创建一个C++ UE4 工程. 2. 在vs 中找到红色这个文件 ...
- 关于MVC中Start.cs文件的作用
当我们建立默认的 .NET Framework4.5.2框架下的Web MVC 应用程序后,调试过程中我发现在程序启动页面加载完成会执行一段代码,这段代码让人有点摸不着头脑,因为之前完全没见过,调试 ...
- 获取Ip地址
public static string GetClientIPv4Address() { string ipv4 = String.Empty; foreach (IPAddress ip in D ...
- svn:cleanup failed previous operation has not finished; run cleanup if it was interrupted
svn:cleanup failed previous operation has not finished; run cleanup if it was interrupted 今天 大脑一时短路 ...
- Android学习笔记
1.问题:Error when loading the SDK:发现了以元素 'd:skin' 开头的无效内容 方法:删除了android-wear 用sdk\tools\lib下的de ...