国产午夜成人免费看片无遮挡_日本免费xxxx色视频_免费人成网上在线观看_黄网址在线永久免费观看

當前位置:雨林木風下載站 > 技術開發教程 > 詳細頁面

基于VC的串行通信技術應用案例

基于VC的串行通信技術應用案例

更新時間:2022-04-26 文章作者:未知 信息來源:網絡 閱讀次數:

作者: 李湘江

在工業控制中,串口是常用的計算機與外部串行設備之間的數據傳輸通道,由于串行通信方便易行,所以應用廣泛。

本文將介紹在Windows平臺下串行通信的工作機制和用Visual C++設計串行通信程序的編程方法及通信方式。

VC中實現串行通信的編程技術
以下我們將介紹VC中幾種實現串行通信的編程技術:

利用VC++的標準通信函數

利用VC++的標準通信函數_inp和_outp可實現串口通信。下面是一個串口初始化的程序:

void init_com(PORT)
{char i;
outp(PORT+3,0x80);
outp(PORT,0x0C);
outp(PORT+1,0);
outp(PORT+3 ,0x3a);
outp(PORT+3 ,0x03);
i=inp(PORT+5) && 0xfe;
outp(PORT+5,i);}

使用串行通信控件MSComm

串行通信控件MSCOmm32.OCX提供了使用RS-232來進行數據通信的所有協議,VC為該控件提供了標準的事件處理函數、過程,并通過屬性和方法提供了串行通信的設置。它使用戶能夠方便地訪問Windows串行通信驅動程序的大多數特性,包括輸入、輸出緩沖區的大小及決定何時使用流控制命令掛起數據傳輸等。

在ClassWizard中為新創建的通信控件定義成員對象(CMSComm m_Serial),通過該對象便可以對串口屬性進行設置,MSComm控件共有27個屬性。以下是通過設置控件屬性對串口進行初始化的實例:

BOOL CSampleDlg:: PortOpen()
{ BOOL m_Opened;
......
m_Serial.SetCommPort(2); // 指定串口號
m_Serial.SetSettings("4800,N,8,1");
// 通信參數設置
m_Serial.SetInBufferSize(1024);
// 指定接收緩沖區大小
m_Serial.SetInBufferCount(0);
// 清空接收緩沖區
m_Serial.InputMode(1);
// 設置數據獲取方式
m_Serial.SetInputLen(0);
// 設置讀取方式
m_Opened=m_Serail.SetPortOpen(1);
// 打開指定的串口
return m_Opened;}

打開所需串口后,我們需要考慮串口通信的時機。在接收或發送數據過程中,可能需要監視并響應一些事件和錯誤,所以事件驅動是處理串行端口交互作用的一種非常有效的方法。使用OnComm事件和CommEvent屬性捕捉并檢查通信事件和錯誤的值。發生通信事件或錯誤時將觸發OnComm事件,CommEvent屬性的值將被改變,應用程序通過檢查CommEvent屬性值并作出相應的反應。

使用API函數

控件雖然簡單易用,但由于必須拿到對話框中使用,在一些需要在線程中實現通信的應用場合下,控件的使用顯得捉襟見肘。API是附帶在Windows內部的一個極其重要的組成部分。Windows的32位API主要是一系列很復雜的函數和消息集合。它可以看作是Windows系統為在其下運行的各種開發系統提供的開放式通用功能增強接口。

通信程序在CreateFile處指定串口設備及相關的操作屬性,再返回一個句柄,該句柄將被用于后續的通信操作,并貫穿整個通信過程。串口打開后,其屬性被設置為默認值,根據具體需要,通過調用GetCommState(hComm,&&dcb)讀取當前串口設備控制塊DCB設置,修改后通過SetCommState(hComm,&&dcb)將其寫入。運用ReadFile()與WriteFile()這兩個API函數實現串口讀寫操作,若為異步通信方式,兩函數中最后一個參數為指向OVERLAPPED結構的非空指針,在讀寫函數返回值為FALSE的情況下,調用GetLastError()函數,返回值為ERROR_IO_PENDING,表明I/O操作懸掛,即操作轉入后臺繼續執行。此時,可以用WaitForSingleObject()來等待結束信號并設置最長等待時間,舉例如下:

