程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> Delphi >> 用Delphi編寫點對點傳文件程序(1)

用Delphi編寫點對點傳文件程序(1)

編輯:Delphi
  Delphi功能強大,用Delphi寫軟件,可以大大縮短軟件的開發周期。關於點對點傳文件的基本思路,就是一個服務器軟件,一個客戶端軟件,使用同一個端口,待連接上以後,客戶端給服務器發送一個請求,包括待傳的文件的文件名,大小等,如果服務器接受,就開始傳文件。當然,文件傳輸的時候可以有兩種模式,ASCII碼和Bin,不過一般通用Bin 就可以了。基於上面的討論,本來用Delphi4的NMStrm,NMStrmServ 控件就可以完成,但是我測試過了,NMStrm控件對於較小的文件還可以使用,而且很方便,但是如果文件一大(1M)就會出錯。所以接下來我們利用Delphi中TServerSocket和TClIEntSocket寫這個程序由於以太包大小的限制以及DelphiSocket的處理機制(Delphi中,當你用一個Socket發送一個較大的Stream,接受方會激發多次OnRead事件,Delphi她只保證多次OnRead事件中每次數據的完整,而不會自己收集數據並返回給用戶。所以不要以為你把待傳文件在一個Socket中Send一次,另一個中Recv一次就可以了。你必須自己收集數據或自己定義協議。),所以我們采用自定義協議的方法。定義協議的規范方法是利用Record End。如:
  TMyFileProtocol=Record
  sSendType=(ST_QUERY,ST_REFUSE,ST_DATA,ST_ABORT,...);
  iLength:integer;
  bufSend:Buffer;
  End;
  我曾試過這個辦法,但失敗了,而且我一直認為我的方法是正確的,但程序一直編譯通不過,估計是Delphi有問題:) 所以我在下列的范例程序中利用另外一種辦法。Socket 類中有兩屬性ReceiveText和ReceiveBuf,在一個OnRead事件中,只能使用一次該兩屬性,所以我們可以利用一個全程變量來保存是該讀Text還是Buf,也就是說讀一次Text,再都一次Buf,這就模擬了TMyFileProtocol。
  
  
  開始程序:
  寫一個最簡單的,主要用於講解方法。
  定義協議:
  Const
  MP_QUERY ='1';
  MP_REFUSE ='2';
  MP_ACCEPT ='3';
  MP_NEXTWILLBEDATA='4';
  MP_DATA ='5';
  MP_ABORT ='6';
  MP_OVER ='7';
  MP_CHAT ='8';
  
  協議簡介:
  首先由ClIEnt發送MP_QUERY,Server接受到後發送MP_ACCEPT或MP_FEFUESE;
  ClIEnt接受到MP_ACCEPT發送MP_FILEPROPERTY,Server接受到後發送MP_NEXTWILLBEDATA;
  ClIEnt接受到發送MP_NEXTWILLBEDATA,Server接受到後發送MP_DATA;
  ClIEnt接受到MP_DATA,發送數據,Server接受數據,並發送MP_NEXTWILLBEDATA;
  循環,直到ClIEnt發送MP_OVER;
  中間可以互相發送MP_CHAT+String;
  
  Server程序:
  放上以下控件:SaveDialog1,btnStartServer,
  ss,(TServerSocket)
  
  btnStartServer.OnClick(Sender:TObject);
  begin
  ss.Port:=2000;
  ss.Open;
  end;
  
  ss.OnClIEntRead(Sender: TObject;Socket: TCustomWinSocket);
  var
  sTemp:string;
  bufRecv:Pointer;
  iRecvLength:integer;
  begin
  if bReadText then
  begin
  sTemp:=Socket.ReceiveText;
  case sTemp[1] of
  MP_QUERY:begin
  //在這裡拒絕
  SaveDialog1.FileName:=Copy(sTemp,2,Length(STemp));
  if SaveDialog1.Execute then
  begin
  Socket.SendText(MP_ACCEPT);
  fsRecv:=TFileStream.Create(SaveDialog1.FileName,fmCreate);
  end
  else Socket.SendText(MP_REFUSE+'去死');
  end;
  MP_FILEPROPERTY:begin
  //要發送StrToInt(Copy(sTemp,2,Length(sTemp))) 次
  //時間進度顯示。。。
  Socket.SendText(MP_NEXTWILLBEDATA);
  end;
  MP_NEXTWILLBEDATA:begin
  Socket.SendText(MP_DATA);
  bReadText:=false;
  end;
  MP_END:begin
  fsRecv.Free
  bReadText:=true;
  end;
  MP_ABORT:begin
  fsRecv.Free;
  bReadText:=true;
  end;
  MP_CHAT:begin
  //Chat Msg
  end;
  end;{of case}
  end
  else begin
  try
  GetMem(bufRecv,2000);//2000 must >iBYTESEND
  Socket.ReceiveBuf(bufRecv^,iRecvLength);
  fsRecv.WriteBuffer(bufRecv^,iRecvLength);
  finally
  FreeMem(bufRecv,2000);
  end;{of try}
  bReadText:=true;
  Socket.SendText(MP_NEXTWILLBEDATA);
  end;
  end;
  
  ClIEnt程序:
  放上以下控件:edtIPAddress,OpenDialog1,btnConnect,btnSendFile,
  cs. (TClIEntSocket)
  
  btnConnect.OnClick(Sender:TObject);
  begin
  cs.Address:=edtIPAddress.Text;
  cs.Port:=2000;
  cs.Connect;
  end;
  
  btnSendFile.OnClick(Sender:TObject);
  begin
  if OpenDialog1.Execute then
  Begin
  cs.Socket.SendText(MP_QUERY+OpenDialog1.FileName);//FileSize???
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved