C# - 多线程 之 Process与Thread与ThreadPool

本文介绍.NET框架下进程和线程的管理方式,包括进程类Process的属性与方法、线程优先级设置、线程生命周期及状态管理等内容,并探讨了线程池的使用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Process

进程类,

// 提供对本地和远程进程的访问,启动/停止本地系统进程
public class Process : Component {
	public int Id { get; }  // 系统生成的进程唯一标识符
	public string ProcessName { get; }  // 进程名称
	public string MachineName { get; }  // 运行进程的计算机名称
	public ProcessThreadCollection Threads { get; }  // 进程关联的线程集合(System.Diagnostics.ProcessThread类型的数组)
	public ProcessModule MainModule { get; }  // 进程主模块
	public ProcessModuleCollection Modules { get; } // 进程加载的模块集合(System.Diagnostics.ProcessModule类型的数组)
	public DateTime StartTime { get; } // 进程启动时间
	public DateTime ExitTime { get; }  // 进程退出时间
	public int ExitCode { get; }  // 进程终止时指定的代码值
	public bool HasExited { get; }  // 进程是否已终止
	public TimeSpan TotalProcessorTime { get; }  // 进程的总的处理器时间
    public TimeSpan UserProcessorTime { get; }   // 进程的用户处理器时间
    public TimeSpan PrivilegedProcessorTime { get; }  // 进程的特权处理器时间
	public ProcessPriorityClass PriorityClass { get; set; } // 进程的总体优先级别类别
	public int BasePriority { get; } // 进程的基本优先级别,由PriorityClass计算
	public IntPtr Handle { get; }  // 进程的本机句柄(进程启动时OS为进程分配的句柄,用来跟踪进程属性)
	public int HandleCount { get; }  // 进程打开的操作系统句柄数
	public int WorkingSet { get; }  // 进程当前使用的物理内存总量
	public int PrivateMemorySize { get; }  // 为进程分配的专用内存字节数
	public int VirtualMemorySize { get; }  // 进程的虚拟内存大小
	public bool Responding { get; }  // 进程的用户界面当前是否响应系统
	public int SessionId { get; }  // 进程的终端服务会话标识符

    public Process();
	// System.IO.StreamReader/StreamWriter
	public StreamReader StandardError { get; }  // 用于读取应用程序的标准错误流
    public StreamWriter StandardInput { get; }  // 用于写入应用程序的标准输入流
    public StreamReader StandardOutput { get; } // 用于读取应用程序的标准输出流
    public bool Start([string fileName]);  // 启动进程
    public bool Start(ProcessStartInfo startInfo);  
    public void Kill();    // 停止进程
    public void Close();   // 关闭进程,释放与此组件关联的所有资源
    public void Refresh(); // 放弃进程的、已缓存到该进程组件内的任何信息
    protected override void Dispose(bool disposing);  // 释放进程使用的所有资源
    public bool WaitForExit([int milliseconds]); // 等待进程退出 
    // 获取新的System.Diagnostics.Process组件并与当前活动的进程关联
    public static Process GetCurrentProcess();
    // 获取[指定计算机上]运行的所有进程资源
    public static Process[] GetProcesses([string machineName]);
    // 获取[指定计算机上]由ProcessID对应的进程
    public static Process GetProcessById(int processId [, string machineName]);
    // 获取[指定计算机上]由ProcessName对应的进程
    public static Process[] GetProcessesByName(string processName [, string machineName]);
}

进程优先级

// 指示进程优先级
public enum ProcessPriorityClass { 
    Idle = 64,   // 指定进程的线程只能在系统空闲时运行
    BelowNormal = 16384,
    Normal = 32, // 指定进程没有特殊的安排需求
    AboveNormal = 32768,
    High = 128,  // 指定进程执行必须立即执行的时间关键任务
    RealTime = 256,   // 可能的最高优先级
}

