(21)多线程实例应用:双色球(6红+1蓝)

2023-09-15 15:29:11

    


一、需求


    1.双色球: 投注号码由6个红色球号码和1个蓝色球号码组成。
    
    2.红色球号码从01--33中选择,红色球不能重复。
    
    3.蓝色球号码从01--16中选择。
        
    4.最终结果7个号码:6+1;即33选6(红)+ 16选1(蓝)
        
    5.产品: 
        能用;用户放心使用;
        原则:靠运气,不能有暗箱操作,号码开奖的随机性。
        
    6.做法思路:
        (1)从左往右---有序变化
        (2)从右往左---有序变化
        (3)同一时刻,球号码都变化~~
        (4)可以做到让所有的球都变化,且都是相互独立的变化,随机性(推荐)


二、程序


    1、界面:6个红球label显示LblRedNum1-LblRedNum6,一个蓝球LblBlue。
        两个Button,BtnStart与BtnStop。一个listBox1.


        
    2、代码:

        private string[] redNums = Enumerable.Range(1, 33).Select(i => i.ToString("00")).ToArray();
        private string[] blueNums = Enumerable.Range(1, 16).Select(i => i.ToString("00")).ToArray();
        private object lockObj = new object();
        private List<Task> tasks = new List<Task>();
        private CancellationTokenSource cts;

        private async void BtnStart_Click(object sender, EventArgs e)
        {
            BtnStart.Enabled = false;
            BtnStop.Enabled = true;
            cts = new CancellationTokenSource();
            CancellationToken ct = cts.Token;

            //初始化,置00
            foreach (Label lbl in Controls.OfType<Label>().Where(b => b.Name.Contains("Red")))
            { lbl.Text = "00"; }

            //开7个线程
            foreach (Label lbl in Controls.OfType<Label>())
            {
                if (lbl.Name.Contains("Blue"))
                {
                    tasks.Add(Task.Run(async () =>//蓝球显示
                    {
                        while (!ct.IsCancellationRequested)
                        {
                            await Task.Delay(200);
                            string strBlue = blueNums[GetRandom(0, 16)];
                            lbl.Invoke(new Action(() =>
                            {
                                lbl.Text = strBlue;
                            }));
                        }
                    }));
                }
                else
                {
                    tasks.Add(Task.Run(async () =>//红球显示
                     {
                         while (!ct.IsCancellationRequested)
                         {
                             await Task.Delay(200);
                             int idx = GetRandom(0, 33);
                             string strRed = redNums[idx];
                             lock (lockObj)
                             {
                                 List<string> list = GetCurNumList();
                                 if (!list.Contains(strRed))
                                 {
                                     lbl.Invoke(new Action(() => { lbl.Text = strRed; }));
                                     redNums[idx] = strRed;
                                 }
                             }
                         }
                     }));
                }
            }

            await Task.Run(() =>//任务取消时显示
                 {
                     Task.WaitAll(tasks.ToArray());
                     Invoke(new Action(() =>
                     {
                         List<string> list = new List<string>();
                         foreach (Label lbl in this.Controls.OfType<Label>().Where(b => b.Name.Contains("Red")))
                         {
                             list.Add(lbl.Text);
                         }
                         list.Sort();
                         list.Add(LblBlue.Text);
                         listBox1.Items.Add(string.Join(",", list.ToArray()));
                     })
                     );
                 });
        }

        private List<string> GetCurNumList()//返回当前红球列表
        {
            List<string> list = new List<string>();
            foreach (Label lbl in this.Controls.OfType<Label>().Where(b => b.Name.Contains("Red")))
            { Invoke(new Action(() => { list.Add(lbl.Text); })); }

            return list;
        }

        private int GetRandom(int min, int max)//产生强随机数
        {
            byte[] bytes = new byte[4];
            using (var rng = RandomNumberGenerator.Create())
            { rng.GetBytes(bytes); }

            int seed = BitConverter.ToInt32(bytes, 0);
            return new Random(seed).Next(min, max);
        }

        private void BtnStop_Click(object sender, EventArgs e)
        {
            cts.Cancel();
            BtnStart.Enabled = true;
            BtnStop.Enabled = false;
        }

        private void Form1_Load(object sender, EventArgs e)
        { BtnStop.Enabled = false; }


三、细节


    1、创建两个数组,根据随机产生的索引,从而得到随机的数。
        比如,随机产生索引3,那么对于红球redNum[3]就是随机的产生的数
        
        随机数用强随机数。根据自定义返回对应的字节。因为需要种子seed是int,所以需要4个字节即可,然后用BitConvert静态函数转换得取seed,从而产生较为真实的随机数。
        
    2、蓝球用一个线程,红球用6个线程同时运行。
        蓝球,不用管。
        红球来自01-33不能重复,因此每次都得从6个红球中比较是否有相同的,相同则再次重新产生,直到得到6个不同的数为止。
        6个线程一样这样操作,但容易竞争,比如都比较出与当前6个label数字不一样的,但这两个线程都产生的是相同的数字,比如是5,那么最后就会有重复的情况,为此设置互斥锁,产生数字时只准一个线程进去,改变当前label列表后,再退出,第二个线程再进去比较,得出不同后更新列表,再退出,如此循环,可使当前label的数字一直不同。
        
    3、结果显示。
        为了防止死锁,开一个子线程来等待前面7个线程的结束,并使用await防止假死。
        
        另外,对6个红球排序后,把蓝球放在最末,加入listbox1显示。(不能全部排序,红蓝是两部分)
    
  

