亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

RS232串口通訊模塊

2019-11-18 17:57:46
字體:
來源:轉載
供稿:網友

應為工作需要自己編寫的RS232通訊模塊,該模塊已經編寫了蠻久,在實際使用中可能有沒有考慮和不完善的地方。

//=================================================================================

//如果使用該模塊請保留該注釋,如果被修改或編輯請將修改后的代碼發送一份給我

//編寫:戴琪英

//E_Mail:qiyingdai@163.com

//2000-09-01

//=================================================================================

unit R232Comm;

interface
uses
  Windows,SysUtils;
const
  INITR12COMM_SUCCESS=0;
  INITR12COMM_FAILURE=-1;
var
  bSendFinish:boolean=True;//發送完標志
  iRecvLen:DWord=0;
  RecvBuff,TempBuff:PChar;
  SendCommand,RecvCommand:String;//發送和接收到的命令
  RecvFinish:BOOL=False;
  RecvBuffInit:BOOL=False;
  SendCommandSuccess:BOOL; //切換臺命令被成功發送標志

function  InitR12CommDev(comNo:PChar):String;  //初始化切換臺串口,返回狀態字符
PRocedure SwitchR12(WriteBuffer:PChar);//對切換臺進行切換函數
procedure SwitchR12Byte(WriteBuffer:Byte);
procedure CommSendNotify;//串口接收到字符事件響應過程
procedure CommRecvNotify; //串口發送緩沖區空事件響應過程
procedure CommWatchThread(var lpdwParam:DWORD);//通信口監視線程
function  ConInfo :String;

implementation
var
  //comMask,comBuf,ComState:Integer;
  dcb:_DCB; //DCB結構用于配置串口,程序中涉及各域含義如下:
            //DWORD DCBlength :DCB結構大小
            //DWORD BaudRate :  波特率
            //DWORD fBinary  : 1 二進制模式
            //DWORD fParity  : 1 進行奇偶校驗
            //BYTE  ByteSize: 字符位數 4~8
            //BYTE  Parity:   奇偶校驗位 0-4分別表示無、奇、偶、傳號、空號校驗
            //BYTE  StopBits: 停止位數 0-2分別表示 1、1.5、2個停止位
            //WORD  XonLim :XON 閾值
            //WORD  XoffLim  XOFF 閾值
            //char  XonChar: XON 字符
            //char  XoffChar: XOFF 字符
            //char  EvtChar:  事件字符
  comStat:_COMSTAT; //COMSTAT結構用于存放有關通信設備的當前信息
                    //程序中涉及各域含義如下:
                    //cbInQue :接收緩沖區中字符個數
                    //cbOutQue:發送緩沖區中字符個數
  dwErrorFlag:LongWord;
  hCommDev,comThreadHwnd:Thandle;//通信串口句柄和通信監視線程句柄
  comMask,comBuf,comState:BOOL;
  read_os,write_os:_OVERLAPPED;  //OVERLAPPED 結構,用于異步操作的Win32函數中
                                //程序中涉及各域含義如下:
                                //DWORD Interval 保留給操作系統使用
                                //DWORD IntervalHigh 保留給操作系統使用
                                //DOWD  hEvent 當I/O操作完成時被設置為有信號狀態
                                //的事件;當調用ReadFile和WriteFile函數之前,調
                                //用進程設置該事件
  postRecvEvent,postSendEvent:Thandle;//發送緩沖區空和接收到字符事件句柄
  dwThreadID1:DWORD; //通信監視線程ID號

