zoukankan      html  css  js  c++  java
  • delphi 对TThread扩充TSimpleThread

    转载自:https://www.cnblogs.com/lackey/p/5371544.html

    对线程的使用,是每个开发者都应该熟练掌握的,也是进阶的重要一环。

    可以这样说,没有线程,连界面假死的问题都解决不了,就更别谈并行处理来提高效率了。

    本例对线程进行改进,打造一个基础的线程,以后线程应用都从此类继承,大大节省了代码,提高了效率。

    经长期实践,此代码能够应付许多情况,值得一学。

    它的应用1:TReadHtmlThread (读网页)

    它的应用2: TElegantThread (把多个线程的请求阻塞到另一个线程)

    它的应用3: TThreadTimer 多线程 Timer 

    unit uSimpleThread;
    interface
    uses
      System.Classes, System.SysUtils, System.SyncObjs;
    
    type
    
      // 显示信息,调用方法 DoOnStatusMsg(AMsg);
      TOnStatusMsg = procedure(AMsg: string) of object;
    
      // 显示调试信息,一般用于显示出错信息,用法 DoOnDebugMsg(AMsg);
      TOnDebugMsg = TOnStatusMsg;
    
      TSimpleThread = class(TThread)
      public type // "执行过程"的类别定义
    
        TGeneralProc = procedure; // 普通的,即 procedure DoSomeThing;
        TObjectProc = procedure of object; // 类的,即 TXxxx.DoSomeThign; 用得多
        TAnonymousProc = reference to procedure; // 匿名的
      private type
        TProcKind = (pkGeneral, pkObject, pkAnonymous); // "执行过程"的类别
      private
    
        FGeneralProc: TGeneralProc;
        FObjProc: TObjectProc;
        FAnoProc: TAnonymousProc;
    
        FProcKind: TProcKind;
    
        FEvent: TEvent; // 用于阻塞,它是一个信号量
        FActiveX: boolean; // 是否在线程中支持 Com ,如果你要在线程中访问 IE 的话,就设定为 True
    
        FOnStatusMsg: TOnStatusMsg;
        FOnDebugMsg: TOnDebugMsg;
    
        FTagID: integer; // 给线程一个代号,在线程池的时候用来作区别
        FParam: integer; // 给线程一个参数,方便识别
    
        procedure SelfStart; // 触发线程运行
    
        procedure DoExecute; // 这个函数里面运行的代码是“线程空间”
        procedure DoOnException(e: exception); // 异常信息显示 调用 DoOnDebugMsg(AMsg);
    
        procedure SetTagID(const Value: integer);
        procedure SetParam(const Value: integer);
    
        procedure SetOnStatusMsg(const Value: TOnStatusMsg);
        procedure SetOnDebugMsg(const Value: TOnDebugMsg);
    
      protected
    
        FWaitStop: boolean; // 结束标志,可以在继承类中使用它,以确定线程是否停止运行
    
        procedure DoOnStatusMsg(AMsg: string); // 显示普通信息
        procedure DoOnDebugMsg(AMsg: string); // 显示调式信息
    
        procedure Execute; override; // 重载 TThread.Execute
    
        procedure OnThreadProcErr(e: exception); virtual; // 异常发生事件
    
        procedure WaitThreadStop; // 等待线程结束
    
        procedure BeforeExecute; virtual; // 看名字,不解释
        Procedure AfterExecute; virtual; // 看名字,不解释
    
        procedure SleepExceptStopped(ATimeOut: Cardinal); // 这个高大上了,要解释一下。
        { 有时线程没有任务时,就会休息一会儿,但是,休息的时候,可能会接收到退出线程的指令
          此函数就是在休息的时候也检查一下停止指令
        }
    
      public
    
        // 改变一下 Create 的参数,AllowedActiveX:是否允许线程代码访问 Com
        constructor Create(AllowedActiveX: boolean = false); reintroduce;
    
        destructor Destroy; override;
    
        procedure ExeProcInThread(AProc: TGeneralProc); overload; // 这三个,对外的接口。
        procedure ExeProcInThread(AProc: TObjectProc); overload;
        procedure ExeProcInThread(AProc: TAnonymousProc); overload;
    
        procedure StartThread; virtual;
        { 启动线程,一般只调用一次。
          以后就由线程的响应事件来执行了
        }
    
        procedure StopThread; virtual; // 停止线程
    
        property OnStatusMsg: TOnStatusMsg read FOnStatusMsg write SetOnStatusMsg;
        property OnDebugMsg: TOnDebugMsg read FOnDebugMsg write SetOnDebugMsg;
        property WaitStop: boolean read FWaitStop;
        property TagID: integer read FTagID write SetTagID;
        property Param: integer read FParam write SetParam;
    
      end;
    
    implementation
    
    uses
      ActiveX;
    
    procedure TSimpleThread.AfterExecute;
    begin
    end;
    
    procedure TSimpleThread.BeforeExecute;
    begin
    end;
    
    constructor TSimpleThread.Create(AllowedActiveX: boolean);
    var
      BGUID: TGUID;
    begin
      inherited Create(false);
      FActiveX := AllowedActiveX;
      FreeOnTerminate := false; // 我们要手动Free线程
      CreateGUID(BGUID);
      FEvent := TEvent.Create(nil, true, false, GUIDToString(BGUID));
    end;
    
    destructor TSimpleThread.Destroy;
    begin
      StopThread; // 先停止
      WaitThreadStop; // 再等待线程停止
      {
        在继承类的 Destroy 中,也要写上这两句. 如:
        暂时未找到更好的办法,这点代码省不了
        destructor TXXThread.Destroy;
        begin
        StopThread;
        WaitThreadStop;
        xxx.Free;
        Inherited;
        end;
      }
      FEvent.Free;
      inherited;
    end;
    
    procedure TSimpleThread.DoExecute; // 此函数内执行的代码,就是在多线程空间里运行
    begin
      BeforeExecute;
      repeat
    
        FEvent.WaitFor;
        FEvent.ResetEvent; // 下次waitfor 一直等
        { 这里尝试了很多些,总 SelfStart 觉得有冲突,经过多次修改并使用证明,
          没有必要在这里加锁,因为只调用 startThread 一次,剩下的交给线程影应事件
        }
    
        if not Terminated then // 如果线程需要退出
        begin
    
          try
    
            case FProcKind of
              pkGeneral: FGeneralProc;
              pkObject: FObjProc;
              pkAnonymous: FAnoProc;
            end;
    
          except
    
            on e: exception do
            begin
              DoOnException(e);
            end;
    
          end;
    
        end;
    
      until Terminated;
      AfterExecute;
      //代码运行到这里,就表示这个线程不存在了。再也回不去了,必须释放资源了。
    end;
    
    procedure TSimpleThread.DoOnDebugMsg(AMsg: string);
    begin
      if Assigned(FOnDebugMsg) then
        FOnDebugMsg(AMsg);
    end;
    
    procedure TSimpleThread.DoOnException(e: exception);
    var
      sErrMsg: string;
    begin
      sErrMsg := 'ClassName:' + ClassName + #13#10;
      sErrMsg := sErrMsg + 'TagID:' + IntToStr(FTagID) + #13#10;
      sErrMsg := sErrMsg + 'Param:' + IntToStr(Param) + #13#10;
      sErrMsg := sErrMsg + 'ErrMsg:' + e.Message + #13#10;
      DoOnDebugMsg(sErrMsg);
      OnThreadProcErr(e);
    end;
    
    procedure TSimpleThread.DoOnStatusMsg(AMsg: string);
    begin
      if Assigned(FOnStatusMsg) then
        FOnStatusMsg(AMsg);
    end;
    
    procedure TSimpleThread.Execute;
    begin
      //是否支持 Com
      if FActiveX then
      begin
        CoInitialize(nil);
        try
          DoExecute;
        finally
          CoUninitialize;
        end;
      end
      else
        DoExecute;
    end;
    
    procedure TSimpleThread.ExeProcInThread(AProc: TGeneralProc);
    begin
      FGeneralProc := AProc;
      FProcKind := pkGeneral;
      SelfStart;
    end;
    
    procedure TSimpleThread.ExeProcInThread(AProc: TObjectProc);
    begin
      FObjProc := AProc;
      FProcKind := pkObject;
      SelfStart;
    end;
    
    procedure TSimpleThread.ExeProcInThread(AProc: TAnonymousProc);
    begin
      FAnoProc := AProc;
      FProcKind := pkAnonymous;
      SelfStart;
    end;
    
    procedure TSimpleThread.OnThreadProcErr(e: exception);
    begin;
    end;
    
    procedure TSimpleThread.SelfStart;
    begin
      //经常多次尝试,最终写成这样,运行没有问题
      if FEvent.WaitFor(0) <> wrSignaled then
        FEvent.SetEvent; // 让waitfor 不再等
    end;
    
    procedure TSimpleThread.StopThread;
    begin
      //继承类的代码中,需要检查 FWaitStop ,来控制线程结束
      FWaitStop := true;
    end;
    
    procedure TSimpleThread.SetOnDebugMsg(const Value: TOnDebugMsg);
    begin
      FOnDebugMsg := Value;
    end;
    
    procedure TSimpleThread.SetOnStatusMsg(const Value: TOnStatusMsg);
    begin
      FOnStatusMsg := Value;
    end;
    
    procedure TSimpleThread.SetParam(const Value: integer);
    begin
      FParam := Value;
    end;
    
    procedure TSimpleThread.SetTagID(const Value: integer);
    begin
      FTagID := Value;
    end;
    
    procedure TSimpleThread.SleepExceptStopped(ATimeOut: Cardinal);
    var
      BOldTime: Cardinal;
    begin
      // sleep 时检测退出指令,以确保线程顺序退出
      // 多个线程同时工作,要保证正确退出,确实不容易
      BOldTime := GetTickCount;
      while not WaitStop do
      begin
        sleep(50);
        if (GetTickCount - BOldTime) > ATimeOut then
          break;
      end;
    end;
    
    procedure TSimpleThread.StartThread;
    begin
      FWaitStop := false;
    end;
    
    procedure TSimpleThread.WaitThreadStop;
    begin
      //等待线程结束
      StopThread;
      Terminate;
      SelfStart;
      WaitFor;
    end;
    
    end.
    
    uSimpleThread.pas
    

      

  • 相关阅读:
    函数节流和防抖
    记一次面试
    继承
    对象的几种创建方法
    对象的简单认识
    HTTP、HTTPS、SOCKS代理的概念(到底是什么意思?)
    Nginx 相关介绍(正向代理和反向代理区别)
    Markdown:怎么用?以及为什么要用Markdown?
    Google 凭什么要赔给 Oracle 88 亿?
    雷军:《我十年的程序员生涯》系列之三(失败的大学创业经历)
  • 原文地址:https://www.cnblogs.com/approx/p/11852398.html
Copyright © 2011-2022 走看看