C# 多线程编程二

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
实时数仓Hologres,5000CU*H 100GB 3个月
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
简介: C# 多线程编程二

多线程基础02

对锁而言,尽可能得使用父对象(级别更大得对象内容)。

通知

通知

是指一个线程告诉另一个线程等待另一个线程回馈结果的操作。

如果没有得到反馈该线程会一直等待,直到另一个线程反馈为止。

通知一般使用Join方法,也可以使用同步类对象

同 步 基 元 说 明
Monitor 监视器,支持锁定操作,防止一个或多个线程同时访问资源
Mutex 互斥器,支持锁定操作,防止一个或多个线程同时访问资源
ReaderWriterLock 读写锁,支持锁定操作,定义支持单个写线程和多个读线程的锁
Semaphore 信号量,也叫信号灯,阻塞线程直到另一个线程信号通知
AutoResetEvent 事件类,支持通知同步,阻塞线程直到另一个线程设置事件
ManualResetEvent 事件类,支持通知同步,阻塞线程直到另一个线程设置事件

同步类对象

AutoResetEvent

通知正在等待的线程已发生事件

提供一种通知机制,可以控制线程执行的先后顺序

AutoResetEvent /ManualResetEvent类的重要方法

方 法 说 明
Set 设置并发送信号
Reset 重置信号,也就是使信号无效
WaitOne 等待一个信号,如果没有信号到来则等待
ManualResetEvent.WaitAny 静态方法,等待一个信号数组,信号数组里面有任何信号到都可以,否则等待
ManualResetEvent.WaitAll 静态方法,等待一个信号数组,信号数组里面的信号全部到齐才可以,否则等待

创建AutoResetEvent对象

AutoResetEvent myResetEvent = new AutoResetEvent(false);

参数说明:

false表示事件开始是无信号状态,

当参数为true表示创建的事件开始是有信号的,就相当于使用false参数创建事件后立即调用了Set方法。

俗称信号灯

orderEvent.WaitOne() //等待信号,一直等

orderEvent.Set() //发出信号,等待的子线程执行

线程池

现状

创建和销毁线程代价高

线程池

线程池是系统自己维护的线程的集合。

线程池技术减少频繁的线程创建与销毁对系统性能的影响

对于每一个进程系统都会给其创建一个线程池

如果想要执行线程操作,只需要向线程池发出一个执行某个操作的请求即可。

CLR线程池

•1) 最小线程数,线程池的线程总大于等于这个值,一般这个值设置为逻辑CPU数,也就是能充分利用CPU同时执行这些线程。

•2) 最大线程数,默认1000,不建议修改这个值,如果这个值过小,很可能运行的线程的都被阻塞,而排队的线程永远得不到执行。

•3) 线程池是非常智能的,并不会发现可用线程不够马上创建新的线程,而是会有一个延迟以确保真的需要新的线程来补充(因为也不建议线程池中的方法执行时间太长比如超过500毫秒,影响线程池的判断)。线程池的目的就是减少实际线程的创建和回收,重复利用线程来做不同的工作。

ThreadPool

线程池静态类

托管线程池中的线程为后台线程,即它们的 IsBackground 属性为 true。这意味着在所有的前台线程都已退出后,ThreadPool 线程不会让应用程序保持运行。

向线程池提交任务:

原型:
static bool QueueUserWorkItem( WaitCallback callBack, Object state )

举例:
WaitCallback callBackCheck=new WaitCallback(this.Check); ThreadPool.QueueUserWorkItem(callBackCheck,info);

参数WaitCallback原型
委托
public delegate void WaitCallback (Object state);

线程池的一些重要知识

ThreadPool中的Thread不能手动取消,也不用手动开始。所以ThreadPool并不适用比较长的线程。只需要把一个 WaitCallback委托塞给ThreadPool,然后剩下的工作将由系统自动完成。系统会在ThreadPool的线程队列中一一启动线程。

当线程池满时,多余的线程会在队列里排队,当线程池空闲时,系统自动调入排队的线程,以保持系统利用率。

