0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于多線程技術(shù)VisualC++實現(xiàn)工控系統(tǒng)的設(shè)計

電子設(shè)計 ? 來源:電子技術(shù)應(yīng)用 ? 作者:齊文,王勇前 ? 2020-10-15 17:51 ? 次閱讀

多線程技術(shù)的引入,不僅可以挖掘潛在的CPU空閑時間,而且還可以提高應(yīng)用程序的反應(yīng)速度,其優(yōu)點在有多個任務(wù)需要完成、有巨大數(shù)據(jù)流量的程序中反映得尤為明顯。而隨著VisualC++的引入,其靈活的線程實現(xiàn)機(jī)制使得程序員從繁瑣的Windows編程中解脫出來。關(guān)于多線程基本機(jī)理和實現(xiàn)方法近年來有許多文章介紹,這里不再贅述。本文將側(cè)重于比較在工控程序中采用各種線程類型和同步方法的優(yōu)劣,并給出一個實用的、有較廣適應(yīng)性的程序主體框架。

1 各種線程類型和同步方法

1.1 線程類型

Visual C++中線程分為工作者線程(Worker Thread)和用戶界面線程(User Interface Thread)兩大類。

用戶界面線程的特點是擁有單獨的消息隊列,可以具有自己的窗口界面,能夠?qū)κ录陀脩糨斎胱龀鲰憫?yīng),具體實現(xiàn)時由CWinThread派生出一個類。但其缺點是當(dāng)需要停止或撤銷當(dāng)前正在運行的線程而向其發(fā)送中止消息后,只有在消息隊列中排在前面的消息被一一處理完之后,線程才能接受中止消息并停止當(dāng)前工作,這對CPU是一種浪費,在對實時性要求較高的工控程序中是不可容忍的。

工作者線程適用于處理后臺任務(wù),而不影響用戶對應(yīng)用程序的使用。工作者線程僅僅由一個函數(shù)體實現(xiàn),其實現(xiàn)簡單,便于編程者控制,與事件同步方法相配合能對中止消息做出較快反應(yīng)。

1.2 同步方法

在多線程應(yīng)用程序中,兩個或更多的線程同時訪問相同數(shù)據(jù)會導(dǎo)致不可預(yù)知的結(jié)果,因此保持線程間的同步是一個不可或缺的環(huán)節(jié)。VisualC++提供了四種同步方法:臨界區(qū)(Critical Section)、信號燈(Semaphore)、互斥量(Mutex)和事件(Event)。

其中采用臨界區(qū)、信號燈或互斥量進(jìn)行同步時,線程間的同步過程由操作系統(tǒng)完全控制,系統(tǒng)僅僅防止多個線程對同一資源的同時使用,而相同優(yōu)先級的線程對同一資源的使用順序是編程者無法控制的。而在一般工控系統(tǒng)中,當(dāng)主控臺下方設(shè)備數(shù)據(jù)變化時,應(yīng)能及時中止當(dāng)前的計算(如果當(dāng)前計算未完成的話)并根據(jù)新的數(shù)據(jù)開始新一輪的計算,因而要求各線程對所處理的數(shù)據(jù)有一定的操作次序。

事件同步是通過將事件自身設(shè)置為有信號或無信號來通知其它線程某一操作已完成或尚未完成,其設(shè)置可由編程人員手工完成,適合于工控程序應(yīng)用。盡管事件同步方式平均效率比上面三種方式稍低,但在工控程序應(yīng)用中相對于因數(shù)據(jù)未能及時更新而導(dǎo)致大量的無用計算及其對實時性的損害來講,還是非常值得的。

下面介紹的是參與某‘九五’預(yù)研項目中所設(shè)計的主控臺程序的基本框架,這個程序框架應(yīng)能適用于大多數(shù)工控系統(tǒng)的主控程序。

2 軟件框架

一般工控系統(tǒng)的主控部分通常所必須完成的兩件事是

1)通過通信端口與下端設(shè)備通信,接收下端設(shè)備傳來的數(shù)據(jù)或向下端設(shè)備發(fā)送指令;(2)對下端設(shè)備所傳數(shù)據(jù)進(jìn)行處理。

