服务器与客户

服务器与客户
服务器与客户

笔者在工作中遇到对局域网中各工作站与服务器之间进行Socket通信的问题。现在将本人总结出来的

TServerSocket和TClientSocket两个组件的基本用法写出来,希望与您分享。

ClientSocket组件为客户端组件。它是通信的请求方,也就是说,它是主动地与服务器端建立连接。

ServerSocket组件为服务器端组件。它是通信的响应方,也就是说,它的动作是监听以及被动接受客户端的连接请

求,并对请求进行回复。

ServerSocket组件可以同时接受一个或多个ClientSocket组件的连接请求,并与每个ClientSocket组件建立单独的

连接,进行单独的通信。因此,一个服务器端可以为多个客户端服务。

设计思路

本例包括一个服务器端程序和一个客户端程序。客户端程序可以放到多个计算机上运行,同时与服务器端进行连接

通信。

本例的重点,一是演示客户端与服务器端如何通信;二是当有多个客户端同时连接到服务器端时,服务器端如何识

别每个客户端,并对请求给出相应的回复。为了保证一个客户端断开连接时不影响其它客户端与服务器端的通信,

同时保证服务器端能够正确回复客户端的请求,在本例中声明了一个记录类型:

type

client_record=record

CHandle: integer; //客户端套接字句柄

CSocket:TCustomWinSocket; //客户端套接字

CName:string; //客户端计算机名称

CAddress:string; //客户端计算机IP地址

CUsed: boolean; //客户端联机标志

end;

利用这个记录类型数据保存客户端的信息,同时保存当前客户端的连接状态。其中,CHandle保存客户端套接字句

柄,以便准确定位每个与服务器端保持连接的客户端;Csocket保存客户端套接字,通过它可以对客户端进行回

复。Cused记录当前客户端是否与服务器端保持连接。

下面对组件ServerSocket和ClientSocket的属性设置简单说明。

ServerSocket的属性:

· Port,是通信的端口,必须设置。在本例中设置为1025;

· ServerTypt,服务器端读写信息类型,设置为stNonBlocking表示异步读写信息,本例中采用这种方式。

· ThreadCacheSize,客户端的最大连接数,就是服务器端最多允许多少客户端同时连接。本例采用默认值10。

其它属性采用默认设置即可。

ClientSocket的属性:

· Port,是通信的端口,必须与服务器端的设置相同。在本例中设置为1025;

· ClientType,客户端读写信息类型,应该与服务器端的设置相同,为stNonBlocking表示异步读写信息。

· Host,客户端要连接的服务器的IP地址。必须设置,当然也可以在代码中动态设置。

其它属性采用默认设置即可。

程序源代码:

· 服务器端源码(uServerMain.pas):

unit uServerMain;

interface

uses

Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, ScktComp, ToolWin, ComCtrls, ExtCtrls, StdCtrls, Buttons;

const

CMax=10; //客户端最大连接数

type

client_record=record

CHandle: integer; //客户端套接字句柄

CSocket:TCustomWinSocket; //客户端套接字

CName:string; //客户端计算机名称

CAddress:string; //客户端计算机IP地址

CUsed: boolean; //客户端联机标志

end;

type

TfrmServerMain = class(TForm)

ServerSocket: TServerSocket;

ControlBar1: TControlBar;

ToolBar1: TToolBar;

tbConnect: TToolButton;

tbClose: TToolButton;

tbDisconnected: TToolButton;

Edit1: TEdit;

Memo1: TMemo;

StatusBar: TStatusBar;

procedure tbConnectClick(Sender: TObject);

procedure tbDisconnectedClick(Sender: TObject);

procedure ServerSocketClientRead(Sender: TObject;

Socket: TCustomWinSocket);

procedure ServerSocketListen(Sender: TObject;

Socket: TCustomWinSocket);

procedure ServerSocketClientConnect(Sender: TObject;

Socket: TCustomWinSocket);

procedure ServerSocketClientDisconnect(Sender: TObject;

Socket: TCustomWinSocket);

procedure tbCloseClick(Sender: TObject);

procedure FormCreate(Sender: TObject);

procedure FormClose(Sender: TObject; var Action: TCloseAction); procedure ServerSocketGetSocket(Sender: TObject; Socket: Integer; var ClientSocket: TServerClientWinSocket);

procedure ServerSocketClientError(Sender: TObject;

Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;

var ErrorCode: Integer);

private

{ Private declarations }

public

{ Public declarations }