进程优先级与进程的每个线程的优先级值一起确定其每个线程的基本优先级别。

其他进程相关  

// 进程路径信息
string startup-Path = System.Diagnostics.Process.GetCurrentProcess().MainModule.FileName;
或  System.Reflection.Assembly.GetExecutingAssembly().Location;
// 进程目录信息
string startup-Directory = System.IO.Path.GetDirectoryName(startupPath);
或  System.IO.Directory.GetCurrentDirectory();
或  System.Environment.CurrentDirectory;
或  System.Windows.Forms.Application.StartupPath; 

Thread

线程类,.NET中的CLR线程可看作是对操作系统线程的封装,CLR线程与操作系统线程对应,Thread类的实例在调用start()方法后,CLR才会创建一个操作系统线程与之绑定。CLR线程池限制线程的创建速度不超过2个/秒。

属性方法

// 创建并控制线程,设置其优先级并获取其状态
public sealed class Thread : CriticalFinalizerObject, _Thread {
	public int ManagedThreadId { get; } // 托管线程的唯一标识符
	public string Name { get; set; }
	public bool IsAlive { get; }  // 当前线程的执行状态(true:线程已启动且尚未正常终止或中止)
    public bool IsBackground { get; set; }
    public bool IsThreadPoolThread { get; }
    public static Thread CurrentThread { get; } // 获取当前正在运行的线程
    public static Context CurrentContext { get; } // 获取当前线程的上下文
    public ExecutionContext ExecutionContext { get; } // 管理当前线程的上下文
    public ApartmentState ApartmentState { get; set; } // 线程的单元状态 
	public ThreadPriority Priority { get; set; }
    public ThreadState ThreadState { get; }

	// 指定线程启动时要调用的委托方法,(指定线程的最大堆栈大小)
	public Thread(ParameterizedThreadStart start [, int maxStackSize]);
	public Thread(ThreadStart start [, int maxStackSize]);
	public override int GetHashCode(); // 当前线程的哈希代码
	public static int GetDomainID();   // 唯一的应用程序域标识符
	public static AppDomain GetDomain(); // 当前线程正在其中运行的当前应用程序域
	public ApartmentState GetApartmentState();
	public void SetApartmentState(ApartmentState state); // 启动前设置线程的单元状态
	public bool TrySetApartmentState(ApartmentState state); 
	public void Start([object parameter]); // 启动线程(->Running)
	public void Suspend(); // 挂起线程
	public void Resume(); // 继续已挂起的线程
	public void Interrupt(); // 中断处于WaitSleepJoin状态的线程
	public void Join();  // 阻塞调用线程,直到某个线程终止为止
	public void Abort([object stateInfo]); // 终止线程(引发ThreadAbortException、开始终止线程的过程)
	public static void ResetAbort(); // 取消为当前线程请求的Abort(System.Object)
	public static void Sleep(); // 阻塞线程
	public static void SpinWait(int iterations); // 线程等待(iterations定义等待时间)
	public static bool Yield(); // true:操作系统选择并转而执行另一个线程

	public static LocalDataStoreSlot AllocateNamedDataSlot(string name);
	public static LocalDataStoreSlot GetNamedDataSlot(string name);
	public static void FreeNamedDataSlot([string name]);
	public static object GetData(LocalDataStoreSlot slot);
	public static void SetData(LocalDataStoreSlot slot, object data);
	public static Type VolatileRead(ref Type address);  // 读取字段值
	public static void VolatileWrite(ref Type address, Type value); // 字段赋值
}  

其中,ThreadStartParameterizedThreadStart定义委托,均表示在System.Threading.Thread上要执行的方法:

public delegate void ThreadStart();
public delegate void ParameterizedThreadStart(object obj);

其中,ApartmentState指定System.Threading.Thread的单元状态

public enum ApartmentState {
    STA = 0,  // System.Threading.Thread将创建并进入一个单线程单元
    MTA = 1,  // System.Threading.Thread将创建并进入一个多线程单元
    Unknown = 2,  // 单元状态未指定
}