///串口初始化函數
//該函數主要完成串口初始化設置和通信線程的啟動
//入口參數:串口號
//返回值;初始化是否成功的狀態字符
function  InitR12CommDev(comNo:PChar) :String;
begin
   ///打開串口
   hCommDev:=CreateFile(comNo,   //串口好
                       GENERIC_READ or GENERIC_WRITE,//對串口以讀寫方式打開
                       0,
                       nil,
                       OPEN_EXISTING,
                       FILE_ATTRIBUTE_NORMAL or FILE_FLAG_OVERLAPPED,//允許重疊操作
                       0
                       );
   if hCommDev=INVALID_HANDLE_VALUE then
        InitR12CommDev:='切換臺通訊端口初始化失敗.'
   else
      InitR12CommDev:='切換臺通訊端口初始化成功.';
   comMask:=SetCommMask(hCommDev,EV_RXFLAG);//設置事件掩碼
   //comBuf:=SetupComm(hCommDev,4096,4096);//設置接收和發送緩沖區大小皆為4096字節
   comBuf:=SetupComm(hCommDev,1,1);//設置接收和發送緩沖區大小皆為4096字節
   if  comBuf=False then
         InitR12CommDev:='切換臺通訊端口初始化失敗.'
   else
      begin
         InitR12CommDev:='切換臺通訊端口初始化成功.';
         //清空緩沖區
         PurgeComm(hCommDev,PURGE_TXABORT or PURGE_RXABORT or
                                      PURGE_TXCLEAR or PURGE_RXCLEAR ) ;
      end;

   //以下對串口進行配置
   dcb.DCBlength:=sizeof(_DCB);
   comState:=GetCommState(hCOmmDev,dcb);  //得到缺省設置
   if  comState=False then
         InitR12CommDev:='切換臺通訊端口初始化失敗.'
   else
       InitR12CommDev:='切換臺通訊端口初始化成功.';
   dcb.BaudRate:=9600;  //波特率 9600
   dcb.ByteSize:=8;//7;  //數據長度7位
   dcb.Parity:=NOPARITY;//ODDPARITY; //校驗方式 奇校驗
   dcb.StopBits:=ONESTOPBIT; //停止位 1 位

   dcb.Flags := 0;         // Enable fBinary
   dcb.Flags := dcb.Flags or 2;          // Enable parity check
   dcb.XonChar:= chr($00) ;
   dcb.XoffChar:= chr($00) ;
   dcb.XonLim:= 100 ;
   dcb.XoffLim:= 100 ;
   dcb.EvtChar := Char($ff);

   comState:=SetCommState(hCommDev,dcb);  //設置串口
   if comState=False then
         InitR12CommDev:='切換臺通訊端口初始化失敗.'
   else
      InitR12CommDev:='切換臺通訊端口初始化成功.';
     //設置通信接收到字符事件句柄
   postRecvEvent:=CreateEvent(NIL,
                              TRUE,//手工重置事件
                              TRUE, //初始化為有信號狀態
                              NIL
                              );
   //設置讀異步I/O操作事件句柄
   read_os.hEvent:=CreateEvent(NIL,
                              TRUE,//手工重置事件
                              FALSE, //初始化為無信號狀態
                              NIL
                              );
   //設置發送緩沖區空事件句柄
   postSendEvent:=CreateEvent(NIL,
                              TRUE,//手工重置事件
                              TRUE, //初始化為有信號狀態
                              NIL);
   //設置寫異步I/O操作事件句柄
   write_os.hEvent:=CreateEvent(NIL,
                              TRUE,//手工重置事件
                              FALSE,//初始化為無信號狀態
                              NIL);
   //創建通信監視線程
   comThreadHwnd:=CreateThread(NIL,
                         0,
                         @CommWatchThread, //通信線程函數的地址
                         nil,
                         0,   //創建后立即運行
                         dwThreadID1);//通信線程ID號
   if comThreadHwnd=INVALID_HANDLE_VALUE  then
      InitR12CommDev:='INITR12COMM_FAILURE'
   else
      InitR12CommDev:='切換臺通訊端口初始化成功.';
end;

///切換臺切換控制函數
///輸入參數;切換命令字符串
procedure SwitchR12(WriteBuffer:PChar);
var
 dwWriteByte,TxCount:DWORD;
 bl:BOOL;
 dwError:DWORD;

