<?xml version="1.0" encoding="UTF-8" ?>
<rss version="2.0">
<channel>
<title><![CDATA[Heck's  Blog]]></title> 
<link>https://www.heckjj.com/index.php</link> 
<description><![CDATA[一瞬间的决定，往往可以改变很多，事实上，让自己成功的往往不是知识，是精神！ 如果你总是为自己找借口，那只好让成功推迟。执行力，今天！]]></description> 
<language>zh-cn</language> 
<copyright><![CDATA[Heck's  Blog]]></copyright>
<item>
<link>https://www.heckjj.com/delphi-design-proxy-services-html/</link>
<title><![CDATA[用Delphi设计实现代理服务器]]></title> 
<author>Heck &lt;@hecks.tk&gt;</author>
<category><![CDATA[编程杂谈]]></category>
<pubDate>Wed, 27 Oct 2010 03:06:51 +0000</pubDate> 
<guid>https://www.heckjj.com/delphi-design-proxy-services-html/</guid> 
<description>
<![CDATA[ 
	&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; <span style="font-family: 微软雅黑;">在编写一个上网计费软件时，涉及到如何对局域网中各工作站上网计费问题。一般来讲，这些工作站通过代理服务器上网，而采用现成的代理服务器软件时，由于代理服务器软件是封闭的系统，很难编写程序获取实时的上网计时信息。因此，考虑是否能编写自己的代理服务器，一方面解决群体上网，另一方面又解决上网的计费问题呢？经过实验性编程，终于圆满地解决了该问题。现写出来，与各位同行分享。<br/> <br/>1、 思路<br/>当前流行的浏览器的系统选项中有一个参数，即“通过代理服务器连接”，经过编程测试，当局域网中一台工作站指定了该属性，再发出Internet请求时，请求数据将发送到所指定的代理服务器上，以下为请求数据包示例：<br/>&nbsp;&nbsp;GET http://home.microsoft.com/intl/cn/ HTTP/1.0<br/>&nbsp;&nbsp;Accept: */*<br/>&nbsp;&nbsp;Accept-Language: zh-cn<br/>&nbsp;&nbsp;Accept-Encoding: gzip, deflate<br/>&nbsp;&nbsp;User-Agent: Mozilla/4.0 (compatible; MSIE 5.0; Windows NT)<br/>&nbsp;&nbsp;Host: home.microsoft.com<br/>&nbsp;&nbsp;Proxy-Connection: Keep-Alive<br/>其中第一行为目标URL及相关方法、协议，“Host”行指定了目标主机的地址。由此知道了代理服务的过程：接收被代理端的请求、连接真正的主机、接收主机返回的数据、将接收数据发送到被代理端。为此可编写一个简单的程序，完成上述网络通信重定向问题。用Delphi设计时，选用ServerSocket作为与被代理工作站通信的套接字控件，选用ClientSocket动态数组作为与远程主机通信的套接字控件。编程时应解决的一个重要问题是多重连接处理问题，为了加快代理服务的速度和被代理端的响应速度，套接字控件的属性应设为非阻塞型；各通信会话与套接字动态绑定，用套接字的SocketHandle属性值确定属于哪一个会话。</span><span style="font-family: 微软雅黑;"><br/>通信的衔接过程如下图所示：<br/>------------------------------------------------------------------------------<br/> <br/>&nbsp;&nbsp;代理服务器<br/> <br/>&nbsp;&nbsp;Serversocket<br/>&nbsp;&nbsp;(1) 接 收<br/>&nbsp;&nbsp;被代理端 发 送 远程主机<br/>&nbsp;&nbsp;(6) (2) (5)<br/>&nbsp;&nbsp;Browser ClientSocket (4) Web<br/>Server<br/>&nbsp;&nbsp;接 收<br/>&nbsp;&nbsp;发 送 (3)<br/>------------------------------------------------------------------------------<br/> <br/>&nbsp;&nbsp;示意图<br/> <br/> <br/>（1）、被代理端浏览器发出Web请求，代理服务器的Serversocket接收到请求。<br/>（2）、代理服务器程序自动创建一个ClientSocket，并设置主机地址、端口等属性，然后连接远程主机。<br/>（3）、远程连通后激发发送事件，将Serversocket接收到的Web请求数据包发送到远程主机。<br/>（4）、当远程主机返回页面数据时，激发ClientSocket的读事件，读取页面数据。<br/>（5）、代理服务器程序根据绑定信息确定属于ServerSocket控件中的哪一个Socket应该将从主机接收的页面信息发送到被代理端。<br/>（6）、ServerSocket中的对应Socket将页面数据发送到被代理端。<br/> <br/>2、 程序编写<br/>使用Delphi设计以上通信过程非常简单，主要是ServerSocket、ClientSocket的相关事件驱动程序的程序编写。下面给出作者编写的实验用代理服务器界面与源程序清单，内含简要功能说明：<br/> <br/>unit main;<br/> <br/>interface<br/> <br/>uses<br/>&nbsp;&nbsp;Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,<br/>&nbsp;&nbsp;ExtCtrls, ScktComp, TrayIcon, Menus, StdCtrls;<br/> <br/>type<br/>&nbsp;&nbsp;session_record=record<br/>&nbsp;&nbsp;Used: boolean; &#123;会话记录是否可用&#125;<br/>&nbsp;&nbsp;SS_Handle: integer; &#123;代理服务器套接字句柄&#125;<br/>&nbsp;&nbsp;CSocket: TClientSocket; &#123;用于连接远程的套接字&#125;<br/>&nbsp;&nbsp;Lookingup: boolean; &#123;是否正在查找服务器&#125;<br/>&nbsp;&nbsp;LookupTime: integer; &#123;查找服务器时间&#125;<br/>&nbsp;&nbsp;Request: boolean; &#123;是否有请求&#125;<br/>&nbsp;&nbsp;request_str: string; &#123;请求数据块&#125;<br/>&nbsp;&nbsp;client_connected: boolean; &#123;客户机联机标志&#125;<br/>&nbsp;&nbsp;remote_connected: boolean; &#123;远程服务器连接标志&#125;<br/>end;<br/> <br/>type<br/>&nbsp;&nbsp;TForm1 = class(TForm)<br/>&nbsp;&nbsp;ServerSocket1: TServerSocket;<br/>&nbsp;&nbsp;ClientSocket1: TClientSocket;<br/>&nbsp;&nbsp;Timer2: TTimer;<br/>&nbsp;&nbsp;TrayIcon1: TTrayIcon;<br/>&nbsp;&nbsp;PopupMenu1: TPopupMenu;<br/>&nbsp;&nbsp;N11: TMenuItem;<br/>&nbsp;&nbsp;N21: TMenuItem;<br/>&nbsp;&nbsp;N1: TMenuItem;<br/>&nbsp;&nbsp;N01: TMenuItem;<br/>&nbsp;&nbsp;Memo1: TMemo;<br/>&nbsp;&nbsp;Edit1: TEdit;<br/>&nbsp;&nbsp;Label1: TLabel;<br/>&nbsp;&nbsp;Timer1: TTimer;<br/>&nbsp;&nbsp;procedure Timer2Timer(Sender: TObject);<br/>&nbsp;&nbsp;procedure N11Click(Sender: TObject);<br/>&nbsp;&nbsp;procedure FormCreate(Sender: TObject);<br/>&nbsp;&nbsp;procedure FormClose(Sender: TObject; var Action: TCloseAction);<br/>&nbsp;&nbsp;procedure N21Click(Sender: TObject);<br/>&nbsp;&nbsp;procedure N01Click(Sender: TObject);<br/>&nbsp;&nbsp;procedure ServerSocket1ClientConnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ServerSocket1ClientDisconnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ServerSocket1ClientError(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;<br/>&nbsp;&nbsp;var ErrorCode: Integer);<br/>&nbsp;&nbsp;procedure ServerSocket1ClientRead(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ClientSocket1Connect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ClientSocket1Disconnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ClientSocket1Error(Sender: TObject; Socket: TCustomWinSocket;<br/>&nbsp;&nbsp;ErrorEvent: TErrorEvent; var ErrorCode: Integer);<br/>&nbsp;&nbsp;procedure ClientSocket1Write(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ClientSocket1Read(Sender: TObject; Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure ServerSocket1Listen(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>&nbsp;&nbsp;procedure AppException(Sender: TObject; E: Exception);<br/>&nbsp;&nbsp;procedure Timer1Timer(Sender: TObject);<br/>&nbsp;&nbsp;private<br/>&nbsp;&nbsp;&#123; Private declarations &#125;<br/>&nbsp;&nbsp;public<br/>&nbsp;&nbsp;Service_Enabled: boolean; &#123;代理服务是否开启&#125;<br/>&nbsp;&nbsp;session: array of session_record; &#123;会话数组&#125;<br/>&nbsp;&nbsp;sessions: integer; &#123;会话数&#125;<br/>&nbsp;&nbsp;LookUpTimeOut: integer; &#123;连接超时值&#125;<br/>&nbsp;&nbsp;InvalidRequests: integer; &#123;无效请求数&#125;<br/>&nbsp;&nbsp;end;<br/> <br/>var<br/>&nbsp;&nbsp;Form1: TForm1;<br/> <br/>implementation<br/> <br/>&#123;$R *.DFM&#125;<br/> <br/>//系统启动定时器，启动窗显示完成后，缩小到System Tray…<br/>procedure TForm1.Timer2Timer(Sender: TObject);<br/>begin<br/>&nbsp;&nbsp;timer2.Enabled:=false; &#123;关闭定时器&#125;<br/>&nbsp;&nbsp;sessions:=0; &#123;会话数=0&#125;<br/>&nbsp;&nbsp;Application.OnException := AppException; &#123;为了屏蔽代理服务器出现的异常<br/>&#125;<br/>&nbsp;&nbsp;invalidRequests:=0; &#123;0错误&#125;<br/>&nbsp;&nbsp;LookUpTimeOut:=60000; &#123;超时值=1分钟&#125;<br/>&nbsp;&nbsp;timer1.Enabled:=true; &#123;打开定时器&#125;<br/>&nbsp;&nbsp;n11.Enabled:=false; &#123;开启服务菜单项失效&#125;<br/>&nbsp;&nbsp;n21.Enabled:=true; &#123;关闭服务菜单项有效&#125;<br/>&nbsp;&nbsp;serversocket1.Port:=988; &#123;代理服务器端口=988&#125;<br/>&nbsp;&nbsp;serversocket1.Active:=true; &#123;开启服务&#125;<br/>&nbsp;&nbsp;form1.hide; &#123;隐藏界面，缩小到System Tray上&#125;<br/>end;<br/> <br/>//开启服务菜单项…<br/>procedure TForm1.N11Click(Sender: TObject);<br/>begin<br/>&nbsp;&nbsp;serversocket1.Active:=true; &#123;开启服务&#125;<br/>end;<br/> <br/> <br/>//停止服务菜单项…<br/>procedure TForm1.N21Click(Sender: TObject);<br/>begin<br/>&nbsp;&nbsp;serversocket1.Active:=false; &#123;停止服务&#125;<br/>&nbsp;&nbsp;N11.Enabled:=True;<br/>&nbsp;&nbsp;N21.Enabled:=False;<br/>&nbsp;&nbsp;Service_Enabled:=false; &#123;标志清零&#125;<br/>end;<br/> <br/> <br/>//主窗口建立…<br/>procedure TForm1.FormCreate(Sender: TObject);<br/>begin<br/>&nbsp;&nbsp;Service_Enabled:=false;<br/>&nbsp;&nbsp;timer2.Enabled:=true; &#123;窗口建立时，打开定时器&#125;<br/>end;<br/> <br/>//窗口关闭时…<br/>procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);<br/>begin<br/>&nbsp;&nbsp;timer1.Enabled:=false; &#123;关闭定时器&#125;<br/>&nbsp;&nbsp;if Service_Enabled then<br/>&nbsp;&nbsp;serversocket1.Active:=false; &#123;退出程序时关闭服务&#125;<br/>end;<br/> <br/>//退出程序按钮…<br/>procedure TForm1.N01Click(Sender: TObject);<br/>begin<br/>&nbsp;&nbsp;form1.Close; &#123;退出程序&#125;<br/>end;<br/> <br/>//开启代理服务后…<br/>procedure TForm1.ServerSocket1Listen(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>begin<br/>&nbsp;&nbsp;Service_Enabled:=true; &#123;置正在服务标志&#125;<br/>&nbsp;&nbsp;N11.Enabled:=false;<br/>&nbsp;&nbsp;N21.Enabled:=true;<br/>end;<br/> <br/>//被代理端连接到代理服务器后，建立一个会话，并与套接字绑定…<br/>procedure TForm1.ServerSocket1ClientConnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i,j: integer;<br/>begin<br/>&nbsp;&nbsp;j:=-1;<br/>&nbsp;&nbsp;for i:=1 to sessions do &#123;查找是否有空白项&#125;<br/>&nbsp;&nbsp;if not session[i-1].Used and not session[i-1].CSocket.active then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;j:=i-1; &#123;有，分配它&#125;<br/>&nbsp;&nbsp;session[j].Used:=true; &#123;置为在用&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;if not session[i-1].Used and session[i-1].CSocket.active then<br/>&nbsp;&nbsp;session[i-1].CSocket.active:=false;<br/>&nbsp;&nbsp;if j=-1 then<br/>&nbsp;&nbsp;begin &#123;无，新增一个&#125;<br/>&nbsp;&nbsp;j:=sessions;<br/>&nbsp;&nbsp;inc(sessions);<br/>&nbsp;&nbsp;setlength(session,sessions);<br/>&nbsp;&nbsp;session[j].Used:=true; &#123;置为在用&#125;<br/>&nbsp;&nbsp;session[j].CSocket:=TClientSocket.Create(nil);<br/>&nbsp;&nbsp;session[j].CSocket.OnConnect:=ClientSocket1Connect;<br/>&nbsp;&nbsp;session[j].CSocket.OnDisconnect:=ClientSocket1Disconnect;<br/>&nbsp;&nbsp;session[j].CSocket.OnError:=ClientSocket1Error;<br/>&nbsp;&nbsp;session[j].CSocket.OnRead:=ClientSocket1Read;<br/>&nbsp;&nbsp;session[j].CSocket.OnWrite:=ClientSocket1Write;<br/>&nbsp;&nbsp;session[j].Lookingup:=false;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;session[j].SS_Handle:=socket.socketHandle; &#123;保存句柄，实现绑定&#125;<br/>&nbsp;&nbsp;session[j].Request:=false; &#123;无请求&#125;<br/>&nbsp;&nbsp;session[j].client_connected:=true; &#123;客户机已连接&#125;<br/>&nbsp;&nbsp;session[j].remote_connected:=false; &#123;远程未连接&#125;<br/>&nbsp;&nbsp;edit1.text:=inttostr(sessions);<br/>end;<br/> <br/>//被代理端断开时…<br/>procedure TForm1.ServerSocket1ClientDisconnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i,j,k: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].SS_Handle=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].client_connected:=false; &#123;客户机未连接&#125;<br/>&nbsp;&nbsp;if session[i-1].remote_connected then<br/>&nbsp;&nbsp;session[i-1].CSocket.active:=false &#123;假如远程尚连接，断开它&#125;<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;session[i-1].Used:=false; &#123;假如两者都断开，则置释放资<br/>源标志&#125;<br/>源标志&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;j:=sessions;<br/>&nbsp;&nbsp;k:=0;<br/>&nbsp;&nbsp;for i:=1 to j do &#123;统计会话数组尾部有几个未用项&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;if session[j-i].Used then<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;inc(k);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;if k&gt;0 then &#123;修正会话数组，释放尾部未用项&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;sessions:=sessions-k;<br/>&nbsp;&nbsp;setlength(session,sessions);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;edit1.text:=inttostr(sessions);<br/>end;<br/> <br/>//通信错误出现时…<br/>procedure TForm1.ServerSocket1ClientError(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;<br/>&nbsp;&nbsp;var ErrorCode: Integer);<br/>var<br/>i,j,k: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].SS_Handle=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].client_connected:=false; &#123;客户机未连接&#125;<br/>&nbsp;&nbsp;if session[i-1].remote_connected then<br/>&nbsp;&nbsp;session[i-1].CSocket.active:=false &#123;假如远程尚连接，断开它&#125;<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;session[i-1].Used:=false; &#123;假如两者都断开，则置释放资<br/>源标志&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;j:=sessions;<br/>&nbsp;&nbsp;k:=0;<br/>&nbsp;&nbsp;for i:=1 to j do<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;if session[j-i].Used then<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;inc(k);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;if k&gt;0 then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;sessions:=sessions-k;<br/>&nbsp;&nbsp;setlength(session,sessions);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;edit1.text:=inttostr(sessions);<br/>&nbsp;&nbsp;errorcode:=0;<br/>end;<br/> <br/>//被代理端发送来页面请求时…<br/>procedure TForm1.ServerSocket1ClientRead(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>tmp,line,host: string;<br/>i,j,port: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do &#123;判断是哪一个会话&#125;<br/>&nbsp;&nbsp;if session[i-1].Used and (session[i-1].SS_Handle=socket.sockethandle)<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].request_str:=socket.ReceiveText; &#123;保存请求数据&#125;<br/>&nbsp;&nbsp;tmp:=session[i-1].request_str; &#123;存放到临时变量&#125;<br/>&nbsp;&nbsp;memo1.lines.add(tmp);<br/>&nbsp;&nbsp;j:=pos(char(13)+char(10),tmp); &#123;一行标志&#125;<br/>&nbsp;&nbsp;while j&gt;0 do &#123;逐行扫描请求文本，查找主机地<br/>址&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;line:=copy(tmp,1,j-1); &#123;取一行&#125;<br/>&nbsp;&nbsp;delete(tmp,1,j+1); &#123;删除一行&#125;<br/>&nbsp;&nbsp;j:=pos(&#039;Host&#039;,line); &#123;主机地址标志&#125;<br/>&nbsp;&nbsp;if j&gt;0 then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;delete(line,1,j+5); &#123;删除前面的无效字符<br/>&#125;<br/>&nbsp;&nbsp;j:=pos(&#039;:&#039;,line);<br/>&nbsp;&nbsp;if j&gt;0 then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;host:=copy(line,1,j-1);<br/>&nbsp;&nbsp;delete(line,1,j);<br/>&nbsp;&nbsp;try<br/>&nbsp;&nbsp;port:=strtoint(line);<br/>&nbsp;&nbsp;except<br/>&nbsp;&nbsp;port:=80;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;end<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;host:=trim(line); &#123;获取主机地址<br/>&#125;<br/>&nbsp;&nbsp;port:=80;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;if not session[i-1].remote_connected then &#123;假如远征<br/>尚未连接&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].Request:=true; &#123;置请求数据就<br/>绪标志&#125;<br/>&nbsp;&nbsp;session[i-1].CSocket.host:=host; &#123;设置远程主机<br/>地址&#125;<br/>&nbsp;&nbsp;session[i-1].CSocket.port:=port; &#123;设置端口<br/>&#125;<br/>&nbsp;&nbsp;session[i-1].CSocket.active:=true; &#123;连接远程<br/>主机&#125;<br/>&nbsp;&nbsp;session[i-1].Lookingup:=true; &#123;置标志&#125;<br/>&nbsp;&nbsp;session[i-1].LookupTime:=0; &#123;从0开始计<br/>时&#125;<br/>&nbsp;&nbsp;end<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;&#123;假如远程已连接，直接发送请求&#125;<br/>&nbsp;&nbsp;session[i-1].CSocket.socket.sendtext(session[i-1].<br/>request_str);<br/>&nbsp;&nbsp;break; &#123;停止扫描请求文本&#125;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;j:=pos(char(13)+char(10),tmp); &#123;指向下一行&#125;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;break; &#123;停止循环&#125;<br/>&nbsp;&nbsp;end;<br/>end;<br/> <br/>//当连接远程主机成功时…<br/>procedure TForm1.ClientSocket1Connect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].CSocket.socket.sockethandle=socket.SocketHandle) and<br/>session[i-1].Used then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].CSocket.tag:=socket.SocketHandle;<br/>&nbsp;&nbsp;session[i-1].remote_connected:=true; &#123;置远程主机已连通标志&#125;<br/>&nbsp;&nbsp;session[i-1].Lookingup:=false; &#123;清标志&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>end;<br/> <br/> <br/>//当远程主机断开时…<br/>procedure TForm1.ClientSocket1Disconnect(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i,j,k: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].CSocket.tag=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].remote_connected:=false; &#123;置为未连接&#125;<br/>&nbsp;&nbsp;if not session[i-1].client_connected then<br/>&nbsp;&nbsp;session[i-1].Used:=false &#123;假如客户机已断开，则置释放资源<br/>标志&#125;<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;for k:=1 to serversocket1.Socket.ActiveConnections do<br/>&nbsp;&nbsp;if (serversocket1.Socket.Connections[k-1].SocketHandle=sessi<br/>on[i-1].SS_Handle) and session[i-1].used then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;serversocket1.Socket.Connections[k-1].Close;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;j:=sessions;<br/>&nbsp;&nbsp;k:=0;<br/>&nbsp;&nbsp;for i:=1 to j do<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;if session[j-i].Used then<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;inc(k);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;if k&gt;0 then &#123;修正会话数组&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;sessions:=sessions-k;<br/>&nbsp;&nbsp;setlength(session,sessions);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;edit1.text:=inttostr(sessions);<br/>end;<br/> <br/>//当与远程主机通信发生错误时…<br/>procedure TForm1.ClientSocket1Error(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket; ErrorEvent: TErrorEvent;<br/>&nbsp;&nbsp;var ErrorCode: Integer);<br/>var<br/>i,j,k: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].CSocket.tag=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;socket.close;<br/>&nbsp;&nbsp;session[i-1].remote_connected:=false; &#123;置为未连接&#125;<br/>&nbsp;&nbsp;if not session[i-1].client_connected then<br/>&nbsp;&nbsp;session[i-1].Used:=false &#123;假如客户机已断开，则置释放资源<br/>标志&#125;<br/>&nbsp;&nbsp;else<br/>&nbsp;&nbsp;for k:=1 to serversocket1.Socket.ActiveConnections do<br/>&nbsp;&nbsp;if (serversocket1.Socket.Connections[k-1].SocketHandle=sessi<br/>on[i-1].SS_Handle) and session[i-1].used then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;serversocket1.Socket.Connections[k-1].Close;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;j:=sessions;<br/>&nbsp;&nbsp;k:=0;<br/>&nbsp;&nbsp;for i:=1 to j do<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;if session[j-i].Used then<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;inc(k);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;errorcode:=0;<br/>&nbsp;&nbsp;if k&gt;0 then &#123;修正会话数组&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;sessions:=sessions-k;<br/>&nbsp;&nbsp;setlength(session,sessions);<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;edit1.text:=inttostr(sessions);<br/>end;<br/> <br/>//向远程主机发送页面请求…<br/>procedure TForm1.ClientSocket1Write(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].CSocket.tag=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;if session[i-1].Request then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;socket.SendText(session[i-1].request_str); &#123;假如有请求，发<br/>送&#125;<br/>&nbsp;&nbsp;session[i-1].Request:=false; &#123;清标志&#125;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>end;<br/> <br/>//远程主机发来页面数据时…<br/>procedure TForm1.ClientSocket1Read(Sender: TObject;<br/>&nbsp;&nbsp;Socket: TCustomWinSocket);<br/>var<br/>i,j: integer;<br/>rec_bytes: integer; &#123;传回的数据块长度&#125;<br/>rec_Buffer: array[0..2047] of char; &#123;传回的数据块缓冲区&#125;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if (session[i-1].CSocket.tag=socket.SocketHandle) and session[i-1].Used<br/>then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;rec_bytes:=socket.ReceiveBuf(rec_buffer,2048); &#123;接收数据&#125;<br/>&nbsp;&nbsp;for j:=1 to serversocket1.Socket.ActiveConnections do<br/>&nbsp;&nbsp;if serversocket1.Socket.Connections[j-1].SocketHandle=session[i<br/>-1].SS_Handle then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;serversocket1.Socket.Connections[j-1].SendBuf(rec_buffer,<br/>rec_bytes); &#123;发送数据&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>end;<br/> <br/>//“页面找不到”等错误信息出现时…<br/>procedure TForm1.AppException(Sender: TObject; E: Exception);<br/>begin<br/>&nbsp;&nbsp;inc(invalidrequests);<br/>end;<br/> <br/>//查找远程主机定时…<br/>procedure TForm1.Timer1Timer(Sender: TObject);<br/>var<br/>i,j: integer;<br/>begin<br/>&nbsp;&nbsp;for i:=1 to sessions do<br/>&nbsp;&nbsp;if session[i-1].Used and session[i-1].Lookingup then &#123;假如正在连接&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;inc(session[i-1].LookupTime);<br/>&nbsp;&nbsp;if session[i-1].LookupTime&gt;lookuptimeout then &#123;假如超时&#125;<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;session[i-1].Lookingup:=false;<br/>&nbsp;&nbsp;session[i-1].CSocket.active:=false; &#123;停止查找&#125;<br/>&nbsp;&nbsp;for j:=1 to serversocket1.Socket.ActiveConnections do<br/>&nbsp;&nbsp;if serversocket1.Socket.Connections[j-1].SocketHandle=ses<br/>sion[i-1].SS_Handle then<br/>&nbsp;&nbsp;begin<br/>&nbsp;&nbsp;serversocket1.Socket.Connections[j-1].Close; &#123;断开<br/>客户机&#125;<br/>&nbsp;&nbsp;break;<br/>&nbsp;&nbsp;end;<br/>&nbsp;&nbsp;end;<br/> <br/>&nbsp;&nbsp;end;<br/>end;<br/>end.<br/> <br/>3、 后记<br/>由于这种设计思路仅仅在被代理端和远程主机之间增加了一个重定向功能，被代理端原有的缓存技术等特点均保留，因此效率较高。经过测试，利用1个33.6K的Modem上网时，三到十个被代理工作站同时上网，仍有较好的响应速度。由于被代理工作站和代理服务器工作站之间的连接一般通过高速链路，因此瓶颈主要出现在代理服务器的上网方式上。通过上述方法，作者成功开发了一套完善的代理服务器软件并与机房计费系统完全集成，实现了利用一台工作站完成上网代理、上网计费、用机计费等功能。<br/>&nbsp;&nbsp;有编程经验的朋友完全可以另行增加代理服务器功能，如设定禁止访问站点、统计客户流量、Web访问列表等等。</span><br/>Tags - <a href="https://www.heckjj.com/tags/delphi/" rel="tag">delphi</a> , <a href="https://www.heckjj.com/tags/%25E5%25AE%259E%25E7%258E%25B0/" rel="tag">实现</a> , <a href="https://www.heckjj.com/tags/%25E4%25BB%25A3%25E7%2590%2586%25E6%259C%258D%25E5%258A%25A1%25E5%2599%25A8/" rel="tag">代理服务器</a>
]]>
</description>
</item><item>
<link>https://www.heckjj.com/delphi-design-proxy-services-html/#blogcomment</link>
<title><![CDATA[[评论] 用Delphi设计实现代理服务器]]></title> 
<author> &lt;user@domain.com&gt;</author>
<category><![CDATA[评论]]></category>
<pubDate>Thu, 01 Jan 1970 00:00:00 +0000</pubDate> 
<guid>https://www.heckjj.com/delphi-design-proxy-services-html/#blogcomment</guid> 
<description>
<![CDATA[ 
	
]]>
</description>
</item>
</channel>
</rss>