與之相對應(yīng),該軟件具有一個主線程和兩個子線程,其中一個子線程為通信線程,另一個為計算線程。主線程是Windows下每個應(yīng)用程序都具備的,負(fù)責(zé)線程間的同步、向計算線程和通信線程傳遞參數(shù)、管理人機(jī)界面、接收用戶輸入、數(shù)據(jù)庫的操作和管理等功能。通信線程通過通信端口(可以是串口、并口或網(wǎng)絡(luò)接口等)負(fù)責(zé)與下端的設(shè)備進(jìn)行通信并交換數(shù)據(jù),當(dāng)存在多級控制結(jié)構(gòu)時,還可用來與更高一級的控制設(shè)備進(jìn)行通信并向上傳遞數(shù)據(jù)。計算線程負(fù)責(zé)核心算法的實現(xiàn),根據(jù)系統(tǒng)的不同完成不同的數(shù)據(jù)處理任務(wù)。程序結(jié)構(gòu)如圖1。

基于多線程技術(shù)VisualC++實現(xiàn)工控系統(tǒng)的設(shè)計

進(jìn)程開始后先由主線程建立通信線程與計算線程。通信線程監(jiān)視通信端口,當(dāng)下方設(shè)備發(fā)來數(shù)據(jù)時,就向主線程發(fā)送自定義的WM_USER_COMM_NOTIFY消息,通知主線程計算數(shù)據(jù)有所改變,主線程則對之進(jìn)行處理,即中止當(dāng)前的計算,并重新開始計算。

3 具體實現(xiàn)

用Visual C++的AppWizard生成一個應(yīng)用程序,這是主控程序的雛形,該應(yīng)用程序暫取名為CtrSys,后面程序名都以此為準(zhǔn)。

3.1 多線程的定義及生成

3.1.1 多線程的定義

向項目中加入threads.cpp文件,在該文件中寫入通信線程和計算線程的控制函數(shù)。

控制函數(shù)有下面的原型:

UINT MyThreadProc(LPVOID lpvThreadParam);

lpvThreadParam參數(shù)是32位的值,這個值就是在線程對象產(chǎn)生時傳遞給線程構(gòu)造函數(shù)的參數(shù)。控制函數(shù)能解釋此值的不同表現(xiàn)方式。它可以被當(dāng)作一個普通變量對待,也可以被視為一個指向包含有多個參數(shù)的結(jié)構(gòu)指針,也可以被忽略。如果參數(shù)指向一個結(jié)構(gòu),這個結(jié)構(gòu)可能不僅僅用來從調(diào)用者傳遞參數(shù)給線程,還可能用來從線程回傳數(shù)據(jù)給調(diào)用者。如果使用這樣的結(jié)構(gòu)回傳給調(diào)用者,結(jié)果準(zhǔn)備好后線程需要通知調(diào)用者。

控制函數(shù)終止時,應(yīng)該返回一個UINT類型的值,表明終止的原因。返回碼0表示成功,其它值表示不同類型的錯誤,這完全依賴實現(xiàn)情況。

按一般程序示例,線程通常在視類或框架窗口類中產(chǎn)生。但在工控程序中,通信與計算線程常常要大量地對計算數(shù)據(jù)進(jìn)行操作,根據(jù)文檔/視的程序框架結(jié)構(gòu),文檔類常常用來存儲所要處理的數(shù)據(jù)。因此把計算與通信線程放在文檔類中產(chǎn)生,并把產(chǎn)生線程的當(dāng)前文檔對象的指針作為線程控制函數(shù)的參數(shù)傳遞給線程。

從而,在控制函數(shù)(CalcThreadProc ()和CommThreadProc())一開始,就要對所傳來的參數(shù)進(jìn)行識別:

CCtrsysDoc* pDoc = (CCtrsysDoc*)pParam;

注意要在文件開頭包括進(jìn)文檔類的頭文件

#include ″CtrsysDoc.h″

3.1.2 多線程的產(chǎn)生

在文檔類的構(gòu)造函數(shù)中產(chǎn)生線程。程序啟動時生成文檔對象,同時啟動兩個線程。

////////////////////////////////////

// CCtrsysDoc construction/destruction

CCtrsysDoc::CCtrsysDoc()

{

……

m_pCalcThread=AfxBeginThread(CalcThreadProc, this);

m_pCommThread=AfxBeginThread(CommThreadProc, this);

}

注意不要用Win32的CreateThread()建立線程,而應(yīng)該用AfxBeginThread()函數(shù),否則所建立的線程不能訪問其它MFC對象。

3.2 線程間的同步

程序中設(shè)置有八個事件用于線程同步:

