背景很简单,就是从给定的m个不同的元素中选出n个,输出所有的组合情况!

例如:从1到m的自然数中,选择n(n<=m)个数,有多少种选择的组合,将其输出!

本方案的代码实现逻辑是比较成熟的方案:

  1. * 一个bit位(boolean)一维数组中,初始化全为0(false), 然后给左边的n个位初始化为1(true)。
  2. * <> 从左向右找第一个10的位置,将10换位程01,然后将这个01左边的所有的1全都移位到数组的最左边,此时得到的1所在位置下标对应序列即为一个组合数。
  3. * <> 循环重复上面的<>步骤的操作,直到所有的1都移动到最右边为止。

先不多说其他的,直接将代码贴在这里,以供有需要的伙伴借鉴:

  1. /**
  2. * @author "shihuc"
  3. * @date 2016年12月1日
  4. */
  5.  
  6. import java.util.ArrayList;
  7. import java.util.Arrays;
  8.  
  9. /**
  10. * @author chengsh05
  11. *
  12. * 组合算法实现,支持产品列表页的筛选模块实现全静态化。
  13. *
  14. * 给定m个不同的数,从中选择出n个数的不同选择方法有多少种?
  15. * 答案:一共有 n!/(n-m)!*m!
  16. *
  17. * 存储计算出来的组合结构,组合元素之间用逗号分隔
  18. * 例如1,2,3的全组合:
  19. * "1,", "2,", "3,","1,2,", "1,3,", "2,3,", "1,2,3,"
  20. */
  21. public class Combination {
  22.  
  23. /**
  24. * 从m个元素中取出n个元素,获取排序好了的组合数列表,同一个组合中的元素按从小到大排序。
  25. *
  26. * @param m 组合的元素基数
  27. * @param n 组合的被选元素个数
  28. * @return 排序好后的组合列表
  29. * @throws Exception
  30. */
  31. public ArrayList<String> getCombinations(int m, int n) throws Exception{
  32.  
  33. if(m < n){
  34. throw new IllegalCombinationInputException();
  35. }
  36.  
  37. ArrayList<String> resCom = calcCombination(m, n);
  38.  
  39. return resCom;
  40. }
  41.  
  42. /**
  43. * 通过移位方式,计算给定m个数中取出n个数的组合列表。
  44. *
  45. * 具体思路:
  46. * 一个bit位(boolean)数组中,初始化全为0(false), 然后给左边的n个位初始化为1(true)。
  47. * <1> 从左向右找第一个10的位置,将10换位程01,然后将这个01左边的所有的1全都 移位到数组的最左边,此时得到的1所在位置下标对应的序列即为一个组合数。
  48. * <2> 循环重复上面的<1>步骤的操作,直到所有的1都移动到最右边为止。
  49. *
  50. * @param m 输入的基数个数
  51. * @param n 组合被选元素格式
  52. * @return 原始组合数列表,即没有排序的组合
  53. */
  54. private ArrayList<String> calcCombination(int m, int n){
  55. boolean base[] = new boolean[m];
  56. Arrays.fill(base, false);
  57. for(int i=; i<n; i++){
  58. base[i] = true;
  59. }
  60. return combination(base,m,n);
  61. }
  62.  
  63. private ArrayList<String> combination(boolean a[], int m, int n){
  64. ArrayList<String> combination = new ArrayList<String>();
  65. while(!isAllZeroLeft(a, m, n)){
  66. for(int i=; i<m-; i++){
  67. if(a[i] == true && a[i+] == false){
  68. String elem = calcElement(a); //计算出一个组合元素
  69. //System.out.println(elem);
  70. combination.add(elem);
  71. oneZeroSwap(a, i, i+); //完成10/01换位
  72. moveFrontOneToLeft(a, i); //完成剩余左边的1全向最左边搬移操作
  73. break;
  74. }
  75. }
  76. }
  77.  
  78. //最后一个元素也是组合的一个,即所有的1(true)都到了数组的最右边
  79. combination.add(calcElement(a));
  80. return combination;
  81. }
  82.  
  83. /**
  84. * 异或操作实现不开辟新的存储空间完成两个数的位置交换。
  85. *
  86. * @param a 待操作数所在的数组
  87. * @param x 待交换位置的第一个数在数组中的下标
  88. * @param y 待交换位置的第二个数在数组中的下标
  89. */
  90. private void oneZeroSwap(boolean a[], int x, int y){
  91. a[x] = a[x] ^ a[y];
  92. a[y] = a[y] ^ a[x];
  93. a[x] = a[x] ^ a[y];
  94. }
  95.  
  96. /**
  97. * 判断10作01交换位置后,是否实现了数组a中右端的n个元素全为1(true),这个结果作为10/01换位结束标识
  98. *
  99. * @param a 10/01换位的输入数组
  100. * @param m 计算组合的元数据个数
  101. * @param n 计算组合的被选取元素个数
  102. * @return true表示10/01换位结束,false表示还可以继续
  103. */
  104. private boolean isAllZeroLeft(boolean a[], int m, int n){
  105. int gap = m - n;
  106. for(int i=; i<gap; i++){
  107. if(a[i]){
  108. return false;
  109. }
  110. }
  111. return true;
  112. }
  113.  
  114. /**
  115. * 将10/01换位之后数组左边的全部1都搬移到数组的最左边。
  116. *
  117. * @param a 待操作的组合数组
  118. * @param end 指明搬移操作的范围,在end数组下标左边的进行搬移, 这个end的值小于数组的长度
  119. */
  120. private void moveFrontOneToLeft(boolean a[], int end){
  121. int oneCnt = ;
  122. for(int i=; i<end; i++){
  123. if(a[i]){
  124. oneCnt++;
  125. a[i] = false;
  126. }
  127. }
  128. for(int i=; i<oneCnt; i++){
  129. a[i] = true;
  130. }
  131. }
  132.  
  133. /**
  134. * 计算当前数组中的组合元素的值,数组元素为1(true)的对应的下标的全集,即为所需的一个组合元素值
  135. *
  136. * @param a 待操作的组合数组
  137. * @return 一个组合的值, 去掉了最后的一个逗号分隔符
  138. */
  139. private String calcElement(boolean a[]){
  140. String elem = "";
  141. for(int i=; i<a.length; i++){
  142. if(a[i]){
  143. elem += (i+) + ",";
  144. }
  145. }
  146. return elem.substring(, elem.length() - );
  147. }
  148.  
  149. /**
  150. * @param args
  151. * @throws Exception
  152. */
  153. public static void main(String[] args) throws Exception {
  154. int m = , n = ;
  155. Combination combination = new Combination();
  156. ArrayList<String> coms = combination.getCombinations(m, n);
  157. for(int i = ; i<coms.size(); i++){
  158. System.out.println(coms.get(i));
  159. }
  160. }
  161. }

