[原创][Delphi多线程]使用TMonitor, TEvent和TQueue配合实现TThreadQueue的经典使用案例.

[简介]

常用网名: 猪头三

出生日期: 1981.XX.XX

QQ: 643439947

个人网站: 80x86汇编小站 https://www.x86asm.org

编程生涯: 2001年~至今[共22年]

职业生涯: 20年

开发语言: C/C++、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python

开发工具: Visual Studio、Delphi、XCode、Eclipse、C++ Builder

技能种类: 逆向 驱动 磁盘 文件

研发领域: Windows应用软件安全/Windows系统内核安全/Windows系统磁盘数据安全/macOS应用软件安全

项目经历: 磁盘性能优化/文件系统数据恢复/文件信息采集/敏感文件监测跟踪/网络安全检测

[序言]

本次内容主要是学习TMonitor, TEvent和TQueue配合实现TThreadQueue的经典使用案例. 本次代码的同步等待机制是使用TEvent的WaitFor(), SetEvent()和ResetEvent()方法.

[下面是经典代码]

Delphi 复制代码
unit Unit1;

interface

uses
  Winapi.Windows, Winapi.Messages,
  System.SysUtils, System.Variants, System.Classes, System.Threading, System.SyncObjs,
  System.Generics.Collections,
  Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs, Vcl.StdCtrls;

type
  TForm1 = class(TForm)
    Bn_ReadThread: TButton;
    Bn_WriteThread: TButton;
    Memo_Result: TMemo;
    procedure FormCreate(Sender: TObject);
    procedure FormShow(Sender: TObject);
    procedure Bn_ReadThreadClick(Sender: TObject);
    procedure Bn_WriteThreadClick(Sender: TObject);
  private
    { Private declarations }

    mpr_ShareQueue   : TQueue<string> ; // 共享数据队列
    mpr_Thread_Lock  : TObject ;        // 同步锁
    mpr_Thread_event : TEvent  ;        // 同步事件

    mpr_Threads_Count: Integer ;        // 写者线程的数量

  public
    { Public declarations }
    procedure mpu_pro_Thread_Write(const cstr_param_Data : string) ;

  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
begin

  mpr_ShareQueue   := TQueue<string>.Create() ;
  mpr_Thread_Lock  := TObject.Create();

  mpr_Threads_Count := 0 ;

  // 同步事件初始化: 手工设置信号, 初始化为非信号状态
  mpr_Thread_event := TEvent.Create(nil, True, False,' ') ;

end;// End FormCreate()



procedure TForm1.FormShow(Sender: TObject);
begin
//
end;// End FormShow()



// 单个读者线程
procedure TForm1.Bn_ReadThreadClick(Sender: TObject);
begin

  TTask.Run(procedure
  var
    str_Data : string ;
  begin

    // 循环读取
    while True do
    begin

      // 等待共享数据队列存储新的数据
      mpr_Thread_event.WaitFor(INFINITE) ;

      // 启动同步锁
      System.TMonitor.Enter(mpr_Thread_Lock) ;
      try

        // 获取数据之后, 同步在界面上显式
        if mpr_ShareQueue.Count > 0 then
        begin

          str_Data := mpr_ShareQueue.Dequeue ;

          // 如果共享数据队列没有数据, 那么则通知读者线程进入等待状态
          if mpr_ShareQueue.Count = 0 then
          begin
            mpr_Thread_event.ResetEvent() ;
          end;

          // 界面显式数据
          TThread.Synchronize(TThread.Current, procedure
          begin
            Memo_Result.Lines.Add(str_Data) ;
          end);

        end;

      finally
        System.TMonitor.Exit(mpr_Thread_Lock) ;
      end;

    end;

  end);

end;// End Bn_ReadThreadClick()



procedure TForm1.mpu_pro_Thread_Write(const cstr_param_Data : string) ;
begin

  TTask.Run(procedure
  begin

    // 随机休眠时间
    TThread.Sleep((Random(5)+1)*1000) ;

    // 向共享数据队列添加数据
    // 启动同步锁
    System.TMonitor.Enter(mpr_Thread_Lock) ;
    try

      mpr_ShareQueue.Enqueue(cstr_param_Data) ;

      // 向读者线程发出信号, 共享数据队列有新数据进入
      mpr_Thread_event.SetEvent();

    finally
      System.TMonitor.Exit(mpr_Thread_Lock) ;
    end;

  end);

end;// End mpu_pro_Thread_Write()
procedure TForm1.Bn_WriteThreadClick(Sender: TObject);
var
  str_Data : string ;
begin

  Inc(mpr_Threads_Count) ;


  str_Data := Format('%d 号线程的数据.', [mpr_Threads_Count]) ;
  Bn_WriteThread.Caption := Format('2: 创建写入线程(%d)', [mpr_Threads_Count]) ;

  mpu_pro_Thread_Write(str_Data) ;

end;// End Bn_WriteThreadClick()


end.

[结尾]

上面的代码, 需要注意的细节是: TEvent的WaitFor(), SetEvent()和ResetEvent()替换了TMonitor.Wait()和TMonitor.Pulse()的等待与唤醒同步操作.

相关推荐
island13142 天前
【Java多线程】:理解线程创建、特性及后台进程
java·开发语言·多线程
白狐欧莱雅4 天前
使用Python多线程抓取某图网数据并下载图片
经验分享·爬虫·python·自动化·多线程·图片·drissonpage
Winston Wood4 天前
Android中View.post的用法
android·多线程
hong_zc5 天前
Java 多线程(八)—— 锁策略,synchronized 的优化,JVM 与编译器的锁优化,ReentrantLock,CAS
java·面试·多线程
Themberfue7 天前
Java多线程详解③(全程干货!!!)Thread Runnable
java·开发语言·学习·线程·多线程
亿牛云爬虫专家8 天前
如何用Python同时抓取多个网页:深入ThreadPoolExecutor
python·多线程·爬虫代理·threadpool·代理ip·抓取·足球
铭正8 天前
C++多线程应用
c++·多线程
concisedistinct8 天前
Python的协程与传统的线程相比,是否能更有效地利用计算资源?在多大程度上,这种效率是可测量的?如何量化Python协程的优势|协程|线程|性能优化
python·性能优化·多线程
Winston Wood11 天前
十分钟了解Android Handler、Looper、Message
android·线程·多线程
concisedistinct12 天前
C++游戏开发中的多线程处理是否真的能够显著提高游戏性能?如果多个线程同时访问同一资源,会发生什么?如何避免数据竞争?|多线程|游戏开发|性能优化
c++·游戏·多线程