今年上半年,我在KwCombinatorics系列文章中,重点介绍了KwCombinatorics组件的使用情况,其实这个组件我5年前就开始用了,非常方便,麻雀虽小五脏俱全。所以一直非常喜欢,才写了几篇文章推荐给大家。最近在计算足球彩票结果组合过程中,使用的到了其功能,生成排列,非常具有代表性,而且也有网友咨询过类似的问题,所以就封装为扩展方法,方便调用。

NET开源目录:【目录】本博客其他.NET开源项目文章目录

彩票数据资料目录【目录】C#搭建足球赛事资料库与预测平台与彩票数据分析目录

本文原文地址:.NET平台开源项目速览(11)KwCombinatorics排列组合使用案例(1)

1.KwCombinatorics基本介绍

KwCombinatorics,它是.NET平台一个高效的生成排列组合序列的开源类库,它提供了4种生成排列与组合序列的方式。虽然原理和功能都很简单,但是这个类库在软件测试、组合数学以及密码学等方面都有很大的用处。很早就接触了这个类库,以前在一些小程序中也使用过,有时候为了遍历所有可能的组合,自己去写循环,生成,的确很繁琐,有了KwCombinatorics 之后,都变得简单写了,接下来将详细介绍该类库的使用。

 KwCombinatorics类库的主页是:http://kwcombinatorics.codeplex.com/

关于KwCombinatorics基本使用入门的文章有:

1.【原创】开源.NET排列组合组件KwCombinatorics使用(一)组合生成 

2.【原创】开源.NET排列组合组件KwCombinatorics使用(二)排列生成

3.【原创】开源.NET排列组合组件KwCombinatorics使用(三)笛卡尔积组合

2.足球彩票排列组合应用

在足球彩票玩法中,经常要选择几场比赛,以及每场比赛不同的结果组合,进行投注,这个时候就要生成所有的排列组合了,以便进行下一步的平衡投注计算。例如,看下面一个截图:

例如,上午选择了3场比赛,前2场,每场选择2个结果,第三场做胆,进行3串1玩法,当然这里的例子很简单,大家脑袋想一想就知道是4个结果。如果是其他玩法,每个有3,4个选择,有5串1 ,那就麻烦了,何苦都必须要自动生成出组合来。所以必须得考虑一个通用的情况。而且在进行预测以及分析的时候,通常要对包含所有结果的几场比赛进行串分析,就需要计算所有排列组合的结果形式了例如下面这种总进球的多串1选择:

3.排列组合生成代码与测试

3.1 N场单个玩法所有结果的组合

比如胜平负有3个玩法,【3,1,0】,那么我有N场比赛,要进行串组合,计算所有的组合情况。写一个通用的函数,使用KwCombinatorics的笛卡尔积来完成,直接看代码和注释,注意使用之前要引用对应的dll和命名空间:

/// <summary>单个不同玩法,任意场次数的结果全部结果组合</summary>
/// <typeparam name="T">结果类型,主要分字符串,整数</typeparam>
/// <param name="source">该玩法的所有结果列表</param>
/// <param name="count">场次数量:每一个场次都可能会出现source中的结果</param>
/// <returns>所有组合的列表</returns>
public static IEnumerable<List<T>> PermuteResultForSingle<T>(this List<T> source, int count)
{
//实例化N场比赛的所有结果数组,每一个List都包括所有结果
List<T>[] sourceList = new List<T>[count];
Int32[] sizes = new Int32[count]; for (int i = 0; i < count; i++)
{
sourceList[i] = source;
sizes[i] = source.Count;
}
//使用笛卡尔积的形式来生成
var pt = new Product(sizes);
//依次生产所有组合,每一个组合都是结果的集合
foreach (var row in pt.GetRows())
{
yield return Product.Permute(row, sourceList);
}
}

考虑到调用方便,这里写成了扩展方法,并使用了泛型,也就是说可以直接处理整数结果形式,也可以处理汉字字符串结果形式。那我们进行一下测试看看。

先看测试代码,我们假设胜平负玩法,2场比赛,应该是有9种结果:

static void Test1()
{
//先定义每一个玩法的所有结果类型,胜平负有3个结果
var result = new List<Int32>() { 3,1,0};
//直接扩展方法计算,并传入场次数量参数,我们计算2场的所有组合
var combList = result.PermuteResultForSingle<Int32>(2);
foreach (var item in combList)
{
//将结果转换为字符串依次打印
//ArrayToString是一个将数组列表转换为字符串的扩展方法
Console.WriteLine(item.ArrayToString());
}
}

