zoukankan      html  css  js  c++  java
  • C++线程同步与互斥之互斥体

    抢红包练习

    第一步:在第一个文本框中输入一个值,比如1000;
    第二步:点击抢红包,同时创建3个线程,每个线程循环进行抢红包的操作,每次抢50;
    第三步:使用Mutex进行线程控制,当第一个文本框中的值<50时,强红包线程结束.
    特别说明:
    1、四个文本框中的值总和应该为1000
    2、强红包线程每次延时50毫秒.
    3、使用WaitForMultipleObjects监听所有线程,当线程全部结束后,调用CloseHandle关闭句柄.

    下面是代码实现,我是用MFC实现的。

    // MutexExDlg.h : 头文件
    //
    
    #pragma once
    
    
    // CMutexExDlg 对话框
    class CMutexExDlg : public CDialogEx
    {
    // 构造
    public:
        CMutexExDlg(CWnd* pParent = NULL);    // 标准构造函数
    
    // 对话框数据
        enum { IDD = IDD_MUTEXEX_DIALOG };
    
        protected:
        virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持
        static HANDLE hThread[3];
        static HANDLE m_Mutex;
        static DWORD WINAPI ThreadProc0(LPVOID lpParameter);
        static DWORD WINAPI ThreadProc1(LPVOID lpParameter);
        static DWORD WINAPI ThreadProc2(LPVOID lpParameter);
        static DWORD WINAPI ThreadProc3(LPVOID lpParameter);
    
    
    // 实现
    protected:
        HICON m_hIcon;
    
        // 生成的消息映射函数
        virtual BOOL OnInitDialog();
        afx_msg void OnPaint();
        afx_msg HCURSOR OnQueryDragIcon();
        DECLARE_MESSAGE_MAP()
    public:
        virtual BOOL PreTranslateMessage(MSG* pMsg);
        afx_msg void OnBnClickedButton1();
        static int m_Edit0;
        static int m_Edit1;
        static int m_Edit2;
        static int m_Edit3;
    };
    // MutexExDlg.cpp : 实现文件
    //
    
    #include "stdafx.h"
    #include "MutexEx.h"
    #include "MutexExDlg.h"
    #include "afxdialogex.h"
    
    #ifdef _DEBUG
    #define new DEBUG_NEW
    #endif
    
    
    // CMutexExDlg 对话框
    
    
    
    CMutexExDlg::CMutexExDlg(CWnd* pParent /*=NULL*/)
        : CDialogEx(CMutexExDlg::IDD, pParent)
    {
        m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
    }
    
    void CMutexExDlg::DoDataExchange(CDataExchange* pDX)
    {
        CDialogEx::DoDataExchange(pDX);
        DDX_Text(pDX, IDC_EDIT1, m_Edit0);
        DDV_MinMaxInt(pDX, m_Edit0, 0, 1000);
        DDX_Text(pDX, IDC_EDIT2, m_Edit1);
        DDV_MinMaxInt(pDX, m_Edit1, 0, 1000);
        DDX_Text(pDX, IDC_EDIT3, m_Edit2);
        DDV_MinMaxInt(pDX, m_Edit2, 0, 1000);
        DDX_Text(pDX, IDC_EDIT4, m_Edit3);
        DDV_MinMaxInt(pDX, m_Edit3, 0, 1000);
    }
    
    BEGIN_MESSAGE_MAP(CMutexExDlg, CDialogEx)
        ON_WM_PAINT()
        ON_WM_QUERYDRAGICON()
        ON_BN_CLICKED(IDC_BUTTON1, &CMutexExDlg::OnBnClickedButton1)
    END_MESSAGE_MAP()
    
    
    // CMutexExDlg 消息处理程序
    
    BOOL CMutexExDlg::OnInitDialog()
    {
        CDialogEx::OnInitDialog();
    
        // 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
        //  执行此操作
        SetIcon(m_hIcon, TRUE);            // 设置大图标
        SetIcon(m_hIcon, FALSE);        // 设置小图标
    
        // TODO:  在此添加额外的初始化代码
    
        return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
    }
    
    // 如果向对话框添加最小化按钮,则需要下面的代码
    //  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
    //  这将由框架自动完成。
    
    void CMutexExDlg::OnPaint()
    {
        if (IsIconic())
        {
            CPaintDC dc(this); // 用于绘制的设备上下文
    
            SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
    
            // 使图标在工作区矩形中居中
            int cxIcon = GetSystemMetrics(SM_CXICON);
            int cyIcon = GetSystemMetrics(SM_CYICON);
            CRect rect;
            GetClientRect(&rect);
            int x = (rect.Width() - cxIcon + 1) / 2;
            int y = (rect.Height() - cyIcon + 1) / 2;
    
            // 绘制图标
            dc.DrawIcon(x, y, m_hIcon);
        }
        else
        {
            CDialogEx::OnPaint();
        }
    }
    
    //当用户拖动最小化窗口时系统调用此函数取得光标
    //显示。
    HCURSOR CMutexExDlg::OnQueryDragIcon()
    {
        return static_cast<HCURSOR>(m_hIcon);
    }
    
    
    // 重写虚函数 PreTranslateMessage 屏蔽掉Esc键和Enter键
    BOOL CMutexExDlg::PreTranslateMessage(MSG* pMsg)
    {
        if (pMsg->message == WM_KEYDOWN)
        {
            int keyCode = (int)pMsg->wParam;
            if (keyCode == VK_ESCAPE || keyCode == VK_RETURN)
            {
                return TRUE;
            }
        }
        return CDialogEx::PreTranslateMessage(pMsg);
    }
    
    HANDLE CMutexExDlg::m_Mutex = NULL;
    int CMutexExDlg::m_Edit0 = 0;
    int CMutexExDlg::m_Edit1 = 0;
    int CMutexExDlg::m_Edit2 = 0;
    int CMutexExDlg::m_Edit3 = 0;
    HANDLE CMutexExDlg::hThread[3] = { NULL };
    
    // 抢红包按钮点击事件处理函数
    void CMutexExDlg::OnBnClickedButton1()
    {
        // 获取编辑框内容到str变量
        CString str;
        GetDlgItem(IDC_EDIT1)->GetWindowText(str);
        // CString 转 int
        m_Edit0 = _ttoi(str);
        m_Edit1 = 0;
        m_Edit2 = 0;
        m_Edit3 = 0;
        // 这里需要创建一个线程 因为 WaitForMultipleObjects 会阻塞住 另外把this指针作为参数传递给线程,用于子线程更新编辑框内容
        HANDLE hThread0 = ::CreateThread(NULL, NULL, ThreadProc0, this, NULL, NULL);
        CloseHandle(hThread0);
    }
    
    DWORD WINAPI CMutexExDlg::ThreadProc0(LPVOID lpParameter)
    {
        // bInitialOwner BOOL,如创建进程希望立即拥有互斥体,则设为TRUE。一个互斥体同时只能由一个线程拥有
        // 创建一个互斥体
        m_Mutex = ::CreateMutex(NULL, FALSE, L"test");
        // 创建三个线程抢红包
        hThread[0] = ::CreateThread(NULL, NULL, ThreadProc1, lpParameter, NULL, NULL);
        hThread[1] = ::CreateThread(NULL, NULL, ThreadProc2, lpParameter, NULL, NULL);
        hThread[2] = ::CreateThread(NULL, NULL, ThreadProc3, lpParameter, NULL, NULL);
        // 使用WaitForMultipleObjects监听所有线程,当线程全部结束后,调用CloseHandle关闭句柄.
        WaitForMultipleObjects(3, hThread, TRUE, -1);
        ::CloseHandle(hThread[0]);
        ::CloseHandle(hThread[1]);
        ::CloseHandle(hThread[2]);
        ::CloseHandle(m_Mutex);
        return 0;
    }
    
    // 线程回调函数1
    DWORD WINAPI CMutexExDlg::ThreadProc1(LPVOID lpParameter)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        while (true)
        {
            WaitForSingleObject(m_Mutex, -1);
            if (m_Edit0 < 50)
            {
                break;
            }
            m_Edit0 = m_Edit0 - 50;
            m_Edit1 = m_Edit1 + 50;
            // int 转 CString
            CString str0;
            str0.Format(_T("%d"), m_Edit0);
            // 设置编辑框内容
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT1), str0);
            CString str1;
            str1.Format(_T("%d"), m_Edit1);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT2), str1);
            Sleep(50);
            // 释放线程拥有的互斥体的控制权
            ReleaseMutex(m_Mutex);
        }
        return 0;
    }
    
    // 线程回调函数2
    DWORD WINAPI CMutexExDlg::ThreadProc2(LPVOID lpParameter)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        while (true)
        {
            WaitForSingleObject(m_Mutex, -1);
            if (m_Edit0 < 50)
            {
                break;
            }
            m_Edit0 = m_Edit0 - 50;
            m_Edit2 = m_Edit2 + 50;
            CString str0;
            str0.Format(_T("%d"), m_Edit0);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT1), str0);
            CString str1;
            str1.Format(_T("%d"), m_Edit2);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT3), str1);
            Sleep(50);
            // 释放线程拥有的互斥体的控制权
            ReleaseMutex(m_Mutex);
        }
        return 0;
    }
    
    // 线程回调函数3
    DWORD WINAPI CMutexExDlg::ThreadProc3(LPVOID lpParameter)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        while (true)
        {
            WaitForSingleObject(m_Mutex, -1);
            if (m_Edit0 < 50)
            {
                break;
            }
            m_Edit0 = m_Edit0 - 50;
            m_Edit3 = m_Edit3 + 50;
            CString str0;
            str0.Format(_T("%d"), m_Edit0);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT1), str0);
            CString str1;
            str1.Format(_T("%d"), m_Edit3);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT4), str1);
            Sleep(50);
            // 释放线程拥有的互斥体的控制权
            ReleaseMutex(m_Mutex);
        }
        return 0;
    }
  • 相关阅读:
    PHP发送邮件
    SQL删除字段及判断字段是否存在的方法
    密码MySQL的root的密码
    java socket 最简单的例子(server 多线程)
    php编写最简单的webservice
    SQL Server 存储过程与触发器
    手动创建最简单的JSP 文件
    Oracle 卸载步骤
    编写 WebService 程序
    eclipse 常用快捷键
  • 原文地址:https://www.cnblogs.com/duxie/p/11117051.html
Copyright © 2011-2022 走看看