新建基于对话框的MFC工程,OpenDevice这个类由DLG类调用。(我的平台是:Win7-x64 / VisualStudio 2012 Ultimate)
OpenDevice.h
<span style="font-size:18px;">#pragma once
class OpenDevice
{
public:
OpenDevice(void);
~OpenDevice(void);
public:
CWinThread *m_pThread;
HANDLE m_hCom;
HANDLE m_hSemaphore;
BOOL SemIncrease(void);
BOOL Init(void);
BOOL OpenSerial(void);
static UINT work_thread(void *args);
};
</span>
OpenDevice.c
<span style="font-size:18px;">#include "stdafx.h"
#include "OpenDevice.h"
OpenDevice::OpenDevice(void)
{
m_hCom = NULL;
m_hSemaphore = NULL;
m_pThread = NULL;
}
OpenDevice::~OpenDevice(void)
{
if(NULL != m_hCom)
{
CloseHandle(m_hCom);
m_hCom = NULL;
}
if(NULL != m_hSemaphore)
{
CloseHandle(m_hSemaphore);
m_hSemaphore = NULL;
}
}
//初始化信号量,创建子线程(创建的时候立即运行)
BOOL OpenDevice::Init(void)
{
//创建并初始化信号量
m_hSemaphore = CreateSemaphore(NULL, 0, 1, NULL);
if(NULL == m_hSemaphore)
{
TRACE(strerror(GetLastError()));
return FALSE;
}
//创建子线程
m_pThread = AfxBeginThread(work_thread, this);
if(NULL == m_pThread)
{
TRACE(strerror(GetLastError()));
return FALSE;
}
return TRUE;
}
BOOL OpenDevice::OpenSerial(void)
{
if(NULL == m_hSemaphore)
{
return FALSE;
}
//等待串口设备插入:当用户点击打开按钮时,增加信号量的值,则WaitForSingleObject返回
//并且减少一个值;然后去打开设备文件,失败后则重新阻塞等待信号量有效。
while(1)
{
TRACE("wait for semaphore...\n");
DWORD res = WaitForSingleObject(m_hSemaphore, INFINITE); //wait 相当于 信号量的P操作
if(WAIT_OBJECT_0 != res)
{
TRACE(strerror(GetLastError()));
continue;
}
//打开串口文件
m_hCom = CreateFile( "COM3",
GENERIC_WRITE | GENERIC_READ,
0,
NULL,
OPEN_EXISTING,
FILE_FLAG_OVERLAPPED,
0);
if(INVALID_HANDLE_VALUE == m_hCom)
{
TRACE(strerror(GetLastError()));
AfxMessageBox("串口打开失败!");
}
else
{
TRACE("open successful\n");
return TRUE;
}
}
}
UINT OpenDevice::work_thread(void *args)
{
OpenDevice *pOpenDevice = (OpenDevice *)args;
BOOL res = pOpenDevice->OpenSerial();
if(TRUE == res)
{
while(1)
{
TRACE("read and write serial!\n");
Sleep(5000);
}
}
return 0;
}
//信号量的V操作
BOOL OpenDevice::SemIncrease(void)
{
if(NULL == m_hSemaphore)
{
return FALSE;
}
BOOL res = ReleaseSemaphore(m_hSemaphore, 1, NULL);
if(0 == res)
{
TRACE(strerror(GetLastError()));
return FALSE;
}
return TRUE;
}</span>
按钮响应函数:
void CsemaphoreDlg::OnBnClickedOpen()
{
// TODO: 在此添加控件通知处理程序代码
m_openDevice.SemIncrease();
}
MFC中信号量的操作和linux中的信号量操作基本相同,只是linux中单独有函数做PV操作。