看看结果:

其实还可以处理字符串类型的情况,例如我们把代码改成这样:

var result = new List<String>() { "胜","平","负"};
//直接扩展方法计算,并传入场次数量参数,我们计算2场的所有组合
var combList = result.PermuteResultForSingle<String>(2);
foreach (var item in combList)
{
//将结果转换为字符串依次打印
Console.WriteLine(item.ArrayToString());
}

结果如下,当然你可以修改数量,计算更多复杂的情况,一 一进行处理。

3.2 混合玩法的多个结果组合

在竞彩的混投玩法中,可以选择任意场次的不同玩法进行组合,比如,你A比赛可以选择胜平负结果,而B比赛选择大小球,然后串起来进行投注。情况有时候会更复杂,这种情况下,玩家为了刺激,会适当增加结果数量,然后进行串投注。其实整体思路和3.1差不多,只不过由于每个玩法的结果类型不一样,需要将参数转换为object,才更好处理。看看代码,也是扩展方法:

/// <summary>混合不同玩法,任意场次的多个结果的组合</summary>
/// <param name="source">场次按顺序选择的玩法的所有结果列表</param>
/// <returns></returns>
public static IEnumerable<List<Object>> PermuteResultForMixture(this List<Object>[] source)
{
//初始化所有场次玩法结果的数据
Int32[] sizes = new Int32[source.Length];
for (int i = 0; i < source.Length; i++) sizes[i] = source[i].Count;
//使用笛卡尔积,只不过这里都是Object,不在针对同一种玩法了
var pt = new Product(sizes);
foreach (var row in pt.GetRows())
{
yield return Product.Permute(row, source);
}
}

看看实际的测试结果,当然你也可以把所有玩法都搞一样的,相对是同一种玩法的N场比赛,选择不同部分结果进行的组合了:

//胜平负场次的结果选择
var R1 = new List<Object>() { "胜","平"};
//总进球数结果的选择
var R2 = new List<Object>() { 2,3,4 };
//半全场结果的选择
var R3 = new List<Object>() { "胜负" ,"平负"};
//综合所有选择
var result = new List<Object>[] { R1, R2, R3 };
//计算所有组合
var combList = result.PermuteResultForMixture();
foreach (var item in combList)
{
//打印组合,逗号分割,应该有2*3*2=12种组合
Console.WriteLine(item.ArrayToString());
}

看结果:

其他使用很简单了,依次类推吧。KwCombinatorics的确是很强大,只要和排列组合相关的需求,没有搞不定的。代码,效率都值得点赞。本文代码很简单,就不列举了,详细的可以看上面的系列文章。

.NET平台开源项目速览(11)KwCombinatorics排列组合使用案例(1)的更多相关文章

  1. .NET平台开源项目速览(12)哈希算法集合类库HashLib

    .NET的System.Security.Cryptography命名空间本身是提供加密服务,散列函数,对称与非对称加密算法等功能.实际上,大部分情况下已经满足了需求,而且.NET实现的都是目前国际上 ...

  2. .NET平台开源项目速览(3)小巧轻量级NoSQL文件数据库LiteDB

    今天给大家介绍一个不错的小巧轻量级的NoSQL文件数据库LiteDB.本博客在2013年也介绍过2款.NET平台的开源数据库: 1.[原创]开源.NET下的XML数据库介绍及入门 2.[原创]C#开源 ...

  3. NET平台开源项目速览(6)FluentValidation验证组件介绍与入门(转载)

    原文地址:http://www.cnblogs.com/asxinyu/p/dotnet_Opensource_project_FluentValidation_1.html 阅读目录 1.基本介绍 ...

  4. .NET平台开源项目速览(17)FluentConsole让你的控制台酷起来

    从该系列的第一篇文章 .NET平台开源项目速览(1)SharpConfig配置文件读写组件 开始,不知不觉已经到第17篇了.每一次我们都是介绍一个小巧甚至微不足道的.NET平台的开源软件,或者学习,或 ...

  5. .NET平台开源项目速览(15)文档数据库RavenDB-介绍与初体验

    不知不觉,“.NET平台开源项目速览“系列文章已经15篇了,每一篇都非常受欢迎,可能技术水平不高,但足够入门了.虽然工作很忙,但还是会抽空把自己知道的,已经平时遇到的好的开源项目分享出来.今天就给大家 ...

  6. .NET平台开源项目速览(13)机器学习组件Accord.NET框架功能介绍

    Accord.NET Framework是在AForge.NET项目的基础上封装和进一步开发而来.因为AForge.NET更注重与一些底层和广度,而Accord.NET Framework更注重与机器 ...

  7. .NET平台开源项目速览(1)SharpConfig配置文件读写组件

    在.NET平台日常开发中,读取配置文件是一个很常见的需求.以前都是使用System.Configuration.ConfigurationSettings来操作,这个说实话,搞起来比较费劲.不知道大家 ...

  8. .NET平台开源项目速览(10)FluentValidation验证组件深入使用(二)

    在上一篇文章:.NET平台开源项目速览(6)FluentValidation验证组件介绍与入门(一) 中,给大家初步介绍了一下FluentValidation验证组件的使用情况.文章从构建间的验证器开 ...

  9. .NET平台开源项目速览(9)软件序列号生成组件SoftwareProtector介绍与使用

    在文章:这些.NET开源项目你知道吗?让.NET开源来得更加猛烈些吧!(第二辑)中,给大家初步介绍了一下Software Protector序列号生成组件.今天就通过一篇简单的文章来预览一下其强大的功 ...

