zoukankan      html  css  js  c++  java
  • C++线程同步之事件

    题目要求:点击抢红包后,先将第一个编辑框的值设置为1000,然后创建三个线程,让右边的编辑框值依次设置为1000(用事件完成)

    // 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 HANDLE m_Event;
        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;
    HANDLE CMutexExDlg::m_Event = 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)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        // 创建一个事件
        m_Event = ::CreateEvent(NULL,
            FALSE, // FALSE 代表 WaitForSingleObject到之后,事件还是未通知状态,需要手动设置已通知状态
            FALSE,  // FALSE 代表 事件创建完之后,不能马上被 WaitForSingleObject 到
            NULL);
        // 创建三个线程抢红包
        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);
        // 设置编辑框内容
        ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT1), L"1000");
        // 将对象设置为已通知状态
        ::SetEvent(m_Event);
        // 使用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_Event, -1);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT2), L"1000");
            Sleep(2000);
            // 将对象设置为已通知状态
            ::SetEvent(m_Event);
        }
        return 0;
    }
    
    // 线程回调函数2
    DWORD WINAPI CMutexExDlg::ThreadProc2(LPVOID lpParameter)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        while (true)
        {
            WaitForSingleObject(m_Event, -1);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT3), L"1000");
            Sleep(2000);
            // 将对象设置为已通知状态
            ::SetEvent(m_Event);
        }
        return 0;
    }
    
    // 线程回调函数3
    DWORD WINAPI CMutexExDlg::ThreadProc3(LPVOID lpParameter)
    {
        CMutexExDlg* pDlg = (CMutexExDlg*)lpParameter;
        while (true)
        {
            WaitForSingleObject(m_Event, -1);
            ::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_EDIT4), L"1000");
            Sleep(2000);
            // 将对象设置为已通知状态
            ::SetEvent(m_Event);
        }
        return 0;
    }
  • 相关阅读:
    云题库错题分析
    数据库相关子查询
    阻止事件冒泡
    21分钟 MySQL 入门教程(转载!!!)
    java接口
    java访问修饰符
    小游戏,快速击键
    个人对Java中多态的一些简单理解
    简述抽象和封装,对你学习Java有一些作用
    Bank,我只是来完成作业的
  • 原文地址:https://www.cnblogs.com/duxie/p/11123573.html
Copyright © 2011-2022 走看看