begin
     //WriteBuffer:=chr($0D)+'03A00';
     TxCount:=StrLen(WriteBuffer);
     if bSendFinish=True then  //發送緩沖區空發送
     begin
         dwWriteByte:=0;
         bSendFinish:=False;
         bl:=WriteFile(hCommDev,Byte(WriteBuffer^),TxCount,dwWriteByte,@write_os);
         if bl=True then
         begin
          bSendFinish:=True;
          PurgeComm(hCommDev,PURGE_TXCLEAR );//如果發送完成,置緩沖區空標志,并清空緩沖區
         end;
         if bl=False then
         begin
           dwError:=GetLastError();
           if (dwError=ERROR_IO_PENDING) or (dwError=ERROR_IO_INCOMPLETE) then
           begin
             bl:=GetOverLappedResult(hCommDev,
                             write_os,dwWriteByte,TRUE);//如果未發送完命令字符
                                                    //等待發送完成
             if bl=True then
             begin
                bSendFinish:=True;
                PurgeComm(hCommDev,PURGE_TXCLEAR ); //發送完成 置緩沖區空標志,并清空緩沖區
                //Result:=True;
             end;
           end;
         end;
     end;
     //Result:=True;
end;

procedure SwitchR12Byte(WriteBuffer:Byte);
var
 dwWriteByte,TxCount:DWORD;
 bl:BOOL;
 dwError:DWORD;

begin
     //WriteBuffer:=chr($0D)+'03A00';
     TxCount:= 1 ;//StrLen(WriteBuffer);
     if bSendFinish=True then  //發送緩沖區空發送
     begin
         dwWriteByte:=0;
         bSendFinish:=False;
         bl:=WriteFile(hCommDev,WriteBuffer,TxCount,dwWriteByte,@write_os);
         if bl=True then
         begin
          bSendFinish:=True;
          PurgeComm(hCommDev,PURGE_TXCLEAR );//如果發送完成,置緩沖區空標志,并清空緩沖區
         end;
         if bl=False then
         begin
           dwError:=GetLastError();
           if (dwError=ERROR_IO_PENDING) or (dwError=ERROR_IO_INCOMPLETE) then
           begin
             bl:=GetOverLappedResult(hCommDev,
                             write_os,dwWriteByte,TRUE);//如果未發送完命令字符
                                                    //等待發送完成
             if bl=True then
             begin
                bSendFinish:=True;
                PurgeComm(hCommDev,PURGE_TXCLEAR ); //發送完成 置緩沖區空標志,并清空緩沖區
                //Result:=True;
             end;
           end;
         end;
     end;
     //Result:=True;
end;

////通信監視線程
procedure CommWatchThread(var lpdwParam:DWORD);
var
    dwTransfer,dwEvtMask,dwError:DWORD;
    os:_OVERLAPPED;
    bl:boolean;

begin
    os.hEvent:=CreateEvent(nil,
                          TRUE,
                          FALSE,
                          NIL);

    comMask:=SetCommMask(hCommDev,EV_RXCHAR or EV_TXEMPTY);//設置監視的事件為接
                                                        //收到字符或發送緩沖區空
    if comMask=True then
    begin
        while True do
        begin
           dwEvtMask:=0;
           bl:=WaitCommEvent(hCommDev,dwEvtMask,@os); //查詢所監視的通信事件是否
                                                       //已經發生
           if bl=False then
           begin
             dwError:=GetLastError();
             if dwError=ERROR_IO_PENDING then
                GetOverlappedResult(hCOmmDev,os,dwTransfer,TRUE);//若未監測到通信事件
                                           //則在此等待事件發生
           end;
           //有事件,進行如下處理
           if (dwEvtMask and EV_RXCHAR)=EV_RXCHAR then //判斷是否為接收到 字符事件
           begin
              WaitForSingleObject(postRecvEvent,$FFFFFFFF);//等待接收事件句柄為有
                                                      //信號狀態
              ResetEvent(postRecvEvent); //置接收事件句柄為無信號狀態,以免接收
                                        //緩沖區被覆蓋
              CommRecvNotify; //調用接收到字符處理函數
              continue; //處理完接收字符,繼續監測通信事件
           end;
           if (dwEvtMask and EV_TXEMPTY)=EV_TXEMPTY then //判斷是否為發送緩沖區空事件
           begin
              WaitForSingleObject(postSendEvent,$FFFFFFFF);//等待發送事件句柄為有
                                                           //信號狀態
              ResetEvent(postSendEvent); //置發送事件句柄為無信號狀態,,以免發送
                                        //緩沖區被覆蓋
              CommSendNotify; //調用發送緩沖區空處理函數
              continue;//處理完,繼續監測通信事件
           end;
        end;
    end;
    CloseHandle(os.hEvent);
