C#线程基础(线程启动和停止)

目录

一、关于线程

二、示例 

三、生成效果


一、关于线程

        在使用多线程前要先引用命名空间System.Threading,引用命名空间后就可以在需要的地方方便地创建并使用线程。

        创建线程对象的构造方法中使用了ThreadStart()委托,当线程开始执行时,将会执行委托中定义的方法。通过IsBackground属性设置线程为后台线程,调用线程对象的Start方法启动线程,开始执行。CancellationToken方法终止线程。

         创建的线程分为前台线程和后台线程,通过线程对象的IsBackground属性可以设置线程为前台线程或后台线程。前台线程与后台线程的区别是,后台线程不会使托管程序处于运行状态。也就是说,如果进程停止所有前台线程,那么系统会认为此进程处于非运行状态,将会停止所有后台线程并关闭。

        适当地使用线程,可以增加程序的运行效率,使程序运行更加流畅,但是线程间的同步问题也显得尤为重要,可以使用线程锁或监视器有效地处理线程同步问题。

二、示例 

// 线程技术
// 使用随机对象产生一个1~100之间的整数,单击“开始”时,动态生成100个按钮并开始计时;
// 单击事件:按钮上的数字<随机数,那么被单击的按钮变为红色,并显示字符串“小”;
// 单击事件:按钮上的数字>随机数,那么被单击的按钮变为红色,并显示字符串“大”;
// 单击事件:按钮上的数字=随机数就会弹出消息框,提示已经猜对了数字,并显示用时及猜测次数。
namespace _033
{public partial class Form1 : Form{private Button? button1;private Label? label1;private Label? label2;private Label? label3;private Label? label4;Thread? thread_1;                    //定义线程readonly Random random_num = new();  //定义随机数对象int Temp_int_num;                    //定义变量用于存放存机数public Form1(){InitializeComponent();Load += Form1_Load;}private void Form1_Load(object? sender, EventArgs e){// // button1// button1 = new Button{Location = new Point(30, 22),Name = "button1",Size = new Size(75, 23),TabIndex = 0,Text = "开始",UseVisualStyleBackColor = true};button1.Click += Button1_Click;// // label1// label1 = new Label{AutoSize = true,Location = new Point(206, 28),Name = "label1",Size = new Size(44, 17),TabIndex = 1,Text = "计时:"};// // label2// label2 = new Label{AutoSize = true,Location = new Point(256, 28),Name = "label2",Size = new Size(0, 17),TabIndex = 2};// // label3// label3 = new Label{AutoSize = true,Location = new Point(305, 28),Name = "label3",Size = new Size(20, 17),TabIndex = 3,Text = "秒"};// // label4// label4 = new Label{AutoSize = true,Location = new Point(115, 28),Name = "label4",Size = new Size(43, 17),TabIndex = 4,Text = ""};// // Form1// AutoScaleDimensions = new SizeF(7F, 17F);AutoScaleMode = AutoScaleMode.Font;ClientSize = new Size(379, 426);Controls.Add(label4);Controls.Add(label3);Controls.Add(label2);Controls.Add(label1);Controls.Add(button1);Name = "Form1";StartPosition = FormStartPosition.CenterScreen;Text = "猜数字";FormClosed += Form1_FormClosed;}private void Button1_Click(object? sender, EventArgs e){RemoveControl();               //清空所有无用对象int bt_int_x = 10;             //X坐标初始值为10int bt_int_y = 60;             //Y坐标初始值为60for (int i = 0; i < 100; i++)  //添加100个按钮{Button bt = new(){Text = (i + 1).ToString(),  //设置button按钮的文本值Name = (i + 1).ToString(),  //设置button按钮的Name属性Width = 35, Height = 35,   Location = new Point(bt_int_x, bt_int_y)//设置button按钮的位置};bt.Click += new EventHandler(Bt_Click);     //定义button按钮的事件bt_int_x += 36;                             //设置下一个按钮的位置if ((i + 1) % 10 == 0)                      //每10个按钮就换行{bt_int_x = 10;      //换行后重新设置X坐标bt_int_y += 36;   //换行后重新设置Y坐标}Controls.Add(bt);   //将button按钮放入窗体控件集合中}thread_1 = new Thread(delegate ()       //新建一条线程使用匿名方法{int P_int_count = 0;            //初始化计数器while (true)                    //开始无限循环{P_int_count = ++P_int_count > 100000000 ? 0 : P_int_count;  //计数器累加Invoke(                     //将代码交给主线程执行(MethodInvoker)delegate //使用匿名方法{label2!.Text = P_int_count.ToString(); //窗体中显示计数});Thread.Sleep(1000);    //线程睡眠1秒}}){IsBackground = true            //设置线程为后台线程};thread_1.Start();                  //开始执行线程Temp_int_num = random_num.Next(1, 100);//生成随机数label4!.Text = "随机数=" + Temp_int_num.ToString( );button1!.Enabled = false;          //停用开始按钮}/// <summary>/// 强行关闭窗体/// </summary>private void Form1_FormClosed(object? sender, FormClosedEventArgs e){Environment.Exit(0);}/// <summary>/// 开始遍历100个按钮/// 清空窗体中动态生成的按钮/// </summary>void RemoveControl(){for (int i = 0; i < 100; i++){if (Controls.ContainsKey((i + 1).ToString()))   //窗体中是否有此按钮{for (int j = 0; j < Controls.Count; j++)    //遍历窗体控件集合{if (Controls[j].Name == (i + 1).ToString())//是否查找到按钮{Controls.RemoveAt(j);                  //删除指定按钮break;}}}}}/// <summary>/// 用于查找窗体中Enable属性为False控件的数量/// 用于计算玩家有多少次没有猜中/// </summary>/// <returns>返回没有猜中数量</returns>string GetCount(){int P_int_temp = 0;                 //初始化计数器foreach (Control c in Controls)     //遍历控件集合{if (!c.Enabled) P_int_temp++;   //计数器累加}return P_int_temp.ToString();       //返回计数器信息}void Bt_Click(object? sender, EventArgs e){Control? P_control = sender as Control; //将sender转换为control类型对象if (int.Parse(P_control!.Name) > Temp_int_num){P_control.BackColor = Color.Red;     //设置按钮背景为红色P_control.Enabled = false;           //停用按钮P_control.Text = "大";               //更改按钮文本}if (int.Parse(P_control.Name) < Temp_int_num){P_control.BackColor = Color.Red;     //设置按钮背景为红色P_control.Enabled = false;           //设置按钮停用P_control.Text = "小";               //更改按钮文本}if (int.Parse(P_control.Name) == Temp_int_num){//thread1.Abort();CancellationToken cancellationToken = new();CancellationToken token = cancellationToken;Thread_Abort(token);MessageBox.Show(string.Format("恭喜你猜对了!共猜了{0}次 用时{1}秒",GetCount(), label2!.Text), "恭喜!");  //显示游戏信息button1!.Enabled = true;                   //启用开始按钮}}/// <summary>/// 用于代替thread1!.Abort();/// SYSLIB0006:不支持 Thread.Abort/// 替代方案我没有深究是否有效,仅提供指明的解决方向/// </summary>static void Thread_Abort(CancellationToken cancellationToken){         // If the CancellationToken is marked as "needs to cancel",// this will throw the appropriate exception.cancellationToken.ThrowIfCancellationRequested();}}
}

三、生成效果

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/592462.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

LeetCode第32题 : 最长有效括号

题目介绍 给你一个只包含 ( 和 ) 的字符串&#xff0c;找出最长有效&#xff08;格式正确且连续&#xff09;括号子串的长度。 示例 1&#xff1a; 输入&#xff1a;s "(()" 输出&#xff1a;2 解释&#xff1a;最长有效括号子串是 "()" 示例 2&#xf…

springCould中的Hystrix【上】-从小白开始【7】

目录 1.简单介绍❤️❤️❤️ 2.主要功能 ❤️❤️❤️ 3.正确案例❤️❤️❤️ 4.使用jmeter压测 ❤️❤️❤️ 5.建模块 80❤️❤️❤️ 6.如何解决上面问题 ❤️❤️❤️ 7.对8001进行服务降级❤️❤️❤️ 8.对80进行服务降级 ❤️❤️❤️ 9.通用降级方法❤️❤️…

1.2 day2 IO进程线程

使用fread、fwrite完成文件拷贝 #include <myhead.h> int main(int argc, const char *argv[]) {if(argc!3){printf("参数有误");}//定义并以只写的方式打开两个文件FILE *fpNULL;FILE *cfpNULL;if((fpfopen(argv[1],"w"))NULL){perror("fopen…

学习Vue单文件组件总结

今天主要学习了组件实例对象的一个重要内置关系和单文件组件。先说一下实例对象的内置关系&#xff0c;在这里要对JS中的原型链有一定的基础&#xff0c;Vue构造函数的prototype原型指向的是Vue的原型对象&#xff0c;new出来的Vue实例对__proto__同样指向的是Vue的原型对象&am…

harbor自建san证书

1.创建证书存放目录 mkdir -p /opt/harbor/harbor/cert && cd /opt/harbor/harbor/cert 2.拷贝openssl配置文件 cp /etc/pki/tls/openssl.cnf /opt/harbor/harbor/cert 3.编辑拷贝出来的openssl配置文件 vi openssl.cnf [ CA_default ] copy_extensions copy …

异地环控设备如何远程维护?贝锐蒲公英解决远程互联难题

青岛某企业致力于孵化设备、养禽设备和养猪设备的研发、生产和服务&#xff0c;历经三十多年发展&#xff0c;目前已成长为行业主要的养殖装备及工程服务提供商&#xff0c;产品覆盖养殖产业链中绝大多数环节&#xff0c;涉及自动化设备、环控设备、整体解决方案等。 在实际应用…

基于Rangenet Lib的自动驾驶LiDAR点云语义分割与可视化

这段代码是一个C程序&#xff0c;用于处理来自KITTI数据集的激光雷达&#xff08;LiDAR&#xff09;扫描数据。程序主要实现以下功能&#xff1a; 1. **读取和解析命令行参数**&#xff1a;使用Boost库中的program_options模块来定义和解析命令行参数。这包括扫描文件路径、模型…

docker安装postgresql15或者PG15

1. 查询版本 docker search postgresql docker pull postgres:15.3 # 也可以拉取其他版本2.运行容器并挂载数据卷 mkdir -p /data/postgresql docker run --name postgres \--restartalways \-e POSTGRES_PASSWORDpostgresql \-p 5433:5432 \-v /data/postgresql:/var/lib/p…

计算机网络-以太网交换基础

一、网络设备的演变 最初的网络在两台设备间使用传输介质如网线等进行连接就可以进行通信。但是随着数据的传输需求&#xff0c;多个设备需要进行数据通信时就需要另外的设备进行网络互联&#xff0c;并且随着网络传输的需求不断更新升级。从一开始的两台设备互联到企业部门内部…

Nginx多ip部署多站点

目录 1.修改网卡配置信息 2.修改主要配置文件nginx.conf 1.修改网卡配置信息 1)来到网卡配置文件存放目录下 cd /etc/sysconfig/network-scripts/ 2)对 ifcfg-ens33 文件进行配置修改前先进行备份 cp ifcfg-ens33 ifcfg-ens33.default 3)先修改成最小配置&#xff0c;使用 d…