session: array[0..CMax] of client_record; //客户端连接数组Sessions: integer; //客户端连接数

end;

var

frmServerMain: TfrmServerMain;

implementation

{$R *.DFM}

//打开套接字连接,并使套接字进入监听状态

procedure TfrmServerMain.tbConnectClick(Sender: TObject);

begin

ServerSocket.Open ;

end;

//关闭套接字连接,不再监听客户端的请求

procedure TfrmServerMain.tbDisconnectedClick(Sender: TObject);

begin

ServerSocket.Close;

StatusBar.Panels[0].Text :=服务器套接字连接已经关闭,无法接受客户端的连接请求.;

end;

//从客户端读取信息

procedure TfrmServerMain.ServerSocketClientRead(Sender: TObject; Socket: TCustomWinSocket);

var

i:integer;

begin

//将从客户端读取的信息添加到Memo1中

Memo1.Lines.Add(Socket.ReceiveText);

for i:=0 to sessions do

begin

//取得匹配的客户端

if session[i].CHandle = Socket.SocketHandle then

begin

session[i].CSocket.SendText(回复客户端+session[i].CAddress+ ==>

+Edit1.Text);

end;

end;

end;

//服务器端套接字进入监听状态,以便监听客户端的连接

procedure TfrmServerMain.ServerSocketListen(Sender: TObject; Socket: TCustomWinSocket);

begin

StatusBar.Panels[0].Text :=等待客户端连接...;

end;

//当客户端连接到服务器端以后

procedure TfrmServerMain.ServerSocketClientConnect(Sender: TObject; Socket: TCustomWinSocket);

var

i,j:integer;

begin

j:=-1;

for i:=0 to sessions do

begin

//在原有的客户端连接数组中有中断的客户端连接

if not session[i].CUsed then

begin

session[i].CHandle := Socket.SocketHandle ;//客户端套接字句柄

session[i].CSocket := Socket; //客户端套接字

session[i].CName := Socket.RemoteHost ; //客户端计算机名称

session[i].CAddress := Socket.RemoteAddress ;//客户端计算机IP

session[i].CUsed := True; //连接数组当前位置已经占用

Break;

end;

j:=i;

end;

if j=sessions then

begin

inc(sessions);

session[j].CHandle := Socket.SocketHandle ;

session[j].CSocket := Socket;

session[j].CName := Socket.RemoteHost ;

session[j].CAddress := Socket.RemoteAddress ;

session[j].CUsed := True;

end;

StatusBar.Panels[0].Text := 客户端 +Socket.RemoteHost + 已经连接; end;

//当客户端断开连接时

procedure TfrmServerMain.ServerSocketClientDisconnect(Sender: TObject; Socket: TCustomWinSocket);

var

i:integer;

begin

for i:=0 to sessions do

begin

if session[i].CHandle =Socket.SocketHandle then

begin

session[i].CHandle :=0;

session[i].CUsed := False;

Break;

end;

end;

StatusBar.Panels[0].Text :=客户端 +Socket.RemoteHost + 已经断开; end;

//关闭窗口

procedure TfrmServerMain.tbCloseClick(Sender: TObject);

begin

Close;

end;

procedure TfrmServerMain.FormCreate(Sender: TObject);

begin

sessions := 0;

end;

procedure TfrmServerMain.FormClose(Sender: TObject;

var Action: TCloseAction);

begin

ServerSocket.Close ;

end;

//当客户端正在与服务器端连接时

procedure TfrmServerMain.ServerSocketGetSocket(Sender: TObject; Socket: Integer; var ClientSocket: TServerClientWinSocket);

begin

StatusBar.Panels[0].Text :=客户端正在连接...;

end;

//客户端发生错误

procedure TfrmServerMain.ServerSocketClientError(Sender: TObject; Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;

var ErrorCode: Integer);

begin

StatusBar.Panels[0].Text :=客户端+Socket.RemoteHost +发生错误!; ErrorCode := 0;

end;

end.

· 客户端源码(uClientMain.pas):

unit uClientMain;

interface

uses

Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, ScktComp, ComCtrls, ToolWin, ExtCtrls, StdCtrls, Buttons;

const

SocketHost = 172.16.1.6; //服务器端地址

type

TfrmClientMain = class(TForm)

ControlBar1: TControlBar;

ToolBar1: TToolBar;

tbConnected: TToolButton;

tbSend: TToolButton;

tbClose: TToolButton;

tbDisconnected: TToolButton;

ClientSocket: TClientSocket;

Edit1: TEdit;

