C++串口編程實例

在windows程序設計與開發過程當中,特別是涉及到開發嵌入式軟硬件系統時,每每會涉及到串口編程。網上以及一些書籍上講解windows下的串口編程知識也挺多的,但我我的以爲,看完書上的知識點有時依然不知道該如何下手開始本身的程序設計和代碼編寫,許多知識若是能結合着詳細的例子每每可以幫助咱們學習得更快,因此,在此,我專門爲串口編程初學者設計了一個詳細的例子,供你們參考和學習。
   
下面我將本身用C++編寫的串口通訊的例子貼出來,其特色以下:
   
1. 本例子使用了比較規範的軟件設計方法,類的設計具備比較好的可擴展性和移植性、代碼的註釋採用doxgen支持的javaDoc風格。
2. 爲了能方便初學者更快地瞭解和入門,幾乎每一行代碼都加上了詳細的註釋,對於註釋中若是依然有不清楚的概念,相信你經過百度和google必定能找到答案。
3. 本例子設計的串口操做類能夠直接移植到其餘的工程中去,你們也能夠根據本身的須要添加其餘的接口。
4. 本例子只實現了串口數據的基本收發功能,其實爲了保證串口數據傳輸的正確性,每每須要設計一些串口通訊協議,協議的設計有待你本身完成,若是之後有時間,我也會嘗試提供一種比較基本的串口通訊協議設計案例給你們學習。
5. 關於本程序的驗證方法,能夠使用虛擬串口軟件VSPM和串口調試助手進行程序的測試與驗證,上述兩個軟件的使用方法請參考: http://ticktick.blog.51cto.com/823160/285610
 
下面即爲例子工程的三個文件,SerialPort.h、SerialPort.cpp、maincpp
 