end;

//發送緩沖區空處理過程
procedure CommSendNotify;
begin
    SetEvent(postSendEvent);//置發送事件未有信號狀態,以便進行下一次發送
end;

///接收到字符處理函數
procedure CommRecvNotify;
var
     RxCount,dwReadByte:DWORD;
     inData :Byte;
begin
     ClearCommError(hCommDev,dwErrorFlag,@ComStat);
     RxCount:=ComStat.cbInQue; //獲取接收緩沖區的字符個數
     if RxCount>0 then
     begin
       if not RecvBuffInit then
       begin
          StrCopy(RecvBuff,'');
          RecvBuffInit:=True;
       end;
       StrCopy(TempBuff,'');
       ReadFile(hCommDev,Byte(TempBuff^),RxCount,dwReadByte,@read_os);//讀字符存入
                                                                      //臨時緩沖區中
       iRecvLen:=iRecvLen+dwReadByte; //接收到字符個數統計

       if iRecvLen >=1 then
       begin
            inData := Byte(TempBuff^);
            if inData = $D9 then
            begin
                 SendCommandSuccess:=True;  //如果狀態一致,則置該標志為真,標志切換成功
            end
            else
            begin
                 SendCommandSuccess:=False;//否則,置該標志為假,表示切換失敗
            end;

            iRecvLen:=0;
            StrCopy(RecvBuff,'');
            RecvBuffInit:=False;
            PurgeComm(hCommDev,PURGE_RXCLEAR ); //清空接收緩沖區
       end
    end;
    ////////////////
    SetEvent(postRecvEvent); //置接收事件句柄為有信號狀態,以便接收新字符

end;

function ConInfo :String;
begin
     if  SendCommandSuccess =True then
     begin
          Result := '切換器聯機監測成功!';
     end
     else
     begin
          Result := '切換器聯機監測失?。?;
     end;
end;