Memo1: TMemo;

StatusBar: TStatusBar;

btnSend: TBitBtn;

procedure tbConnectedClick(Sender: TObject);

procedure tbDisconnectedClick(Sender: TObject);

procedure ClientSocketRead(Sender: TObject; Socket: TCustomWinSocket); procedure tbSendClick(Sender: TObject);

procedure tbCloseClick(Sender: TObject);

procedure FormShow(Sender: TObject);

procedure ClientSocketConnect(Sender: TObject;

Socket: TCustomWinSocket);

procedure ClientSocketConnecting(Sender: TObject;

Socket: TCustomWinSocket);

procedure ClientSocketDisconnect(Sender: TObject;

Socket: TCustomWinSocket);

procedure FormClose(Sender: TObject; var Action: TCloseAction); procedure ClientSocketError(Sender: TObject; Socket: TCustomWinSocket; ErrorEvent: TErrorEvent; var ErrorCode: Integer);

private

{ Private declarations }

public

{ Public declarations }

end;

var

frmClientMain: TfrmClientMain;

implementation

{$R *.DFM}

//打开套接字连接

procedure TfrmClientMain.tbConnectedClick(Sender: TObject);

begin

ClientSocket.Open ;

end;

//关闭套接字连接

procedure TfrmClientMain.tbDisconnectedClick(Sender: TObject);

begin

ClientSocket.Close;

end;

//接受服务器端的回复

procedure TfrmClientMain.ClientSocketRead(Sender: TObject;

Socket: TCustomWinSocket);

begin

Memo1.Lines.Add(Socket.ReceiveText);

end;

//发送信息到服务器端

procedure TfrmClientMain.tbSendClick(Sender: TObject);

begin

ClientSocket.Socket.SendText(Edit1.Text);

end;

procedure TfrmClientMain.tbCloseClick(Sender: TObject);

begin

Close;

end;

//设置要连接的服务器端地址

procedure TfrmClientMain.FormShow(Sender: TObject);

begin

ClientSocket.Host := SocketHost;

end;

//已经连接到服务器端

procedure TfrmClientMain.ClientSocketConnect(Sender: TObject; Socket: TCustomWinSocket);

begin

tbSend.Enabled := True;

tbDisconnected.Enabled :=True;

btnSend.Enabled := True;

StatusBar.Panels[0].Text := 已经连接到 + Socket.RemoteHost ;

end;

//正在连接到服务器端

procedure TfrmClientMain.ClientSocketConnecting(Sender: TObject; Socket: TCustomWinSocket);

begin

StatusBar.Panels[0].Text := 正在连接到服务器... ;

end;

//当断开与服务器端的连接时发生

procedure TfrmClientMain.ClientSocketDisconnect(Sender: TObject; Socket: TCustomWinSocket);

begin

tbSend.Enabled := False;

btnSend.Enabled := False;

tbDisconnected.Enabled := False;

StatusBar.Panels[0].Text := 已经断开与 + Socket.RemoteHost + 的连接; end;

procedure TfrmClientMain.FormClose(Sender: TObject;

var Action: TCloseAction);

begin

ClientSocket.Close ;

end;

//当与服务器端的连接发生错误时

procedure TfrmClientMain.ClientSocketError(Sender: TObject;

Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;

var ErrorCode: Integer);

begin

StatusBar.Panels[0].Text := 与服务器端的连接发生错误;

ErrorCode := 0;

end;

end.

小结

上述方法是比较简单的实现方法,同时也是相对较容易理解的方法。通过这个方法,笔者成功实现了局域网内多个

客户端与服务器端进行Socket通信的功能,同时可以保证一个客户端的连接、通信或是断开都不影响其它客户端的

正常通信。

附录:

服务器端窗体和客户端窗体及组件的属性设置参加相应的DFM文件。

uServerMain.pas对应的DFM文件(uServerMain.dfm)

object frmServerMain: TfrmServerMain

Left = 297

Top = 258

BorderIcons = [biSystemMenu, biMinimize]

BorderStyle = bsSingle

Caption = ServerSocket

ClientHeight = 279

ClientWidth = 476

Color = clBtnFace

Font.Charset = DEFAULT_CHARSET

Font.Color = clWindowText

Font.Height = -11

https://www.360docs.net/doc/6110986110.html, = MS Sans Serif

Font.Style = []

OldCreateOrder = False

OnClose = FormClose

OnCreate = FormCreate

PixelsPerInch = 96

TextHeight = 13