线程调度优先级

public enum ThreadPriority {
    Lowest = 0,      
    BelowNormal = 1,      
    Normal = 2, // 默认优先级    
    AboveNormal = 3,
    Highest = 4,
}  

线程生命周期

线程执行状态 

public enum ThreadState {
    Running = 0,  // 线程已启动,它未被阻塞且没有挂起的System.Threading.ThreadAbortException
    StopRequested = 1,  // 正在请求线程停止(仅用于内部)
    SuspendRequested = 2,  // 正在请求线程挂起
    Background = 4,  // 线程正作为后台线程执行
    Unstarted = 8,  // 尚未对线程调用Start()方法、线程未启动
    Stopped = 16,  // 线程已停止  
    WaitSleepJoin = 32,  // 线程已被阻止
    Suspended = 64,  // 线程已挂起
    AbortRequested = 128,  // 已对线程调用Abort(System.Object)方法,但线程尚未收到System.Threading.ThreadAbortException试图终止它的挂起  
    Aborted = 256, // 线程状态包括AbortRequested且该线程已死,但其状态尚未更改为Stopped
}

其中,ThreadState.WaitSleepJoin状态的可能原因:

  • 调用Sleep(System.Int32)方法或Join()方法;
  • 请求锁定:调用Monitor.Enter(System.Object)方法或Monitor.Wait(System.Object,System.Int32,System.Boolean)方法;
  • 等待线程同步对象:ManualResetEvent或AutoResetEvent; 

ThreadPool

线程池,由系统维护的容纳线程的容器,一个应用程序最多只能有一个线程池。ThreadPool适于并发运行若干个运行时间不长且互不干扰的函数,通过复用有限个固定线程为大量操作服务,减少创建和销毁线程的代价、提高效率。托管线程池中的线程默认为后台线程,即线程的IsBackground属性为true,意味着所有前台线程退出后,ThreadPool线程不会让应用程序保持运行。

  • 不能控制线程池中线程的开始、挂起、和中止;
  • 不适应于执行时间比较长的线程;

 

// 提供一个线程池,该线程池可用于发送工作项、处理异步 I/O、代表其他线程等待以及处理计时器
public static class ThreadPool{
    // 将工作函数排入线程池(,同时指定工作函数需要的参数)
    // 每排入一个工作函数,相当于请求创建一个线程,当线程池中有线程可用时、执行工作函数
    public static bool QueueUserWorkItem(WaitCallback callBack);
    public static bool QueueUserWorkItem(WaitCallback callBack, object state);
    // 设置线程池维护的最小空闲线程数
    public static bool SetMinThreads(int workerThreads, int completionPortThreads);
    // 设置线程池中可同时处于活动状态的线程数目,大于此数目的请求将保持排队状态,直到有线程变为可用
    public static bool SetMaxThreads(int workerThreads, int completionPortThreads);	
    // 检索当前线程池维护的空闲线程的最小数目
    public static void GetMinThreads(out int workerThreads, out int completionPortThreads);
    // 检索当前线程池维护可同时处于活动状态的的线程的最大数目
    public static void GetMaxThreads(out int workerThreads, out int completionPortThreads);
    // 获取当前可用的线程数(= GetMaxThreads - 当前线程池中活动的线程数)
    public static void GetAvailableThreads(out int workerThreads, out int completionPortThreads);
}

其中,WaitCallback表示线程池线程要执行的回调方法:

public delegate void WaitCallback(object state);

以工作中QC框架中的一段代码示例:

 

参考

  •  ;

其他线程池

ManagedThreadPool

由Stephen Toub完全用C#托管代码实现的线程池,quite excellent pool

参考

  •  ;

SmartThreadPool