BOOL bReadStatus;
bReadStatus = ReadFile( m_hIDComDev, buffer,
dwBytesRead, &&dwBytesRead, &&m_OverlappedRead );
if(!bReadStatus){
if(GetLastError()==ERROR_IO_PENDING){
WaitForSingleObject(m_OverlappedRead.hEvent,1000);
return ((int)dwBytesRead);}
return(0);}
return ((int)dwBytesRead);

多線程下實現串行通信
Windows內部的搶先調度程序在活動的線程之間分配CPU時間,Windows區分兩種不同類型的線程,一種是用戶界面線程(User Interface Thread),它包含消息循環或消息泵,用于處理接收到的消息;另一種是工作線程(Work Thread),它沒有消息循環,用于執行后臺任務、監視串口事件的線程即為工作線程。

多線程程序的編寫在端口的配置,連接部分與單線程的相同,在端口配置完畢后,最重要的是根據實際情況,建立多線程之間的同步對象,如信號燈、臨界區和事件等。

一切就緒后即可啟動工作線程,程序如下:

CWinThrea CommThread = AfxBegin
Thread(CommWatchThread, // 線程函數名
(LPVOID) m_pTTYInfo, // 傳遞的參數
THREAD_PRIORITY_ABOVE_NORMAL,
// 設置線程優先級
(UINT) 0, // 最大堆棧大小
(DWORD) CREATE_SUSPENDED , // 創建標志
(LPSECURITY_ATTRIBUTES) NULL);
if(WaitCommEvent(pTTYInfo->idComDev,&&dwEvtMask,NULL))
{
if((dwEvtMask && pTTYInfo->dwEvtMask )== pTTYInfo->dwEvtMask)
{
WaitForSingleObject(pTTYInfo->hPostEvent,0xFFFFFFFF);
ResetEvent(pTTYInfo->hPostEvent);
// 置同步事件對象為非信號態
::PostMessage(CSampleView,ID_COM1_DATA,0,0); // 發送通知消息}}
BEGIN_MESSAGE_MAP(CSampleView, CView)
//{{AFX_MSG_MAP(CSampleView)
ON_MESSAGE(ID_COM1_DATA, OnProcessCom1Data)
ON_MESSAGE(ID_COM2_DATA, OnProcessCom2Data)
.....
//}}AFX_MSG_MAP
END_MESSAGE_MAP()

多線程的實現可以使得各端口獨立,準確地實現串行通信,使串行通信具有更廣泛的靈活性與嚴格性,且充分利用CPU時間。但在具體的實時監控系統中如何協調多個線程、線程之間以何種方式實現同步,這是多線程串行通信程序實現的難點。

串行通信的操作方式
下面我們將介紹串行通信的幾種操作方式:

1.同步方式

同步方式中,讀串口的函數試圖在串口的接收緩沖區中讀取規定數目的數據,直到規定數目的數據全部被讀出或設定的超時時間已到時才返回。例如:

COMMTIMEOUTS timeOver;
memset(&&timeOver,0,sizeof(timeOver));
DWORD timeMultiplier,timeConstant;
timeOver.ReadTotalTimeoutMultiplier=timeMultiplier;
timeOver.ReadTotalTimeoutConstant=timeConstant;
SetCommTimeouts(hComport,&&timeOver);
……
ReadFile(hComport,inBuffer,nWantRead,&&nRealRead,NULL);

COMMTIMEOUTS結構用于設置讀寫函數的等待時間。

在ReadFile函數中hComport為待讀串口句柄;inBuffer為輸入緩沖區大小;nWantRead為每次調用ReadFile時,函數試圖讀出的字節數;nRealRead為實際讀出的字節數;最后一個參數值NULL代表ReadFile將采用同步文件讀寫的方式。

如果所規定的待讀取數據的數目nWantRead較大且設定的超時時間也較長,而接收緩沖區中數據較少,則可能引起線程阻塞。解決這一問題的方法是檢查COMSTAT結構的cbInQue成員,該成員的大小即為接收緩沖區中處于等待狀態的數據的實際個數。如果令nWantRead的值等于COMSTAT.cbInQue,就能較好地防止線程阻塞。

2.查詢方式

查詢方式,即一個進程中的某一線程定時地查詢串口的接收緩沖區,如果緩沖區中有數據,就讀取數據;若緩沖區中沒有數據,該線程將繼續執行,因此會占用大量的CPU時間,它實際上是同步方式的一種派生。例如:

COMMTIMEOUTS timeOver;
memset(&&timeOver,0,sizeof(timeOver));
timeOver.ReadIntervalTimeout=MAXWORD;
SetCommTimeouts(hComport.&&timeOver);
……
ReadFile(hComport.inBuffer.nWantRead.&&nRealRead,NULL);

除了COMMTIMEOUTS結構的變量timeOver設置不同外,查詢方式與同步方式在程序代碼方面很類似,但二者的工作方式卻差別很大。盡管ReadFile采用的也是同步文件讀寫方式,但由于timeOver的區間超過時間設置為MAXWORD,所以ReadFile每次將讀出接收隊列中的所有處于等待狀態的數據,一次最多可讀出nWantRead個字節的數據。

3.異步方式

異步方式中,利用Windows的多線程結構,可以讓串口的讀寫操作在后臺進行,而應用程序的其他部分在前臺執行。例如:

OVERLAPPED wrOverlapped;
COMMTIMEOUTS timeOver;
memset(&&timeOver.0.sizeof(timeOver));
DWORDtimeMultiplier,timeConstant;
timeOver.ReadTotalTimeoutMultiplier=timeMultiplier;
timeOver.ReadTotalTimeoutConstant=timeConstant;
SetCommTimeouts(hComport,&&timeOver);
wrOverlapped.hEvent=CreateEvent(NULL.TRUE,FALSE,NULL);
……
ReadFile(hComport,inBuffer,nWantRead,&&nRealRead,&&wrOverlapped);
GetOverlappedResult(hComport,&&wrOverlapped,&& nRealRead,TRUE);
……
ResetEvent(wrOverlapped.hEvent);

上面代碼中的ReadFile由于采用了異步方式,所以它只返回數據是否已開始讀入的狀態,并不返回實際的讀入數據,即ReadFile中的nRealRead無效。實際讀入的數據是由GetOverlappedResult函數返回的,該函數的最后一個參數值為TRUE,表示它等待異步操作結束后才返回到應用程序,此時,GetOverlappedResult函數與WaitForSingleObject函數等效。

當采用異步方式時,在用CreateFile打開串口設備時,CreateFile函數的參數fdwAttrsAndFlags必須設為FILE_FLAG_ OVERLAPPED。在Windows中,只有在串行設備上才支持異步文件讀寫,并且,GetOverlappedResult函數也只支持串行設備或用DeviceloControl函數打開的文件。

4.事件驅動方式

若對端口數據的響應時間要求較嚴格,可采用事件驅動方式。事件驅動方式通過設置事件通知,當所希望的事件發生時,Windows發出該事件已發生的通知,這與DOS環境下的中斷方式很相似。Windows定義了9種串口通信事件,較常用的有以下三種:

EV_RXCHAR:接收到一個字節,并放入輸入緩沖區;

EV_TXEMPTY:輸出緩沖區中的最后一個字符,發送出去;

EV_RXFLAG:接收到事件字符(DCB結構中EvtChar成員),放入輸入緩沖區。

在用SetCommMask()指定了有用的事件后,應用程序可調用WaitCommEvent()來等待事件的發生。SetCommMask(hComm,0)可使WaitCommEvent()中止。例如:

COMSTAT comStat;
DWORD dwEvent;
SetCommMask(hComport,EV_RXCHAR);
……
if(WaitCommEvent(hComport,&&dwEvent,NULL))
 if((dwEvent&&EV_RXCHAR)&&&&comstat.cbInQue)
   ReadFile(hComport,inBuffer,comstat.cbInQue,&&nRealRead,NULL);

程序中,我們首先用SetCommMask函數設置事件代碼,上面的代碼中為EV_RXCHAR,表示接收到一個字符時觸發這一事件,然后調用WaitCommEvent函數等待該事件的發生。注意,WaitCommEvent函數第3個參數1pOverlapped可以是一個OVERLAPPED結構的變量指針,也可以是NULL,當用NULL時,表示該函數是同步的,否則表示該函數是異步的。

5.幾種方式的比較

在一般要求情況下,查詢方式是一種最直接的讀串口方式。但定時查詢存在一個致命弱點,即查詢是定時發生的,可能發生得過早或過晚。在數據變化較快的情況下,特別是主控計算機的串口通過擴展板擴展至多個時,需定時地對所有串口輪流查詢,此時容易發生數據的丟失。雖然定時間隔越小,數據的實時性越高,但系統的資源也被占去越多。

Windows中提出文件讀寫的異步方式,主要是針對文件I/O相對較慢的速度而進行的改進,它利用了Windows的多線程結構。雖然在Windows中沒有實現任何對文件I/O的異步操作,但它卻能對串口進行異步操作。采用異步方式,可以提高系統的整體性能,在對系統強壯性要求較高的場合,建議采用這種方式。

事件驅動方式是一種高效的串口讀方式。這種方式的實時性較高,特別是對于擴展了多個串口的情況,并不要求像查詢方式那樣定時地對所有串口輪流查詢,而是像中斷方式那樣,只有當設定的事件發生時,應用程序得到Windows操作系統發出的消息后,才進行相應處理,避免了數據丟失。在實時性要求較高的場合,筆者建議采用這種方式。

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統下載排行

国产午夜成人免费看片无遮挡_日本免费xxxx色视频_免费人成网上在线观看_黄网址在线永久免费观看

  • <label id="pxtpz"><meter id="pxtpz"></meter></label>
      1. <span id="pxtpz"><optgroup id="pxtpz"></optgroup></span>

        欧美一区二区三区男人的天堂| 亚洲欧洲国产专区| 欧美xxxxxxxxx| 亚洲国产精品一区二区久久恐怖片 | 欧美日本一道本| 国产精品网站在线播放| 国产成人福利片| 中文字幕在线不卡一区二区三区| 裸体一区二区三区| 精品日韩欧美在线| 成人深夜福利app| 中文字幕一区二区三区在线观看| 26uuu精品一区二区三区四区在线 26uuu精品一区二区在线观看 | 丝袜美腿亚洲一区| 成人精品电影在线观看| 久久精品视频在线看| 成人免费黄色大片| 亚洲综合色婷婷| 91精品国产91久久久久久一区二区 | 亚洲综合视频网| 色偷偷久久一区二区三区| 亚洲国产一区视频| 久久众筹精品私拍模特| 99热精品国产| 奇米777欧美一区二区| 国产亲近乱来精品视频| 欧美最新大片在线看| 韩国在线一区二区| 夜夜嗨av一区二区三区| 久久先锋资源网| 欧美精三区欧美精三区| 粉嫩av一区二区三区在线播放| 亚洲国产视频在线| 国产精品天美传媒| 精品久久久久久久久久久久久久久 | 亚洲精品久久久蜜桃| 日韩一区二区精品葵司在线| caoporn国产精品| 精品一区二区三区免费毛片爱| 国产欧美日韩在线| 777亚洲妇女| 欧美伊人精品成人久久综合97| 国产传媒日韩欧美成人| 蜜乳av一区二区| 亚洲二区在线观看| 亚洲免费在线观看| 国产精品免费视频网站| 久久只精品国产| 日韩欧美的一区| 777xxx欧美| 欧美日韩国产在线观看| 色综合天天综合狠狠| 日韩黄色免费电影| 日韩一区在线看| 亚洲国产精品精华液ab| 精品国产伦一区二区三区观看体验| 欧美丝袜丝nylons| 99久久久精品| 大陆成人av片| 国产不卡视频在线观看| 国产69精品一区二区亚洲孕妇| 极品少妇xxxx精品少妇偷拍| 蜜桃久久久久久| 精品在线一区二区三区| 美国十次了思思久久精品导航| 日韩精品一二三四| 免费在线看成人av| 日韩一区精品视频| 日日夜夜精品视频免费| 99r精品视频| 色综合久久久久网| 欧美日韩激情在线| 777午夜精品免费视频| 日韩视频一区二区| 久久精子c满五个校花| 欧美韩国日本不卡| 国产精品久久久久一区二区三区| 国产精品高潮呻吟久久| 一区二区三区中文在线观看| 亚洲一区二区三区四区在线观看 | 国产精品视频麻豆| 高潮精品一区videoshd| 国产又粗又猛又爽又黄91精品| 日韩精品免费视频人成| 日韩高清一级片| 秋霞电影一区二区| 国内外精品视频| 韩国三级中文字幕hd久久精品| 精一区二区三区| 懂色av一区二区三区蜜臀| 成人一级片在线观看| 91蜜桃网址入口| 欧美中文一区二区三区| 福利视频网站一区二区三区| 久久99精品久久久| 福利一区福利二区| 欧美四级电影网| 日韩免费电影一区| 亚洲视频你懂的| 日韩avvvv在线播放| 国产精一区二区三区| 91影院在线免费观看| 91精品在线观看入口| 国产精品久久三区| 蜜臀av亚洲一区中文字幕| 成人app在线| 这里是久久伊人| 国产精品久久久久久福利一牛影视| 亚洲va欧美va天堂v国产综合| 国产精品一卡二| 欧美日韩色一区| 国产欧美日韩中文久久| 日韩精品高清不卡| 91亚洲男人天堂| 欧美日韩国产首页在线观看| 亚洲欧美日韩系列| 午夜伦理一区二区| 国产精品99久久不卡二区| 在线视频一区二区免费| 欧美韩国日本不卡| 激情偷乱视频一区二区三区| 在线亚洲+欧美+日本专区| 久久这里只有精品视频网| 日韩电影在线看| 色综合欧美在线视频区| 国产亚洲污的网站| 精品一区二区三区免费观看 | 一区二区不卡在线视频 午夜欧美不卡在 | 一本一本久久a久久精品综合麻豆 一本一道波多野结衣一区二区 | 色诱视频网站一区| 久久日韩精品一区二区五区| 日韩高清不卡在线| 欧美日韩视频在线观看一区二区三区 | 色菇凉天天综合网| 国产精品对白交换视频| 国产传媒日韩欧美成人| 久久婷婷国产综合精品青草| 日韩精品亚洲一区| 欧美一区二区三区免费视频| 午夜精品久久久久影视| 欧美蜜桃一区二区三区| 天天综合色天天综合| 欧美人伦禁忌dvd放荡欲情| 天天综合网天天综合色| 日韩一区二区在线观看| 精品一区二区三区久久| 亚洲精品一线二线三线无人区| 床上的激情91.| 久久精品视频网| 国产精品亚洲成人| 亚洲婷婷综合色高清在线| 91麻豆高清视频| 亚洲一区二区三区四区的| 在线成人免费观看| 久久国产精品一区二区| 国产欧美视频一区二区三区| 91免费看视频| 肉色丝袜一区二区| 久久久久久久av麻豆果冻| 99视频在线观看一区三区| 一区二区三区国产豹纹内裤在线| 欧美美女黄视频| 国内精品嫩模私拍在线| 亚洲色图欧洲色图婷婷| 欧美精品久久99久久在免费线 | 国产精品一区二区三区网站| 中文字幕乱码日本亚洲一区二区| 91在线视频免费观看| 亚洲国产成人高清精品| 欧美v日韩v国产v| 99久久精品国产麻豆演员表| 亚洲午夜私人影院| 国产欧美精品区一区二区三区| 亚洲精品欧美激情| 欧美日韩国产美女| 精品一二线国产| 国产精品国产馆在线真实露脸| 欧美在线一区二区| 蜜桃av一区二区在线观看| 国产精品三级久久久久三级| 欧美亚洲图片小说| 九色综合狠狠综合久久| 中文字幕一区二区三区在线播放 | 日韩美女一区二区三区| 国产成+人+日韩+欧美+亚洲| 亚洲国产精品人人做人人爽| 欧美国产日韩a欧美在线观看| 欧美日韩中文字幕精品| av色综合久久天堂av综合| 久久99热这里只有精品| 一区二区三区久久| 国产精品日日摸夜夜摸av| 日韩精品一区二区三区蜜臀 | 欧美日韩精品免费| av中文字幕不卡| 国产精品一区三区| 狠狠色综合播放一区二区| 午夜精品爽啪视频| 亚洲电影中文字幕在线观看| 亚洲视频一二三区|