object ControlBar1: TControlBar

Left = 0

Top = 0

Width = 476

Height = 30

Align = alTop

AutoSize = True

TabOrder = 0

object ToolBar1: TToolBar

Left = 11

Top = 2

Width = 459

Height = 22

ButtonHeight = 21

ButtonWidth = 55

Caption = ToolBar1

EdgeInner = esNone

EdgeOuter = esNone

Flat = True

ShowCaptions = True

TabOrder = 0

object tbConnect: TToolButton

Left = 0

Top = 0

Caption = 连接

ImageIndex = 0

OnClick = tbConnectClick

end

object tbDisconnected: TToolButton Left = 55

Top = 0

Caption = 断开

ImageIndex = 4

OnClick = tbDisconnectedClick

end

object tbClose: TToolButton

Left = 110

Top = 0

Caption = 关闭

ImageIndex = 3

OnClick = tbCloseClick

end

end

end

object Edit1: TEdit

Left = 0

Top = 232

Width = 473

Height = 21

TabOrder = 1

Text = 你好!

end

object Memo1: TMemo

Left = 0

Top = 30

Width = 476

Height = 195

Align = alTop

TabOrder = 2

end

object StatusBar: TStatusBar

Left = 0

Top = 257

Width = 476

Height = 22

Panels = <

item

Width = 50

end>

SimplePanel = False

end

object ServerSocket: TServerSocket

Active = False

Port = 1025

ServerType = stNonBlocking

OnListen = ServerSocketListen

OnGetSocket = ServerSocketGetSocket OnClientConnect = ServerSocketClientConnect OnClientDisconnect = ServerSocketClientDisconnect OnClientRead = ServerSocketClientRead OnClientError = ServerSocketClientError

Left = 368

end

end

uClientMain.pas对应的DFM文件(uClientMain.dfm) object frmClientMain: TfrmClientMain

Left = 361

Top = 290

BorderIcons = [biSystemMenu, biMinimize] BorderStyle = bsSingle

Caption = ClientSocket

ClientHeight = 230

ClientWidth = 402

Color = clBtnFace

Font.Charset = DEFAULT_CHARSET Font.Color = clWindowText Font.Height = -11

https://www.360docs.net/doc/6110986110.html, = MS Sans Serif Font.Style = []

OldCreateOrder = False

Position = poScreenCenter OnClose = FormClose

OnShow = FormShow

PixelsPerInch = 96

TextHeight = 13

object ControlBar1: TControlBar Left = 0

Top = 0

Width = 402

Height = 30

Align = alTop

AutoSize = True

TabOrder = 0

object ToolBar1: TToolBar

Left = 11

Top = 2

Width = 385

Height = 22

ButtonHeight = 21

ButtonWidth = 55

Caption = ToolBar1

EdgeInner = esNone

EdgeOuter = esNone

Flat = True

ShowCaptions = True

TabOrder = 0

object tbConnected: TToolButton Left = 0

Top = 0

Caption = 连接

ImageIndex = 0

OnClick = tbConnectedClick

end

object tbSend: TToolButton

Left = 55

Top = 0

Caption = 发送

Enabled = False

ImageIndex = 1

OnClick = tbSendClick

end

object tbDisconnected: TToolButton Left = 110

Top = 0

Caption = 断开

Enabled = False

ImageIndex = 3

OnClick = tbDisconnectedClick

end

object tbClose: TToolButton

Left = 165

Top = 0

Caption = 退出

ImageIndex = 2

OnClick = tbCloseClick

end

end

end

object Edit1: TEdit

Left = 0

Top = 184

Width = 321

Height = 21

TabOrder = 1

Text = 问候

end

object Memo1: TMemo

Left = 0

Top = 30

Width = 402

Height = 147

Align = alTop

TabOrder = 2

end

object StatusBar: TStatusBar

Left = 0

Top = 208

Width = 402

Height = 22

Panels = <

item

Width = 50

end>

SimplePanel = False

end

object btnSend: TBitBtn

Left = 336

Top = 183

Width = 60

Height = 22

Caption = 发送

Enabled = False

TabOrder = 4

OnClick = tbSendClick

end

object ClientSocket: TClientSocket Active = False

ClientType = ctNonBlocking

Port = 1025

OnConnecting = ClientSocketConnecting OnConnect = ClientSocketConnect OnDisconnect = ClientSocketDisconnect OnRead = ClientSocketRead

OnError = ClientSocketError

Left = 320

end

end

相关主题
相关文档
最新文档