更多推荐

Flutter实现地图上汇聚到一点的效果。

要求效果:实现的效果:代码:选择点的界面:import'dart:math';import'package:flutter/material.dart';import'package:get/get.dart';import'package:kq_flutter_widgets/widgets/animate/mapC

hive基于新浪微博的日志数据分析——项目及源码

有需要本项目的全套资源资源以及部署服务可以私信博主!!!该系统的目的是利用大数据技术,分析新浪微博的日志数据,从而探索用户行为、内容传播和移动设备等各个层面的特性和动向。这项研究为公司和个人在制定营销战略、设计产品和提供用户服务时,提供了有价值的参考和辅助。利用Hive平台,该系统可以处理和分析大量的微博数据,为我们提

数据治理-EDRM电子取证

EDRM是电子取证标准和指南的组织,该框架提供了一种电子取证的方法,对于涉及确定相关内部数据的存储方式和位置、适用什么保留策略、哪些数据不可访问以及哪些工具可用于协助识别流程的人员来说,这种方法非常方便。EDRM模型假定数据或信息治理已到位。该模型包括8个可以迭代的电子取证阶段,随着电子取证的发展,可取证的数据和信息的

C++:string类的常用接口说明及其模拟实现

本文主要介绍string类和该类常用的接口,并根据接口功能对其进行模拟实现。目录一、string类的常用接口说明1.string类对象的常见构造2.string类对象的容量操作3.string类对象的访问及遍历操作4.string类对象的修改操作5.string类非成员函数二、string的模拟实现1.默认成员函数和成

【CDN和UDN】CDN和UDN技术特点以及使用场景

内容分发网络(CDN)和用户自定义网络(UDN)是两种不同的网络技术,在选择时,往往不能准备把握具不同的技术特点和应用场景。CDN主要用于加速内容分发,而UDN则主要用于支持用户自定义的网络需求。本文简要介绍下内容分发网络(CDN):一种通过在多个地理位置部署缓存服务器来加速内容分发的网络技术。当用户请求内容时,CDN

【Seata】05 - Seata Saga 模式简单整理、Docker 部署 Nacos 单机(基于 Jpom)相关配置

文章目录前言参考目录Saga模式知识点简单整理1、适用场景、优缺点2、Saga模式的使用3、可能出现的问题以及解决方法Docker部署Nacos单机(基于Jpom)步骤1:拉取镜像步骤2:构建容器步骤3:Nacos设置Seata配置文件步骤4:修改SeataServer相关配置步骤5:修改SeataClient相关配置

@Autowire、@Recourse用啥?

在使用IDEA写Spring相关的项目的时候,在字段上使用@Autowired注解时,总是会有一个波浪线提示:Fieldinjectionisnotrecommended.这是为啥呢?今天就来一探究竟。众所周知,在Spring里面有三种可选的注入方式:构造器注入、Setter方法注入、Field注入,我们先来看下这三种

设计模式:外观模式(C++实现)

外观模式(FacadePattern)是一种结构设计模式,它提供了一个统一的接口,用于访问子系统中的一组接口。外观模式隐藏了子系统的复杂性,使得客户端只需要与外观对象交互,而不需要直接与子系统中的对象进行交互。以下是一个简单的C++外观模式的示例:#include<iostream>//子系统类AclassSubsys

Springboot2.7集成websocket及相关问题

1、集成websocket完整代码导入maven依赖<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency>服务端代码(1)注入be

使用亚马逊云服务器在 G4 实例上运行 Android 应用程序

随着Android应用程序和游戏变得越来越丰富,其中有些甚至比PC上的软件更易于使用和娱乐,因此许多人希望能够在云上运行Android游戏或应用程序,而在EC2实例上运行Android的解决方案可以让开发人员更轻松地测试和运行Android应用程序。在这篇博客文章中,我们将展示如何使用NICEDCV在Anbox中运行A

如何通过文件自动备份软件进行自动化备份?

​为什么要使用文件自动备份软件有一位做客户资料保管登记的朋友,每天会在电脑上录入很多新的客户资料,并需要进行相关维护。比如删掉一些取消合作的客户,或者添加一些备注等等。对于像他这种工作性质的人来说,很需要一个可以进行文件自动备份的软件,无论电脑出现什么问题,可以将重要资料还原。其实每个人有对自己而言特别珍贵或重要的资料

热文推荐