目前,利用 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,這表示串口已正確地發送出資料並正確地接收到資料,則串口通訊成功。
spcomm控制項的Parity屬性值即可。
Even 表示偶校正
None 表示無校正
Odd 表示奇數同位
Mark 表示校正位為1
Space 表示校正位為0