随机推荐

  1. js前端模块化之加载器原理解析(一)

    先来说一下前端模块化的价值:引用模块此处有详细的介绍,可以自行前往观看. 一.总结如下优点: (1)解决命名冲突(2)烦琐的文件依赖(3)模块的版本管理(4)提高可维护性(5)前端性能优化(6)跨环境 ...

  2. Activemq mqtt 点对点聊天实现(转载)

    我这想到一个点对点聊天的方法,不用没割人都建立一个topic了,思路还是自定义一个分发策略,具体如下: 1.  建立一个topic,所有人都用匹配订阅的方式订阅以该topic为头的topic,例如:所 ...

  3. 深入浅出HTTP协议(WEB开发和面试必备)

    1. 基础概念篇   a.简介 HTTP是Hyper Text Transfer Protocol(超文本传输协议)的缩写.它的发展是万维网协会(World Wide Web Consortium)和 ...

  4. 误删/usr文件夹解决办法

    http://blog.chinaunix.net/uid-2623904-id-3044156.html http://www.centoscn.com/CentOS/Intermediate/20 ...

  5. SQLServer触发器创建、删除、修改、查看

    一: 触发器是一种特殊的存储过程﹐它不能被显式地调用﹐而是在往表中插入记录﹑更新记录或者删除记录时被自动地激活.所以触发器可以用来实现对表实施复杂的完整性约束. 二: SQL Server为每个触发器 ...

  6. Total Commander解压位置

    TC解压到当前文件夹下 TC也是用了一段时间,现在勉强也算用习惯了,今天在解压文件的时候感觉步骤麻烦,之前解压都是解压到另一个窗口,所以一直是ALT+8同步窗口,然后解压文件.但一般解压文件都是解压到 ...

  7. PHP:Xdebug配置

    在配置Xdebug时,之前经历了无数次失败,终于配置成功! 以下是配置失败的原因: 1.下载时,选用Xdebug的版本不正确: 2.配置时,Xdebug文件名或文件的路径不正确: 在参考 http:/ ...

  8. (转)为什么需要正则表达式 by 王珢

    为什么需要正则表达式 by 王垠 学习Unix最开头,大家都学过正则表达式(regexp).可是有没有人考虑过我们为什么需要正则表达式? 正则表达式本来的初衷是用来从无结构的字符串中提取信息,殊不知这 ...

  9. Akka-remote使用入门

    在上一篇文章中讲了akka-actor的简单使用,那主要是展现了akka在一台机器上的并发应用,这一篇接着介绍akka-remote使用,简单了解akka在不同机器上的并发应用.我们知道,在一台机器上 ...

  10. 为什么我如此热爱这样一个比赛(转自vici)

    为什么我如此的热爱这样一个比赛呢?因为它总能带给我一个目标,让我去努力实现它.因为可以看到胜利的希望,于是不断的去追逐.虽然其中的过程可能是比较艰辛的.   对于天才选手,作为天生的冠军,大概凭借天赋 ...