代码中定义的Exception的类:

  1. /**
  2. * @author "shihuc"
  3. * @date 2016年12月1日
  4. */
  5.  
  6. /**
  7. * @author chengsh05
  8. *
  9. */
  10. public class IllegalCombinationInputException extends Exception{
  11.  
  12. private static final long serialVersionUID = 678024281707796100L;
  13.  
  14. public IllegalCombinationInputException(){
  15. super("The combination base number should be not less than the selection number.");
  16. }
  17.  
  18. }

此算法思路,在很多场景下还是值得借鉴的!

算法是计算机科学的灵魂,坚持算法学习和应用......

计算一维组合数的java实现的更多相关文章

  1. 构建一个学生Student,根据类Student的定义,创建五个该类的对象,输出每个学生的信息,计算并输出这五个学生Java语言成绩的平均值,以及计算并输出他们Java语言成绩的最大值和最小值。

    定义一个表示学生信息的类Student,要求如下: (1)类Student的成员变量: sNO 表示学号: sName表示姓名: sSex表示性别: sAge表示年龄: sJava:表示Java课程成 ...

  2. 组合数的计算以及组合数对p取余后结果的计算

    前奏:统计 n! 中的所有质因子中pi的个数 普通方法:复杂度O(nlogn), 当n为10的18次方无法承受 // 复杂度O(nlogn), n为10的18次方无法承受 int cal(int n, ...

  3. SDUT1574组合数的计算(组合数)

    http://acm.sdut.edu.cn/sdutoj/problem.php?action=showproblem&problemid=1574 这个题,比较奇怪,是用递推去做的,我试了 ...

  4. 基于OpenStreetMap计算驾车距离(Java)

    最近公司有个项目需要计算6000个点之间的驾车距离,第一时间想到的是利用Google的Distance Matrix API,但是免费Key每天只能计算2500个元素(元素 = 起点数量 * 终点数量 ...

  5. 算法笔记_044:表达式计算求值(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 输入一个只包含加减乖除和括号的合法表达式,求表达式的值.其中除表示整除. 输入格式 输入一行,包含一个表达式. 输出格式 输出这个表达式的 ...

  6. 复利计算C语言转java的相关代码

    static void principal()// 计算本金 { int N, m; double i, F, P; System.out.printf("复利终值:"); F = ...

  7. 根据两点经纬度计算距离和角度——java实现

    原理:见上一篇博客   http://blog.csdn.net/xiaobai091220106/article/details/50879365 百度地图拾取经纬度坐标:http://api.ma ...

  8. [Java]简单计算下一段Java代码段运行了多少秒

    long startTime = System.currentTimeMillis(); ...... long endTime = System.currentTimeMillis(); logge ...

  9. 组合数计算-java

    排列组合是计算应用经常使用的算法,通常使用递归的方式计算,但是由于n!的过于大,暴力计算很不明智.一般使用以下两种方式计算. 一,递归的思想:假设m中取n个数计算排列组合数,表示为comb(m,n). ...

随机推荐

  1. ECLIPSE/JAVAWEB (二)三大框架之Hibernate框架 持续更新中...

    (一)发展历史 在Struts框架中使用jdbc连接来读写数据库,我们最常见的就是打开数据库连接.使用复杂的sql语句进行读写.关闭连接,获得的数据又需要转换或封装后往外传,这是一个非常繁琐的过程. ...

  2. JavaScript面试题

    一道常被人轻视的前端JS面试题 标签(空格分隔): JavaScript function Foo() { getName = function () { alert (1); }; return t ...

  3. Mac小知识(不定时更新)

    1.显示隐藏文件夹(在mac命令行中输入以下代码即可): 1)显示隐藏文件夹 defaults write com.apple.finder AppleShowAllFiles Yes &&a ...

  4. vimperator setting records

    vimperator confugration files :highlight Hint color:#000;background:rgb(250,230,150);border-radius:4 ...

  5. codeforces 723D(DFS)

    题目链接:http://codeforces.com/problemset/problem/723/D 题意:n*m的矩阵中,'*'代表陆地,'.'代表水,连在一起且不沿海的水形成湖泊.问最少填多少块 ...

  6. session过期,登录页被内嵌iframe的解决方案

    在登录页的js加上: if(window !=top){ top.location.href = location.href; } 就可以完美解决这个问题!

  7. Maven学习笔记(1)之安装Maven

    此笔记是学习Maven时自己摸索+各种百度而来,并非全部原创,望与各位一同学习,勿拍~勿拍~ 安装步骤 1.下载Maven的最新版本,地址:http://maven.apache.org/downlo ...

  8. 简单的URL解析

    简单的URL解析,直接举例说明了 function getUrl(){ //如果存在则取到来后面的参数,注意,?不需要取出,substring从1开始,否则取""; var qs ...

  9. 第五章 搭建 S3C6.410 开发板的 测试环境

    一.简介: 对于嵌入式驱动开发者来说,你必须要了解什么是开发板:它与我们经常用的手机类似, 包含了显示屏. 键盘. Wi-Fi. 蓝牙等模块等,是开发者必备的硬件设备.但与手机不同的是:在开发板上安装 ...

  10. NSCharacter​Set在字符串操作中得使用

    NSCharacterSet以及它的可变版本NSMutableCharacterSet,用面向对象的方式表示一组Unicode字符,它经常与NSString及NSScanner组合起来使用,在不同的字 ...