在线H5网页版植物大战僵尸游戏源代码

源码介绍 HTML5植物大战僵尸网页版游戏源码&#xff0c;直接把源码上传到服务器就能使用和访问

裁员+失恋或许不能比这更遭了,敬一塌糊涂与充满感动的2023,也敬曾经的挚爱与寒冬的冰霜

~ 随机抽取评论区的 3位 小伙伴送上精美礼品 ~ 参与方式&#xff1a;关注、点赞、收藏&#xff0c;评论 "2024&#xff0c;一天当做两天卷&#xff01;" 活动时间&#xff1a;截止到 2024-01-21 00:00:00 礼品清单&#xff1a;CSDN活动周边、自选图书 本文目录 序 …

React实现拖拽效果

基于 React 的拖拽效果 Demo 一个基于 React 的拖拽功能实现的 Demo. 两个关键点 1, draggable 属性 2, drag 事件 draggable 属性 img 标签默认是支持拖拽的, 当时其他 HTML 标签, 想要其拖动的话, 需要为其添加 draggable“true” 属性 drag 事件 drag 相关的事件有:…

C++ STL set用法详解

我们都知道&#xff0c;set是STL里的一种数据结构&#xff0c;这篇博客就是set用法的详解。 1.set的创建。 set初始化一般是 set<数据结构名称> 名字; 具体例子&#xff1a; 创建一个int型&#xff0c;名称是s的set。 set<int> s; set还可以创建STL里的数据…