附件中是工程文件,須要使用vs2008打開
     
  
  
           
  
  
  1. //////////////////////////////////////////////////////////////////////////  
  2. /// COPYRIGHT NOTICE  
  3. /// Copyright (c) 2009, 華中科技大學tickTick Group  (版權聲明)  
  4. /// All rights reserved.  
  5. ///   
  6. /// @file    SerialPort.h    
  7. /// @brief   串口通訊類頭文件  
  8. ///  
  9. /// 本文件完成串口通訊類的聲明  
  10. ///  
  11. /// @version 1.0     
  12. /// @author  盧俊   
  13. /// @E-mail:lujun.hust@gmail.com  
  14. /// @date    2010/03/19  
  15. ///  
  16. ///  修訂說明:  
  17. //////////////////////////////////////////////////////////////////////////  
  18.  
  19. #ifndef SERIALPORT_H_  
  20. #define SERIALPORT_H_  
  21.  
  22. #include <Windows.h>  
  23.  
  24. /** 串口通訊類  
  25.  *     
  26.  *  本類實現了對串口的基本操做  
  27.  *  例如監聽發到指定串口的數據、發送指定數據到串口  
  28.  */ 
  29. class CSerialPort  
  30. {  
  31. public:  
  32.     CSerialPort(void);  
  33.     ~CSerialPort(void);  
  34.  
  35. public:  
  36.       
  37.     /** 初始化串口函數  
  38.      *  
  39.      *  @param:  UINT portNo 串口編號,默認值爲1,即COM1,注意,儘可能不要大於9  
  40.      *  @param:  UINT baud   波特率,默認爲9600  
  41.      *  @param:  char parity 是否進行奇偶校驗,'Y'表示須要奇偶校驗,'N'表示不須要奇偶校驗  
  42.      *  @param:  UINT databits 數據位的個數,默認值爲8個數據位  
  43.      *  @param:  UINT stopsbits 中止位使用格式,默認值爲1  
  44.      *  @param:  DWORD dwCommEvents 默認爲EV_RXCHAR,即只要收發任意一個字符,則產生一個事件  
  45.      *  @return: bool  初始化是否成功  
  46.      *  @note:   在使用其餘本類提供的函數前,請先調用本函數進行串口的初始化  
  47.      *       \n本函數提供了一些經常使用的串口參數設置,若須要自行設置詳細的DCB參數,可以使用重載函數  
  48.      *           \n本串口類析構時會自動關閉串口,無需額外執行關閉串口  
  49.      *  @see:      
  50.      */ 
  51.     bool InitPort( UINT  portNo = 1,UINT  baud = CBR_9600,char  parity = 'N',UINT  databits = 8, UINT  stopsbits = 1,DWORD dwCommEvents = EV_RXCHAR);  
  52.  
  53.     /** 串口初始化函數  
  54.      *  
  55.      *  本函數提供直接根據DCB參數設置串口參數  
  56.      *  @param:  UINT portNo  
  57.      *  @param:  const LPDCB & plDCB  
  58.      *  @return: bool  初始化是否成功  
  59.      *  @note:   本函數提供用戶自定義地串口初始化參數  
  60.      *  @see:      
  61.      */ 
  62.     bool InitPort( UINT  portNo ,const LPDCB& plDCB );  
  63.  
  64.     /** 開啓監聽線程  
  65.      *  
  66.      *  本監聽線程完成對串口數據的監聽,並將接收到的數據打印到屏幕輸出  
  67.      *  @return: bool  操做是否成功  
  68.      *  @note:   當線程已經處於開啓狀態時,返回flase  
  69.      *  @see:      
  70.      */ 
  71.     bool OpenListenThread();  
  72.  
  73.     /** 關閉監聽線程  
  74.      *  
  75.      *    
  76.      *  @return: bool  操做是否成功  
  77.      *  @note:   調用本函數後,監聽串口的線程將會被關閉  
  78.      *  @see:      
  79.      */ 
  80.     bool CloseListenTread();  
  81.  
  82.     /** 向串口寫數據  
  83.      *  
  84.      *  將緩衝區中的數據寫入到串口  
  85.      *  @param:  unsigned char * pData 指向須要寫入串口的數據緩衝區  
  86.      *  @param:  unsigned int length 須要寫入的數據長度  
  87.      *  @return: bool  操做是否成功  
  88.      *  @note:   length不要大於pData所指向緩衝區的大小  
  89.      *  @see:      
  90.      */ 
  91.     bool WriteData(unsigned char* pData, unsigned int length);  
  92.  
  93.     /** 獲取串口緩衝區中的字節數  
  94.      *  
  95.      *    
  96.      *  @return: UINT  操做是否成功  
  97.      *  @note:   當串口緩衝區中無數據時,返回0  
  98.      *  @see:      
  99.      */ 
  100.     UINT GetBytesInCOM();  
  101.  
  102.     /** 讀取串口接收緩衝區中一個字節的數據  
  103.      *  
  104.      *    
  105.      *  @param:  char & cRecved 存放讀取數據的字符變量  
  106.      *  @return: bool  讀取是否成功  
  107.      *  @note:     
  108.      *  @see:      
  109.      */ 
  110.     bool ReadChar(char &cRecved);  
  111.  
  112. private:  
  113.  
  114.     /** 打開串口  
  115.      *  
  116.      *    
  117.      *  @param:  UINT portNo 串口設備號  
  118.      *  @return: bool  打開是否成功  
  119.      *  @note:     
  120.      *  @see:      
  121.      */ 
  122.     bool openPort( UINT  portNo );  
  123.  
  124.     /** 關閉串口  
  125.      *  
  126.      *    
  127.      *  @return: void  操做是否成功  
  128.      *  @note:     
  129.      *  @see:      
  130.      */ 
  131.     void ClosePort();  
  132.       
  133.     /** 串口監聽線程  
  134.      *  
  135.      *  監聽來自串口的數據和信息  
  136.      *  @param:  void * pParam 線程參數  
  137.      *  @return: UINT WINAPI 線程返回值  
  138.      *  @note:     
  139.      *  @see:      
  140.      */ 
  141.     static UINT WINAPI ListenThread(void* pParam);  
  142.  
  143. private:  
  144.  
  145.     /** 串口句柄 */   
  146.     HANDLE  m_hComm;  
  147.  
  148.     /** 線程退出標誌變量 */   
  149.     static bool s_bExit;  
  150.  
  151.     /** 線程句柄 */   
  152.     volatile HANDLE    m_hListenThread;  
  153.  
  154.     /** 同步互斥,臨界區保護 */   
  155.     CRITICAL_SECTION   m_csCommunicationSync;       //!< 互斥操做串口  
  156.  
  157. };  
  158.  
  159. #endif //SERIALPORT_H_ 
  
  
           
  
  
  1. //////////////////////////////////////////////////////////////////////////  
  2. /// COPYRIGHT NOTICE  
  3. /// Copyright (c) 2009, 華中科技大學tickTick Group  (版權聲明)  
  4. /// All rights reserved.  
  5. ///   
  6. /// @file    SerialPort.cpp    
  7. /// @brief   串口通訊類的實現文件  
  8. ///  
  9. /// 本文件爲串口通訊類的實現代碼  
  10. ///  
  11. /// @version 1.0     
  12. /// @author  盧俊    
  13. /// @E-mail:lujun.hust@gmail.com  
  14. /// @date    2010/03/19  
  15. ///   
  16. ///  
  17. ///  修訂說明:  
  18. //////////////////////////////////////////////////////////////////////////  
  19.  
  20. #include "StdAfx.h"  
  21. #include "SerialPort.h"  
  22. #include <process.h>  
  23. #include <iostream>  
  24.  
  25. /** 線程退出標誌 */   
  26. bool CSerialPort::s_bExit = false;  
  27. /** 當串口無數據時,sleep至下次查詢間隔的時間,單位:秒 */   
  28. const UINT SLEEP_TIME_INTERVAL = 5;  
  29.  
  30. CSerialPort::CSerialPort(void)  
  31. : m_hListenThread(INVALID_HANDLE_VALUE)  
  32. {  
  33.     m_hComm = INVALID_HANDLE_VALUE;  
  34.     m_hListenThread = INVALID_HANDLE_VALUE;  
  35.  
  36.     InitializeCriticalSection(&m_csCommunicationSync);  
  37.  
  38. }  
  39.  
  40. CSerialPort::~CSerialPort(void)  
  41. {  
  42.     CloseListenTread();  
  43.     ClosePort();  
  44.     DeleteCriticalSection(&m_csCommunicationSync);  
  45. }  
  46.  
  47. bool CSerialPort::InitPort( UINT portNo /*= 1*/,UINT baud /*= CBR_9600*/,char parity /*= 'N'*/,  
  48.                             UINT databits /*= 8*/UINT stopsbits /*= 1*/,DWORD dwCommEvents /*= EV_RXCHAR*/ )  
  49. {  
  50.  
  51.     /** 臨時變量,將制定參數轉化爲字符串形式,以構造DCB結構 */   
  52.     char szDCBparam[50];  
  53.     sprintf_s(szDCBparam, "baud=%d parity=%c data=%d stop=%d", baud, parity, databits, stopsbits);  
  54.  
  55.     /** 打開指定串口,該函數內部已經有臨界區保護,上面請不要加保護 */   
  56.     if (!openPort(portNo))  
  57.     {  
  58.         return false;  
  59.     }  
  60.  
  61.     /** 進入臨界段 */   
  62.     EnterCriticalSection(&m_csCommunicationSync);  
  63.  
  64.     /** 是否有錯誤發生 */   
  65.     BOOL bIsSuccess = TRUE;  
  66.  
  67.     /** 在此能夠設置輸入輸出的緩衝區大小,若是不設置,則系統會設置默認值.  
  68.      *  本身設置緩衝區大小時,要注意設置稍大一些,避免緩衝區溢出  
  69.      */ 
  70.     /*if (bIsSuccess )  
  71.     {  
  72.         bIsSuccess = SetupComm(m_hComm,10,10);  
  73.     }*/ 
  74.  
  75.     /** 設置串口的超時時間,均設爲0,表示不使用超時限制 */ 
  76.     COMMTIMEOUTS  CommTimeouts;  
  77.     CommTimeouts.ReadIntervalTimeout         = 0;  
  78.     CommTimeouts.ReadTotalTimeoutMultiplier  = 0;  
  79.     CommTimeouts.ReadTotalTimeoutConstant    = 0;  
  80.     CommTimeouts.WriteTotalTimeoutMultiplier = 0;  
  81.     CommTimeouts.WriteTotalTimeoutConstant   = 0;   
  82.     if ( bIsSuccess)  
  83.     {  
  84.         bIsSuccess = SetCommTimeouts(m_hComm, &CommTimeouts);  
  85.     }  
  86.  
  87.     DCB  dcb;  
  88.     if ( bIsSuccess )  
  89.     {  
  90.         // 將ANSI字符串轉換爲UNICODE字符串  
  91.         DWORD dwNum = MultiByteToWideChar (CP_ACP, 0, szDCBparam, -1, NULL, 0);  
  92.         wchar_t *pwText = new wchar_t[dwNum] ;  
  93.         if (!MultiByteToWideChar (CP_ACP, 0, szDCBparam, -1, pwText, dwNum))  
  94.         {  
  95.             bIsSuccess = TRUE;  
  96.         }  
  97.  
  98.         /** 獲取當前串口配置參數,而且構造串口DCB參數 */   
  99.         bIsSuccess = GetCommState(m_hComm, &dcb) && BuildCommDCB(pwText, &dcb) ;  
  100.         /** 開啓RTS flow控制 */   
  101.         dcb.fRtsControl = RTS_CONTROL_ENABLE;   
  102.  
  103.         /** 釋放內存空間 */   
  104.         delete [] pwText;  
  105.     }  
  106.  
  107.     if ( bIsSuccess )  
  108.     {  
  109.         /** 使用DCB參數配置串口狀態 */   
  110.         bIsSuccess = SetCommState(m_hComm, &dcb);  
  111.     }  
  112.           
  113.     /**  清空串口緩衝區 */ 
  114.     PurgeComm(m_hComm, PURGE_RXCLEAR | PURGE_TXCLEAR | PURGE_RXABORT | PURGE_TXABORT);  
  115.  
  116.     /** 離開臨界段 */   
  117.     LeaveCriticalSection(&m_csCommunicationSync);  
  118.  
  119.     return bIsSuccess==TRUE;  
  120. }  
  121.  
  122. bool CSerialPort::InitPort( UINT portNo ,const LPDCB& plDCB )  
  123. {  
  124.     /** 打開指定串口,該函數內部已經有臨界區保護,上面請不要加保護 */   
  125.     if (!openPort(portNo))  
  126.     {  
  127.         return false;  
  128.     }  
  129.       
  130.     /** 進入臨界段 */   
  131.     EnterCriticalSection(&m_csCommunicationSync);  
  132.  
  133.     /** 配置串口參數 */   
  134.     if (!SetCommState(m_hComm, plDCB))  
  135.     {  
  136.         return false;  
  137.     }  
  138.  
  139.     /**  清空串口緩衝區 */ 
  140.     PurgeComm(m_hComm, PURGE_RXCLEAR | PURGE_TXCLEAR | PURGE_RXABORT | PURGE_TXABORT);  
  141.  
  142.     /** 離開臨界段 */   
  143.     LeaveCriticalSection(&m_csCommunicationSync);  
  144.  
  145.     return true;  
  146. }  
  147.  
  148. void CSerialPort::ClosePort()  
  149. {  
  150.     /** 若是有串口被打開,關閉它 */ 
  151.     if( m_hComm != INVALID_HANDLE_VALUE )  
  152.     {  
  153.         CloseHandle( m_hComm );  
  154.         m_hComm = INVALID_HANDLE_VALUE;  
  155.     }  
  156. }  
  157.  
  158. bool CSerialPort::openPort( UINT portNo )  
  159. {  
  160.     /** 進入臨界段 */   
  161.     EnterCriticalSection(&m_csCommunicationSync);  
  162.  
  163.     /** 把串口的編號轉換爲設備名 */   
  164.     char szPort[50];  
  165.     sprintf_s(szPort, "COM%d", portNo);  
  166.  
  167.     /** 打開指定的串口 */   
  168.     m_hComm = CreateFileA(szPort, /** 設備名,COM1,COM2等 */   
  169.               GENERIC_READ | GENERIC_WRITE, /** 訪問模式,可同時讀寫 */     
  170.               0,                           /** 共享模式,0表示不共享 */   
  171.               NULL,                         /** 安全性設置,通常使用NULL */   
  172.               OPEN_EXISTING,                /** 該參數表示設備必須存在,不然建立失敗 */   
  173.               0,      
  174.               0);      
  175.  
  176.     /** 若是打開失敗,釋放資源並返回 */   
  177.     if (m_hComm == INVALID_HANDLE_VALUE)  
  178.     {  
  179.         LeaveCriticalSection(&m_csCommunicationSync);  
  180.         return false;  
  181.     }  
  182.  
  183.     /** 退出臨界區 */   
  184.     LeaveCriticalSection(&m_csCommunicationSync);  
  185.  
  186.     return true;  
  187. }  
  188.  
  189. bool CSerialPort::OpenListenThread()  
  190. {  
  191.     /** 檢測線程是否已經開啓了 */   
  192.     if (m_hListenThread != INVALID_HANDLE_VALUE)  
  193.     {  
  194.         /** 線程已經開啓 */   
  195.         return false;  
  196.     }  
  197.  
  198.     s_bExit = false;  
  199.     /** 線程ID */   
  200.     UINT threadId;  
  201.     /** 開啓串口數據監聽線程 */   
  202.     m_hListenThread = (HANDLE)_beginthreadex(NULL, 0, ListenThread, this, 0, &threadId);  
  203.     if (!m_hListenThread)  
  204.     {  
  205.         return false;  
  206.     }  
  207.     /** 設置線程的優先級,高於普通線程 */   
  208.     if (!SetThreadPriority(m_hListenThread, THREAD_PRIORITY_ABOVE_NORMAL))  
  209.     {  
  210.         return false;  
  211.     }  
  212.  
  213.     return true;  
  214. }  
  215.  
  216. bool CSerialPort::CloseListenTread()  
  217. {     
  218.     if (m_hListenThread != INVALID_HANDLE_VALUE)  
  219.     {  
  220.         /** 通知線程退出 */   
  221.         s_bExit = true;  
  222.  
  223.         /** 等待線程退出 */   
  224.         Sleep(10);  
  225.  
  226.         /** 置線程句柄無效 */   
  227.         CloseHandle( m_hListenThread );  
  228.         m_hListenThread = INVALID_HANDLE_VALUE;  
  229.     }  
  230.     return true;  
  231. }  
  232.  
  233. UINT CSerialPort::GetBytesInCOM()  
  234. {  
  235.     DWORD dwError = 0;  /** 錯誤碼 */   
  236.     COMSTAT  comstat;   /** COMSTAT結構體,記錄通訊設備的狀態信息 */   
  237.     memset(&comstat, 0, sizeof(COMSTAT));  
  238.  
  239.     UINT BytesInQue = 0;  
  240.     /** 在調用ReadFile和WriteFile以前,經過本函數清除之前遺留的錯誤標誌 */   
  241.     if ( ClearCommError(m_hComm, &dwError, &comstat) )  
  242.     {  
  243.         BytesInQue = comstat.cbInQue; /** 獲取在輸入緩衝區中的字節數 */   
  244.     }  
  245.  
  246.     return BytesInQue;  
  247. }  
  248.  
  249. UINT WINAPI CSerialPort::ListenThread( void* pParam )  
  250. {  
  251.     /** 獲得本類的指針 */   
  252.     CSerialPort *pSerialPort = reinterpret_cast<CSerialPort*>(pParam);  
  253.  
  254.     // 線程循環,輪詢方式讀取串口數據  
  255.     while (!pSerialPort->s_bExit)   
  256.     {  
  257.         UINT BytesInQue = pSerialPort->GetBytesInCOM();  
  258.         /** 若是串口輸入緩衝區中無數據,則休息一會再查詢 */   
  259.         if ( BytesInQue == 0 )  
  260.         {  
  261.             Sleep(SLEEP_TIME_INTERVAL);  
  262.             continue;  
  263.         }  
  264.  
  265.         /** 讀取輸入緩衝區中的數據並輸出顯示 */ 
  266.         char cRecved = 0x00;  
  267.         do 
  268.         {  
  269.             cRecved = 0x00;  
  270.             if(pSerialPort->ReadChar(cRecved) == true)  
  271.             {  
  272.                 std::cout << cRecved ;   
  273.                 continue;  
  274.             }  
  275.         }while(--BytesInQue);  
  276.     }  
  277.  
  278.     return 0;  
  279. }  
  280.  
  281. bool CSerialPort::ReadChar( char &cRecved )  
  282. {  
  283.     BOOL  bResult     = TRUE;  
  284.     DWORD BytesRead   = 0;  
  285.     if(m_hComm == INVALID_HANDLE_VALUE)  
  286.     {  
  287.         return false;  
  288.     }  
  289.  
  290.     /** 臨界區保護 */   
  291.     EnterCriticalSection(&m_csCommunicationSync);  
  292.  
  293.     /** 從緩衝區讀取一個字節的數據 */   
  294.     bResult = ReadFile(m_hComm, &cRecved, 1, &BytesRead, NULL);  
  295.     if ((!bResult))  
  296.     {   
  297.         /** 獲取錯誤碼,能夠根據該錯誤碼查出錯誤緣由 */   
  298.         DWORD dwError = GetLastError();  
  299.  
  300.         /** 清空串口緩衝區 */   
  301.         PurgeComm(m_hComm, PURGE_RXCLEAR | PURGE_RXABORT);  
  302.         LeaveCriticalSection(&m_csCommunicationSync);  
  303.  
  304.         return false;  
  305.     }  
  306.  
  307.     /** 離開臨界區 */   
  308.     LeaveCriticalSection(&m_csCommunicationSync);  
  309.  
  310.     return (BytesRead == 1);  
  311.  
  312. }  
  313.  
  314. bool CSerialPort::WriteData( unsigned char* pData, unsigned int length )  
  315. {  
  316.     BOOL   bResult     = TRUE;  
  317.     DWORD  BytesToSend = 0;  
  318.     if(m_hComm == INVALID_HANDLE_VALUE)  
  319.     {  
  320.         return false;  
  321.     }  
  322.  
  323.     /** 臨界區保護 */   
  324.     EnterCriticalSection(&m_csCommunicationSync);  
  325.  
  326.     /** 向緩衝區寫入指定量的數據 */   
  327.     bResult = WriteFile(m_hComm, pData, length, &BytesToSend, NULL);  
  328.     if (!bResult)    
  329.     {  
  330.         DWORD dwError = GetLastError();  
  331.         /** 清空串口緩衝區 */   
  332.         PurgeComm(m_hComm, PURGE_RXCLEAR | PURGE_RXABORT);  
  333.         LeaveCriticalSection(&m_csCommunicationSync);  
  334.  
  335.         return false;  
  336.     }  
  337.  
  338.     /** 離開臨界區 */   
  339.     LeaveCriticalSection(&m_csCommunicationSync);  
  340.  
  341.     return true;  
  342. }  
  
  
           
  
  
  1. // main.cpp : Defines the entry point for the console application.  
  2. //  
  3.  
  4. #include "stdafx.h"  
  5. #include "SerialPort.h"  
  6. #include <iostream>  
  7.  
  8. int _tmain(int argc, _TCHAR* argv[])  
  9. {  
  10.  
  11.     CSerialPort mySerialPort;  
  12.  
  13.     if (!mySerialPort.InitPort(2))  
  14.     {  
  15.         std::cout << "initPort fail !" << std::endl;  
  16.     }  
  17.     else 
  18.     {  
  19.         std::cout << "initPort success !" << std::endl;  
  20.     }  
  21.  
  22.     if (!mySerialPort.OpenListenThread())  
  23.     {  
  24.         std::cout << "OpenListenThread fail !" << std::endl;  
  25.     }  
  26.     else 
  27.     {  
  28.         std::cout << "OpenListenThread success !" << std::endl;  
  29.     }  
  30.  
  31.     int temp;  
  32.     std::cin >> temp;  
  33.  
  34.     return 0;  
  35. }  
相關文章
相關標籤/搜索