{
procedure CommSendNotify;
begin
    SetEvent(postSendEvent);//置發送事件未有信號狀態,以便進行下一次發送
end;

///接收到字符處理函數
{procedure CommRecvNotify;
var
     RxCount,dwReadByte:DWORD;
     inData :Byte;
begin
     ClearCommError(hCommDev,dwErrorFlag,@ComStat);
     RxCount:=ComStat.cbInQue; //獲取接收緩沖區的字符個數
     if RxCount>0 then
     begin
       if not RecvBuffInit then
       begin
          StrCopy(RecvBuff,'');
          RecvBuffInit:=True;
       end;
       StrCopy(TempBuff,'');
       ReadFile(hCommDev,Byte(TempBuff^),RxCount,dwReadByte,@read_os);//讀字符存入
       //ReadFile(hCommDev,Byte(TempBuff^),RxCount,dwReadByte,@read_os);//讀字符存入
                                              //臨時緩沖區中
       iRecvLen:=iRecvLen+dwReadByte; //接收到字符個數統計
       {
       if iRecvLen<13 then
       begin
          strcat(Recvbuff,TempBuff); //若接收到的切換臺狀態字符小于13個,
                          //將臨時緩沖區中的字符拷貝到接收命令緩沖區,準備繼續讀
       end
       else
       begin
         strcat(Recvbuff,TempBuff);
         RecvCommand:=RecvBuff;
         //若接收到13個切換臺狀態字符進行如下處理
         if (RecvCommand[7]='P')
            and(RecvCommand[8]=SendCommand[7])     //比較讀入的切換臺端口狀態
            and  (RecvCommand[9]=SendCommand[8])   //是否與切換指令中切換的端口
            and (RecvCommand[10]=SendCommand[9])   //一致
            and (RecvCommand[11]=SendCommand[10])  then

         begin
            SendCommandSuccess:=True;  //如果狀態一致,則置該標志為真,標志切換成功
         end
         else
         begin
           SendCommandSuccess:=False;//否則,置該標志為假,表示切換失敗
         end;
         iRecvLen:=0;
         StrCopy(RecvBuff,'');
         RecvBuffInit:=False;
         PurgeComm(hCommDev,PURGE_RXCLEAR ); //清空接收緩沖區
       end;
       }
       {
       if iRecvLen >=1 then
       begin
            inData := Byte(TempBuff^);
            if inData = $D9 then
            begin
                 SendCommandSuccess:=True;  //如果狀態一致,則置該標志為真,標志切換成功

            end
            else
            begin
                 SendCommandSuccess:=False;//否則,置該標志為假,表示切換失敗
            end;

            iRecvLen:=0;
            StrCopy(RecvBuff,'');
            RecvBuffInit:=False;
            PurgeComm(hCommDev,PURGE_RXCLEAR ); //清空接收緩沖區
       end
    end;
    ////////////////
    SetEvent(postRecvEvent); //置接收事件句柄為有信號狀態,以便接收新字符

end;
}

initialization
    RecvBuff:=StrAlloc(50*sizeof(Char));
    TempBuff:=StrAlloc(50*sizeof(Char));
Finalization
    StrDispose(RecvBuff);
    StrDispose(TempBuff);
    CloseHandle(PostRecvEvent);
    CloseHandle(read_os.hEvent);
    CloseHandle(PostSendEvent);
    CloseHandle(write_os.hEvent);
end.



上一篇:一個導出Excel非??斓念?/a>

下一篇:創建一個簡單的&quot;專家&quot;

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
學習交流
熱門圖片

新聞熱點

疑難解答

圖片精選

網友關注

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲欧美成人一区二区在线电影| 福利精品视频在线| 国语自产精品视频在线看一大j8| 欧美极品美女电影一区| 91精品久久久久久久久久久久久久| 国产视频在线观看一区二区| 91亚洲精品在线| 日本韩国欧美精品大片卡二| 亚洲视频网站在线观看| 97不卡在线视频| 亚洲免费影视第一页| 国外视频精品毛片| 久久视频在线免费观看| 国产91av在线| 亚洲精品国产精品国自产在线| 亚洲精品免费在线视频| 97国产在线观看| 亚洲xxxxx| 欧美激情区在线播放| 美女撒尿一区二区三区| 91视频国产精品| 久久精品人人爽| 精品一区二区亚洲| 国内精品久久久久| 久久久亚洲国产天美传媒修理工| 国产香蕉97碰碰久久人人| 一区二区三区www| 97在线精品国自产拍中文| 久久久久久久久中文字幕| 欧美成人午夜剧场免费观看| 亚洲国产精品热久久| 国产精品久久久久福利| 日韩一区二区在线视频| 国产高清在线不卡| 在线精品高清中文字幕| 在线成人激情视频| 久久国产精彩视频| 国产原创欧美精品| 中文字幕v亚洲ⅴv天堂| 亚洲国产成人av在线| 亚洲精品在线观看www| 国产一区二区三区视频在线观看| 国产精品久久精品| 欧美成人午夜激情在线| 国产精品夫妻激情| 欧美日韩亚洲精品内裤| 久久久在线免费观看| 久久综合亚洲社区| 欧美日韩一二三四五区| 久久成人亚洲精品| 一区二区三区日韩在线| 亚洲欧美中文另类| 亚洲最新av网址| 国产suv精品一区二区三区88区| 久久久视频精品| 成人xvideos免费视频| 亚洲欧美日韩精品久久奇米色影视| 欧美精品少妇videofree| 欧美激情国内偷拍| 国a精品视频大全| 亚洲国产精品99久久| 亚洲免费一在线| 日韩美女免费观看| 亚洲无限乱码一二三四麻| 日韩视频在线免费观看| 日本精品久久电影| 亚洲国产精品成人va在线观看| 欧美夫妻性生活视频| 久久青草福利网站| 日韩高清有码在线| 国产一区欧美二区三区| 欧美成人免费大片| 国产精品福利在线观看| 欧美午夜片欧美片在线观看| 国产精品aaaa| 国产欧美亚洲精品| 久久国产精品亚洲| 日韩av有码在线| 欧美性色xo影院| 黑丝美女久久久| 亚洲自拍欧美色图| 57pao国产成人免费| 91精品国产综合久久男男| 久久精品久久久久| 日本一区二区在线播放| 欧美电影电视剧在线观看| 日韩欧美在线免费观看| 日韩高清欧美高清| 成人免费看黄网站| 久久久之久亚州精品露出| 日韩激情av在线播放| 亚洲aⅴ男人的天堂在线观看| 亚洲18私人小影院| 亚洲新中文字幕| 亚洲天堂影视av| 日韩免费高清在线观看| 久久91亚洲精品中文字幕| 亚洲爱爱爱爱爱| 九九九热精品免费视频观看网站| 国产精品老牛影院在线观看| 久久久99免费视频| 亚洲欧美制服另类日韩| 亚洲国产另类 国产精品国产免费| 久久久极品av| 伊人青青综合网站| 中文字幕av一区二区三区谷原希美| 国产精品日韩在线一区| 国模精品视频一区二区| 欧美激情videoshd| 91久久精品一区| 欧美精品久久一区二区| 久久香蕉国产线看观看av| 伊人久久五月天| 91精品久久久久久| 成人综合网网址| 青青草原一区二区| 国产欧美精品日韩精品| 国产精品久久久久久久久免费看| 国产手机视频精品| 欧美另类69精品久久久久9999| 在线日韩精品视频| 欧美日韩精品在线观看| 亚洲毛片一区二区| 狠狠爱在线视频一区| 久久精品视频va| 国产精品91久久久| 色综合久久久888| 亚洲va久久久噜噜噜久久天堂| 欧美第一黄网免费网站| 成人免费看片视频| 亚洲人成电影网站| 欧美性猛xxx| 欧美影院久久久| 欧美日韩在线看| 中文字幕日韩有码| 国产a∨精品一区二区三区不卡| 久久国产精品久久久久久| 最好看的2019年中文视频| 亚洲精品综合久久中文字幕| 国产欧美日韩中文| 亚洲高清福利视频| 亚洲成色777777女色窝| 国产精品九九九| 欧美日韩xxxxx| 亚洲性夜色噜噜噜7777| 亚洲黄一区二区| 原创国产精品91| 亚洲va欧美va国产综合剧情| 日韩精品免费看| 欧美第一黄网免费网站| 一区二区三区日韩在线| 亚洲欧美日韩一区二区在线| 国产精品流白浆视频| 日韩中文字幕在线观看| 欧美激情女人20p| 夜夜嗨av一区二区三区免费区| 日韩视频免费看| 91精品久久久久久久久久另类| 日韩欧美在线网址| 色777狠狠综合秋免鲁丝| 亚洲国产成人在线视频| 亚洲第一区第二区| 欧日韩不卡在线视频| 日韩小视频在线观看|