Element-ui自定义input框非空校验

1、vue自定义非空指令&#xff1a; main.js中自定义非空指令 当input框或下拉框中数据更新时&#xff0c;触发校验 Vue.directive(isEmpty,{update:function(el,binding,vnode){if(vnode.componentInstance.value""){el.classList.add("is-required");}e…

2024更新腾讯云轻量应用服务器优惠价格表和CVM优惠活动

腾讯云服务器租用价格表&#xff1a;轻量应用服务器2核2G3M价格62元一年、2核2G4M价格118元一年&#xff0c;540元三年、2核4G5M带宽218元一年&#xff0c;2核4G5M带宽756元三年、轻量4核8G12M服务器446元一年、646元15个月&#xff0c;云服务器CVM S5实例2核2G配置280.8元一年…

LVDS接口ADC数据处理流程案例参考

一ADC&#xff1a; 16bit精度DDRLVDS8个outpin 二&#xff1a;FPGA处理流程&#xff1a; 1.差分数据转单端idelaye2generate for generate for(i0;i<7;ii1)begin:GEN_IN IBUFDS #(.DIFF_TERM("TRUE"), // Differential Termination.IBUF_LOW_PWR("F…

分布式【zookeeper面试题12连问】

1. 面试官&#xff1a;工作中使用过Zookeeper嘛&#xff1f;你知道它是什么&#xff0c;有什么用途呢&#xff1f; 「小菜鸡的我&#xff1a;」 有使用过的&#xff0c;使用ZooKeeper作为**「dubbo的注册中心」&#xff0c;使用ZooKeeper实现「分布式锁」**。ZooKeeper&#…

k8s中实现pod自动扩缩容

一、k8s应用自动扩缩容概述 1&#xff09;背景&#xff1a; 在实际的业务场景中&#xff0c;我们经常会遇到某个服务需要扩容的场景&#xff08;例如&#xff1a;测试对服务压测、电商平台秒杀、大促活动、或由于资源紧张、工作负载降低等都需要对服务实例数进行扩缩容操作&…

vue3(十三)-基础入门之路由配置与重定向

一、一级路由与重定向 1、创建 App.vue 在父组件中导入子组件 Navbar <template><div><navbar></navbar></div> </template><style lang"scss"></style><script> import navbar from /components/Navbarex…