HANDLE m_hEventPost; //用來允許通信線程向主框架

發(fā)送WM_USER_COMM_NOTIFY消息

HANDLE m_hEventStartCalc; //主框架通知計算線程開始計算

HANDLE m_hEventCalcStarted; //計算線程通知主框架計算已經(jīng)開始

HANDLE m_hEventStopCalc; //主框架通知計算線程中止計算

HANDLE m_hEventCalcStopped; //計算線程通知主框架計算已經(jīng)中止

HANDLE m_hEventCalcDone; //計算線程通知主框架計算已經(jīng)結(jié)束

HANDLE m_hEventUpdateSourceData; //主框架通知計算線程更新數(shù)據(jù)

HANDLE m_hEventSourceDataUpdated; //通信線程通知主框架數(shù)據(jù)已更新完畢

這八個事件是主線程和兩個子線程之間同步所必需的,讀者可根據(jù)自己程序的需要另行添加。

因各線程都以文檔對象指針為參數(shù),這些事件都在文檔類頭文件中定義,這些事件在文檔類的構(gòu)造函數(shù)中生成并賦初值。

CCtrsysDoc::CCtrsysDoc()

{

……

m_hEventPost=CreateEvent(NULL,TRUE,TRUE, NULL);

m_hEventCalcDone=CreateEvent(NULL,TRUE,F(xiàn)ALSE, NULL);

m_hEventCalcStarted=CreateEvent(NULL,TRUE,F(xiàn)ALSE,NULL);

m_hEventStartCalc=CreateEvent(NULL, TRUE,F(xiàn)ALSE, NULL);

m_hEventSourceDataUpdated=CreateEvent(NULL,TRUE,F(xiàn)ALSE, NULL);

m_hEventUpdateSourceData=CreateEvent(NULL,TRUE,F(xiàn)ALSE, NULL);

m_hEventCalcStopped=CreateEvent(NULL,TRUE,F(xiàn)ALSE, NULL);

m_hEventStopCalc=CreateEvent(NULL, TRUE,F(xiàn)ALSE, NULL);

……

}

線程的同步工作主要在主框架CMainFrame類的WM_USER_COMM_NOTIFY消息響應(yīng)函數(shù)OnCommNotify中進(jìn)行。當(dāng)下方通信設(shè)備參數(shù)改變時,通信線程發(fā)送給CMainFrame類一個WM_USER_COMM_NOTIFY消息。CMainFrame類接收到消息后,在消息響應(yīng)函數(shù)OnCommNotify中終止計算線程的當(dāng)前計算,計算成功終止后由通信線程更新計算所需的數(shù)據(jù)源,待更新完畢后,重新開始計算。線程同步部分流程如圖2。

3.3 通信線程

通信線程部分流程如圖3所示。

3.4 計算線程

編程者應(yīng)根據(jù)數(shù)據(jù)處理過程,在運算量較大或循環(huán)次數(shù)較多的地方設(shè)置對m_hEventStopCalc事件的查詢。當(dāng)數(shù)據(jù)發(fā)生更新時,使用其它線程類型和同步方法往往必須等到數(shù)據(jù)處理部分結(jié)束,這樣整個一次數(shù)據(jù)處理都是無用計算;而采用上述方法,因數(shù)據(jù)更新所造成的無用計算僅僅是一步循環(huán)或幾行指令,相比而言,所導(dǎo)致的延時和CPU浪費是微不足道的。

計算線程部分流程如圖4所示。

責(zé)任編輯:gt

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 應(yīng)用程序
    +關(guān)注

    關(guān)注

    37

    文章

    3198

    瀏覽量

    57357
  • 工控系統(tǒng)
    +關(guān)注

    關(guān)注

    1

    文章

    98

    瀏覽量

    14501
  • 線程
    +關(guān)注

    關(guān)注

    0

    文章

    501

    瀏覽量

    19580