智能线程池,开源线程池组件,用SmartThreadPool可以简单实现支持多线程的程序,线程池来管理线程、可以减少死锁的出现,同时支持简单的生产者-消费者模式。

  • 下载SmartThreadPool:http://smartthreadpool.codeplex.com/
  • 引用SmartThreadPool.dll

参考

转载于:https://www.cnblogs.com/wjcx-sqh/p/6045027.html

SmartThreadPool是大名鼎鼎的.Net线程池项目,基于.Net开发,比.Net内置的线程池更胜一筹。1、为什么需要使用线程池(Thread Pool)减少线程间上下文切换。线程执行一定的时间片后,系统会自动把cpu切换给另一个线程使用,这时还需要保存当 前的线程上下文状态,并加载新线程的上下文状态。当程序中有大量的线程时,每个线程分得的时间片会越来越少,可能会出现线程未处理多少操作,就需要切换到 另一线程,这样频繁的线程间上下文切换会花费大量的cpu时间。减少内存占用。系统每创建一条物理线程,需要大概花费1MB的内存空间,许多程序喜欢先创建多条物理线程,并 周期轮询来处理各自的任务,这样既消耗了线程上下文切换的时间,还浪费了内存。这些任务可能只需要一条线程就能满足要求。假如某一任务需要执行较长的周 期,线程池还可以自动增加线程,并在空闲时,销毁线程,释放占用的内存。2、为什么不使用.Net默认的线程池.Net默认的线程池(ThreadPool)是一个静态类,所以是没办法自己创建一个新的程序池的。默认的线程池应用程序域 (AppDomain)挂钩,一个AppDomain只有一个线程池。假如在线程池中执行了一个周期较长的任务,一直占用着其中一个线程,可能就会影响到 应用程序域中的其他程序的性能。例如,假如在Asp.Net的线程池中执行一个周期较长的任务,就会影响请求的并发处理能力(线程池默认有个最大线程 数)。 3、SmartThreadPool特性和优点    SmartThreadPool特性如下:可创建线程池实例。可动态调整线程池工作线程数量。WorkItem 可以返回信息。未执行 WorkItem 可被取消。WorkItem 执行时可使用调用者上下文。调用者可等待多个或全部 WorkItem 执行结束。WorkItem 允许拥有一个执行结束时被执行的 PostExecute 回调委托。可以向 WorkItem 传递一个状态对象,并且会在执行结束时自动调用 IDisposable.Dispose()。WorkItem 异常会传递给调用者。支持 WorkItem 分组。可挂起线程池或分组。可以设置 WorkItem 优先级。可以设置线程优先级。4、使用示例 最简单的使用方法:// 创建一个线程池 SmartThreadPool smartThreadPool = new SmartThreadPool();    // 执行任务 smartThreadPool.QueueWorkItem(() => {      Console.WriteLine("Hello World!"); });带返回值的任务:// 创建一个线程池 SmartThreadPool smartThreadPool = new SmartThreadPool();   // 执行任务 var result = smartThreadPool.QueueWorkItem(() => {     var sum = 0;     for (var i = 0; i  {     //模拟计算较长时间     Thread.Sleep(5000);       return 3; });   var result2 = smartThreadPool.QueueWorkItem(() => {     //模拟计算较长时间     Thread.Sleep(3000);       return 5; });   bool success = SmartThreadPool.WaitAll(     new IWorkItemResult[] { result1, result2 });   if (success) {     // 输出结果     Console.WriteLine(result1.Result);     Console.WriteLine(result2.Result); }5、结论 使用SmartThreadPool可以简单就实现支持多线程的程序,由线程池来管理线程,可以减少死锁的出现。SmartThreadPool还支持简单的生产者-消费者模式,当不需要对任务进行持久化时,还是很好用的。 6、扩展阅读 http://www.codeproject.com/KB/threads/smartthreadpool.aspx http://smartthreadpool.codeplex.com/http://www.albahari.com/threading/ 标签:线程池
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值