1. 程式人生 > >Delphi 串列埠通訊(1)

Delphi 串列埠通訊(1)

利用 Delphi實現串列埠通訊的常用的方法有 3種: 一是利用控制元件,如 MSCOMM控制元件和 SPCOMM控制元件; 二是使用 API函式; 三是呼叫其他串列埠通訊程式。其中利用 API編寫串列埠通訊程式較為複雜,需要掌握大量的通訊知識。相比較而言, 利用 SPCOMM控制元件則相對較簡單,並且該控制元件具有豐富的與串列埠通訊密切相關的屬性及事件,提供了對串列埠的各種操作,而且還支援多執行緒。下面本文結合例項詳細介紹 SPCOMM控制元件的使用。 SPCOMM的安裝 1.選擇下拉選單 Component中的 Install Component選項,彈出如圖 1所示的視窗。 圖 1 在 Unit file name處填寫 SPCOMM控制元件所在的路徑,其他各項可用預設值,點選 OK按鈕。 2.安裝後,在 System控制元件面板中將出現一個紅色控制元件 COM。現在就可以像 Delphi自帶控制元件一樣使用 COM控制元件了。 SPCOMM的屬性、方法和事件 1.屬性
●CommName:表示 COM1、 COM2等串列埠的名字; ●BaudRate:根據實際需要設定的波特率,在串列埠開啟後也可更改此值,實際波特率隨之更改; ●ParityCheck:表示是否需要奇偶校驗; ●ByteSize:根據實際情況設定的位元組長度; ●Parity:奇偶校驗位; ●StopBits:停止位; ●SendDataEmpty:這是一個布林型屬性,為 true時表示傳送快取為空,或者傳送佇列裡沒有資訊;為 false時表示傳送快取不為空,或者傳送佇列裡有資訊。 2.方法 ●Startcomm方法用於開啟串列埠,當開啟失敗時通常會報錯。錯誤主要有 7種:⑴串列埠已經開啟;⑵開啟串列埠錯誤;⑶檔案控制代碼不是通訊控制代碼;⑷不能夠安裝通訊快取;⑸不能產生事件;⑹不能產生讀程序;⑺不能產生寫程序; ●StopComm方法用於關閉串列埠,沒有返回值; ●WriteCommData(pDataToWrite: PChar;dwSizeofDataToWrite:Word )方法是個帶有布林型返回值的函式,用於將一個字串傳送到寫程序,傳送成功返回 true,傳送失敗返回 false。執行此函式將立即得到返回值,傳送操作隨後執行。該函式有兩個引數,其中 pDataToWrite是要傳送的字串, dwSizeofDataToWrite是傳送字串的長度。 3.事件 ●OnReceiveData :procedure (Sender: TObject;Buffer: Pointer;BufferLength: Word) of object 當有資料輸入快取時將觸發該事件,在這裡可以對從串列埠收到的資料進行處理。 Buffer中是收到的資料, BufferLength是收到的資料長度。 ●OnReceiveError : procedure(Sender: TObject; EventMask : DWORD) 當接收資料出現錯誤時將觸發該事件。 SPCOMM的使用
下面是一個利用 SPCOMM控制元件的串列埠通訊的例子。 以實現 PC機與微控制器 8051之間的通訊為例,首先要調通它們之間的握手訊號。假定它們之間的通訊協議是: PC到 8051一幀資料 6個位元組, 8051到 PC一幀資料也為 6個位元組。當 PC發出( F0,01,FF,FF,01,F0)後 8051能收到一幀( F0,01,FF,FF,01,F0),表示資料通訊握手成功,兩者之間就可以按照協議相互傳輸資料。 建立一個新的工程 COMM.DPR,把窗體的 NAME屬性定為 FCOMM,把窗體的標題定義為測試通訊,按照圖 2新增控制元件 (圖 2中黑色矩形圍住的控制元件即為 COMM1)。 圖 2 1.設定 COMM1屬性: ●波特率: 4800; ●奇偶校驗位:無; ●位元組長度: 8; ●停止位: 1; ●串列埠: COM1。 Memo1中將顯示傳送和接收的資料。將新的窗體儲存為 Comm.pas。 2.編寫原始碼 //變數說明 var fcomm: TFCOMM; viewstring:string; i:integer; rbuf,sbuf:array[16] of byte; //開啟串列埠 procedure TFCOMM.FormShow(Sender: TObject); begin comm1.StartComm; end; //關閉串列埠 procedure TFCOMM.FormClose(Sender: TObject; var Action: TCloseAction); begin comm1.StopComm; end; //自定義傳送資料過程 procedure senddata; var i:integer; commflg:boolean; begin viewstring:=‘’ ; commflg:=true; for i:=1 to 6 do begin if not fcomm.comm1.writecommdata(@sbuf[i],1) then begin commflg:=false; break; end; //傳送時位元組間的延時 sleep(2); viewstring:=viewstring+ inttohex(sbuf[i],2)+‘’ ; end; viewstring:=‘傳送’+ viewstring; fcomm.memo1.lines.add(viewstring); fcomm.memo1.lines.add(‘’ ); if not commflg then messagedlg(‘傳送失敗 !’ ,mterror,[mbyes],0); end; //傳送按鈕的點選事件 procedure TFCOMM.Btn_sendClick(Sender: TObject); begin sbuf[1]:=byte($ f0); //幀頭 sbuf[2]:=byte($ 01); //命令號 sbuf[3]:=byte($ ff); sbuf[4]:=byte($ ff); sbuf[5]:=byte($ 01); sbuf[6]:=byte($ f0); //幀尾 senddata;//呼叫傳送函式 end; //接收過程 procedure TFCOMM.Comm1ReceiveData(Sender: TObject; Buffer: Pointer;BufferLength: Word); var i:integer; begin viewstring:=‘’ ; move(buffer^,pchar(@rbuf^),bufferlength); for i:=1 to bufferlength do viewstring:=viewstring+ inttohex(rbuf[i],2)+‘’ ; viewstring:=‘接收’+ viewstring; memo1.lines.add(viewstring); memo1.lines.add(‘’ ); end; 如果 memo1上顯示傳送 F0 01 FF FF 01 F0和接收到 F0 01 FF FF 01 F0,這表示串列埠已正確地傳送出資料並正確地接收到資料,則串列埠通訊成功。