当需要复杂的同步技术,例如事件,或需要对一个线程调用Join方法时线程池就不能满足需求了.

以下情况中不宜使用ThreadPool而应该使用单独的Thread:
1.需要为线程指定详细的优先级
2.线程执行需要很长时间
3.需要前台线程。
4.在线程执行中需要对线程操作,如打断,挂起等

总结

进程和线程相关概述

进程就是一个正在执行的应用程序。是系统进行资源分配的基本单位。

线程是在进程的内部执行的指令序列,共享进程的内存和系统资源。

多线程编程概述

v.Net**下如何创建线程**

Thread

线程的不同状态

ThreadState

线程同步技术

加锁:lock,monitor

通知:AutoResetEvent

线程池

类的扩展方法的简单使用:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.IO.Ports;
using System.Runtime.CompilerServices;

namespace day31test03
{
   
    class Program
    {
   
        /// <summary>
        /// 扩展方法
        /// </summary>
        /// <param name="args"></param>
        static void Main(string[] args)
        {
   
            //SerialPort port = new SerialPort();
            Test test = new Test();
            test.ddd(4, 5);
            test.add(4, 5);
        }
    }
    class Test
    {
   
        public void add(int a, int b)
        {
   
            Console.WriteLine(a+b);
        }
    }
    static class T
    {
   
        public static int ddd(this Test t, int x, int y)
        {
   
            return 2*(x - y);
        }
    }
}

并行的多线程,让CPU多个核都能跑起来。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace day31test05
{
   
    class Program
    {
   
        /// <summary>
        /// 并行的多线程
        /// 多任务
        /// </summary>
        /// <param name="args"></param>
        static void Main(string[] args)
        {
   
            Action action = new Action(Dowork);
            System.Threading.Tasks.Task task = new Task(action);
            task.Start();

            Action action1 = new Action(Dowork);
            System.Threading.Tasks.Task task1 = new Task(action1);
            task1.Start();
            Console.Read();
        }
        public static void Dowork()
        {
   
            do
            {
   

            } while (true);
        }
    }
}

匿名方法委托,并行for和foreach

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace day31test06
{
   
    class Program
    {
   
        /// <summary>
        /// 基于匿名方法的多线程
        /// 避免了委托
        /// </summary>
        /// <param name="args"></param>
        static void Main(string[] args)
        {
   
            /*System.Threading.Thread t3 = new System.Threading.Thread(a =>
            {
                Console.WriteLine("基于匿名方法的多线程");
            });
            t3.Start();*/

            //并行for,特点:快
            //System.Threading.Tasks.Parallel.For(0, 100, a => { System.Threading.Thread.Sleep(100); Console.WriteLine(System.DateTime.Now); });

            //并行foreach,特点:
            String[] str = {
    "1" , "s", "w"};
            System.Threading.Tasks.Parallel.ForEach(str, (item, kk) => {
    Console.WriteLine(item); });
            Console.Read();
        }
    }
}
目录
相关文章
|
2月前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
17天前
|
存储 安全 编译器
学懂C#编程:属性(Property)的概念定义及使用详解
通过深入理解和使用C#的属性,可以编写更清晰、简洁和高效的代码,为开发高质量的应用程序奠定基础。
65 12
|
15天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
98 2
|
2月前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
1月前
|
缓存 Java 调度
多线程编程核心:上下文切换深度解析
在现代计算机系统中,多线程编程已成为提高程序性能和响应速度的关键技术。然而,多线程编程中一个不可避免的概念就是上下文切换(Context Switching)。本文将深入探讨上下文切换的概念、原因、影响以及优化策略,帮助你在工作和学习中深入理解这一技术干货。
48 10
|
2月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
1月前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
1月前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
51 3
|
1月前
|
算法 调度 开发者
多线程编程核心:上下文切换深度解析
在多线程编程中,上下文切换是一个至关重要的概念,它直接影响到程序的性能和响应速度。本文将深入探讨上下文切换的含义、原因、影响以及如何优化,帮助你在工作和学习中更好地理解和应用多线程技术。
43 4
|
12天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程