C# 计算排列组合数,及列出所有组合形式的算法

前段时间有同学问到,如何编程求排列组合数,以及列出所有排列组合形式的算法。乘着放假,写了一种实现的方法!怕时间长了,淹没在硬盘里,记录在此!

        /// <summary>
        /// 计算Int32类型的整数的阶乘,目前最大只能对20以内的正整数求阶乘
        /// </summary>
        /// <param name="n">Int32类型的正整数</param>
        /// <returns></returns>
        public static long Factor(this int n)
        {
            long result = -1;

            checked
            {
                try
                {
                    if (n < 0)
                    {
                        result = -1;
                    }
                    else
                    {
                        if (n == 0 || n == 1)
                        {
                            result = 1;
                        }
                        else
                        {
                            result = n * (n - 1).Factor();
                        }
                    }
                }
                catch (OverflowException)
                {
                    result = -1;
                }
            }
            return result;
        }

计算阶乘的递归算法,后续会用到,虽然用了long类型,但也只能计算到20的阶乘

        /// <summary>
        /// 计算从n个不同元素中任选m个元素的排列个数.n应该大于等于m!
        /// </summary>
        /// <param name="n">供排列选择的元素个数,正整数</param>
        /// <param name="m">排列选取的元素个数,正整数</param>
        /// <returns>排列个数</returns>
        public static long Permutation(int n,int m)
        {
            int[] N = new int[n];
            int[] SubM = new int[n-m];
            long result = 0;
            if (n < m)
            {
                result = 0;
            }
            else
            { //初始化数组N和M
                for (int i = 0; i < n; i++)
                {
                    N[i] = i + 1;
                    if (i < n - m)
                    {
                        SubM[i] = i + 1;
                    }
                }
                //消除两个数组中的重复元素
                for (int i = 0; i < n - m; i++)
                {
                    if (SubM[i] == N[i])
                    {
                        N[i] = 1;
                    }
                }
                //计算N中剩余元素的累乘
                result = 1;
                checked
                {
                    try
                    {
                        for (int i = 0; i < n; i++)
                        {
                            result *= N[i];
                        }
                    }
                    catch (OverflowException)
                    {
                        result = -1;
                    }

                }
            }
            return result;
        }

计算排列数,主要思想是类似于小学数学中求分式相乘中的消除法,主要是为了避免求阶乘的限制

        /// <summary>
        /// 计算从n个不同元素中选取m个元素的组合个数。n应该大于等于m
        /// </summary>
        /// <param name="n">供组合选择的元素个数,正整数</param>
        /// <param name="m">组合选取的元素个数,正整数</param>
        /// <returns>组合个数</returns>
        public static long Combination(int n, int m)
        {
            long factM = m.Factor();
            long result = 0;
            int[] N, M, subM;

            if (n < m)
            {
                result = 0;
            }
            else
            {
                if (factM > 0)
                {
                    result = Permutation(n, m) / factM;
                }
                else
                {
                    N = new int[n]; M = new int[m]; subM = new int[n - m];
                    //初始化三个数组
                    for (int i = 0; i < n; i++)
                    {
                        N[i] = i + 1;
                        if (i < m)
                        {
                            M[i] = i + 1;
                        }
                        if (i < n - m)
                        {
                            subM[i] = i + 1;
                        }
                    }
                    //消除重复元素,因为当m的阶乘溢出时才会进入此分支,所以只考虑和数组M进行消除
                    for(int i = 0; i < m; i++)
                    {
                        if (N[i] == M[i])
                        {
                            N[i] = 1;
                        }
                    }
                    //计算数组N和subM的累乘
                    long rN = 1, rSubM = 1;
                    for (int i = 0; i < n; i++)
                    {
                        rN *= N[i];
                        if (i < n - m)
                        {
                            rSubM *= subM[i];
                        }
                    }
                    //计算组合个数
                    result = rN / rSubM;
                }
            }

            return result;
        }

计算组合数,思路和求排列数一致,不再赘述

/// <summary>
        /// 获得从n个不同元素中任意选取m个元素的组合的所有组合形式的列表
        /// </summary>
        /// <param name="elements">供组合选择的元素</param>
        /// <param name="m">组合中选取的元素个数</param>
        /// <returns>返回一个包含列表的列表,包含的每一个列表就是每一种组合可能</returns>
        public static List<List<T>> GetCombinationList<T>(List<T> elements,int m)
        {
            List<List<T>> result = new List<List<T>>();//存放返回的列表
            List<List<T>> temp = null; //临时存放从下一级递归调用中返回的结果
            List<T> oneList = null; //存放每次选取的第一个元素构成的列表,当只需选取一个元素时,用来存放剩下的元素分别取其中一个构成的列表;
            T oneElment; //每次选取的元素
            List<T> source = new List<T>(elements); //将传递进来的元素列表拷贝出来进行处理,防止后续步骤修改原始列表,造成递归返回后原始列表被修改;
            int n = 0; //待处理的元素个数

            if (elements != null)
            {
                n = elements.Count;
            }
            if(n==m && m != 1)//n=m时只需将剩下的元素作为一个列表全部输出
            {
                result.Add(source);
                return result;
            }
            if (m == 1)  //只选取一个时,将列表中的元素依次列出
            {
                foreach(T el in source)
                {
                    oneList = new List<T>();
                    oneList.Add(el);
                    result.Add(oneList);
                    oneList = null;
                }
                return result;
            }

            for (int i = 0; i <= n - m; i++)
            {
                oneElment = source[0];
                source.RemoveAt(0);
                temp = GetCombinationList(source, m - 1);
                for (int j = 0; j < temp.Count; j++)
                {
                    oneList = new List<T>();
                    oneList.Add(oneElment);
                    oneList.AddRange(temp[j]);
                    result.Add(oneList);
                    oneList = null;
                }
            }


            return result;
        }

求所有可能的组合形式,一个递归实现,依次选择待组合元素中的前n-m个元素,每次选择后在剩下的元素中选取m-1个元素,直到m==1或者列表元素个数等于m则停止递归
排列的所有形式可以在此基础上得到,但是还没有想到满意的算法!思考中……
PS:为了更明白的表达解题思路,算法没有做优化!

  • 5
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值