收藏 人收藏

    評論

    相關(guān)推薦

    Java多線程的用法

    能力。 什么是進(jìn)程 是指正在運行的程序的實例。 每個進(jìn)程都擁有自己的內(nèi)存空間、代碼、數(shù)據(jù)和文件等資源,可以獨立運行、調(diào)度和管理。在操作系統(tǒng)中,進(jìn)程是系統(tǒng)資源分配的最小單位,是實現(xiàn)多任務(wù)的基礎(chǔ)。 Java
    的頭像 發(fā)表于 09-30 17:07 ?843次閱讀

    labview的多線程技術(shù)

    用labview開發(fā)多線程技術(shù)難道是在同一個框圖中多用幾個WHILE循環(huán)結(jié)構(gòu)嗎?請玩過的大神指教
    發(fā)表于 07-25 10:33

    labview多線程技術(shù)

    labview多線程技術(shù)
    發(fā)表于 01-10 16:00

    Windows95環(huán)境下如何用VisualC++進(jìn)行ODBC編程?

    Windows95環(huán)境下用VisualC++進(jìn)行ODBC編程的方法及技巧是什么?
    發(fā)表于 11-11 06:18

    再談多線程技術(shù)在變電站監(jiān)控系統(tǒng)中的應(yīng)用

    介紹了Win32 中多線程技術(shù)的概念及在Delphi 開發(fā)環(huán)境中實現(xiàn)的方法,給出了在變電站監(jiān)控系統(tǒng)開發(fā)中如何較好地應(yīng)用多線程
    發(fā)表于 05-30 10:17 ?15次下載

    多線程技術(shù)在串口通信中的應(yīng)用

            首先介紹了多線程技術(shù)的基本原理,然后討論了多線程技術(shù)在串口通信中的應(yīng)用,并給出了
    發(fā)表于 09-04 09:10 ?18次下載

    基于多線程技術(shù)的連續(xù)小波變換實現(xiàn)

             簡要介紹了連續(xù)小波變換(CWT)的快速算法,討論了采用多線程技術(shù)編程實現(xiàn)CWT 的原理及方法。實驗表明,該方法可改善CW
    發(fā)表于 09-15 09:46 ?9次下載

    數(shù)控測井軟件中的多線程編程技術(shù)

    為滿足實時測井需求,提高系統(tǒng)資源的利用率和系統(tǒng)性能,在數(shù)控測井軟件開發(fā)中將多線程編程技術(shù)應(yīng)用其中,并對多線程編程
    發(fā)表于 01-14 17:17 ?32次下載

    Delphi教程_實現(xiàn)多線程共享探測

    Delphi教程實現(xiàn)多線程共享探測,很好的Delphi學(xué)習(xí)資料。
    發(fā)表于 03-16 14:59 ?3次下載

    基于OPC技術(shù)多線程數(shù)據(jù)采集系統(tǒng)實現(xiàn)_喬富強

    基于OPC技術(shù)多線程數(shù)據(jù)采集系統(tǒng)實現(xiàn)_喬富強
    發(fā)表于 01-12 19:56 ?1次下載

    linux多線程編程技術(shù)

    1 引言 線程(thread)技術(shù)早在60年代就被提出,但真正應(yīng)用多線程到操作系統(tǒng)中去,是在80年代中期,solaris是這方面的佼佼者。傳統(tǒng)的 Unix也支持
    發(fā)表于 10-24 16:01 ?5次下載

    什么是多線程編程?多線程編程基礎(chǔ)知識

    摘要:多線程編程是現(xiàn)代軟件技術(shù)中很重要的一個環(huán)節(jié)。要弄懂多線程,這就要牽涉到多進(jìn)程。本文主要以多線程編程以及多線程編程相關(guān)知識而做出的一些結(jié)
    發(fā)表于 12-08 16:30 ?1.2w次閱讀

    如何實現(xiàn)VisualC++與MATLAB接口進(jìn)行混合編程詳細(xì)方法說明

    本文首先論述了VisualC++和Matab混合編程的重要性和優(yōu)點,介紹了在Visualc++中利用Matlab引擎調(diào)用Matab函數(shù)的方法,和在Matlab中調(diào)用Visual C ++代碼的方法
    發(fā)表于 07-16 17:09 ?20次下載

    SpringBoot實現(xiàn)多線程

    SpringBoot實現(xiàn)多線程
    的頭像 發(fā)表于 01-12 16:59 ?1708次閱讀
    SpringBoot<b class='flag-5'>實現(xiàn)</b><b class='flag-5'>多線程</b>

    java實現(xiàn)多線程的幾種方式

    Java實現(xiàn)多線程的幾種方式 多線程是指程序中包含了兩個或以上的線程,每個線程都可以并行執(zhí)行不同的任務(wù)或操作。Java中的
    的頭像 發(fā)表于 03-14 16:55 ?432次閱讀