zoukankan      html  css  js  c++  java
  • 多线程同步之 CriticalSection(临界区)

    先看一段程序, 代码文件:


    unit Unit1;

    interface

    uses
    Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
    Dialogs, StdCtrls;

    type
    TForm1 = class(TForm)
    ListBox1: TListBox;
    Button1: TButton;
    procedure FormCreate(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    end;

    var
    Form1: TForm1;

    implementation

    {$R *.dfm}

    function MyThreadFun(p: Pointer): DWORD; stdcall;
    var
    i: Integer;
    begin
    for i := 0 to 99 do Form1.ListBox1.Items.Add(IntToStr(i));
    Result := 0;
    end;

    procedure TForm1.Button1Click(Sender: TObject);
    var
    ID: DWORD;
    begin
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    end;

    procedure TForm1.FormCreate(Sender: TObject);
    begin
    ListBox1.Align := alLeft;
    end;

    end.



    窗体文件:


    object Form1: TForm1
    Left = 0
    Top = 0
    Caption = 'Form1'
    ClientHeight = 154
    ClientWidth = 214
    Color = clBtnFace
    Font.Charset = DEFAULT_CHARSET
    Font.Color = clWindowText
    Font.Height = -11
    Font.Name = 'Tahoma'
    Font.Style = []
    OldCreateOrder = False
    OnCreate = FormCreate
    PixelsPerInch = 96
    TextHeight = 13
    object ListBox1: TListBox
    Left = 9
    Top = 9
    Width = 121
    Height = 97
    ItemHeight = 13
    TabOrder = 0
    end
    object Button1: TButton
    Left = 131
    Top = 112
    Width = 75
    Height = 25
    Caption = 'Button1'
    TabOrder = 1
    OnClick = Button1Click
    end
    end



    在这段程序中, 有三个线程几乎是同时建立, 向窗体中的 ListBox1 中写数据, 最后写出的结果是这样的:



    能不能让它们别打架, 一个完了另一个再来? 这就要用到多线程的同步技术.
    前面说过, 最简单的同步手段就是 "临界区".

    先说这个 "同步"(Synchronize), 首先这个名字起的不好, 我们好像需要的是 "异步"; 其实异步也不准确...
    管它叫什么名字呢, 它的目的就是保证不冲突、有次序、都发生.

    "临界区"(CriticalSection): 当把一段代码放入一个临界区, 线程执行到临界区时就独占了, 让其他也要执行此代码的线程先等等; 这和前面用的 Lock 和 UnLock 差不多; 使用格式如下:

    var CS: TRTLCriticalSection;   {声明一个 TRTLCriticalSection 结构类型变量; 它应该是全局的}
    InitializeCriticalSection(CS); {初始化}
    EnterCriticalSection(CS); {开始: 轮到我了其他线程走开}
    LeaveCriticalSection(CS); {结束: 其他线程可以来了}
    DeleteCriticalSection(CS); {删除: 注意不能过早删除}

    //也可用 TryEnterCriticalSection 替代 EnterCriticalSection.



    用上临界区, 重写上面的代码, 运行效果图:



    代码文件:


    unit Unit1;

    interface

    uses
    Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
    Dialogs, StdCtrls;

    type
    TForm1 = class(TForm)
    ListBox1: TListBox;
    Button1: TButton;
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    end;

    var
    Form1: TForm1;

    implementation

    {$R *.dfm}

    var
    CS: TRTLCriticalSection;

    function MyThreadFun(p: Pointer): DWORD; stdcall;
    var
    i: Integer;
    begin
    EnterCriticalSection(CS);
    for i := 0 to 99 do Form1.ListBox1.Items.Add(IntToStr(i));
    LeaveCriticalSection(CS);
    Result := 0;
    end;

    procedure TForm1.Button1Click(Sender: TObject);
    var
    ID: DWORD;
    begin
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    end;

    procedure TForm1.FormCreate(Sender: TObject);
    begin
    ListBox1.Align := alLeft;
    InitializeCriticalSection(CS);
    end;

    procedure TForm1.FormDestroy(Sender: TObject);
    begin
    DeleteCriticalSection(CS);
    end;

    end.



    Delphi 在 SyncObjs 单元给封装了一个 TCriticalSection 类, 用法差不多, 代码如下:


    unit Unit1;

    interface

    uses
    Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
    Dialogs, StdCtrls;

    type
    TForm1 = class(TForm)
    ListBox1: TListBox;
    Button1: TButton;
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    end;

    var
    Form1: TForm1;

    implementation

    {$R *.dfm}

    uses SyncObjs;

    var
    CS: TCriticalSection;

    function MyThreadFun(p: Pointer): DWORD; stdcall;
    var
    i: Integer;
    begin
    CS.Enter;
    for i := 0 to 99 do Form1.ListBox1.Items.Add(IntToStr(i));
    CS.Leave;
    Result := 0;
    end;

    procedure TForm1.Button1Click(Sender: TObject);
    var
    ID: DWORD;
    begin
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    CreateThread(nil, 0, @MyThreadFun, nil, 0, ID);
    end;

    procedure TForm1.FormCreate(Sender: TObject);
    begin
    ListBox1.Align := alLeft;
    CS := TCriticalSection.Create;
    end;

    procedure TForm1.FormDestroy(Sender: TObject);
    begin
    CS.Free;
    end;

    end.


    万一老师

    http://www.cnblogs.com/del/archive/2009/02/12/1389368.html
  • 相关阅读:
    [OpenCV]基于arm64和Python2、Python3的opencv-python-contrib编译
    [Jupyter_Notebook]Windows下Jupyter-Notebook更换默认目录
    【Vmware】NAT模式下网络无法连接
    COCO数据集转mask
    [COCO数据集]关于instances中的分割信息按部分类别进行获取及保存
    Leetcode147-对链表进行插入排序(Python3实现)
    Leetcode1415-长度为 n 的开心字符串中字典序第 k 小的字符串(Python3实现)
    Leetcode1353-最多可以参加的会议数目(Python3实现)
    RabbitMQ 官方NET教程(六)【RPC】
    RabbitMQ 官方NET教程(五)【Topic】
  • 原文地址:https://www.cnblogs.com/fengju/p/6173896.html
Copyright © 2011-2022 走看看