读一读Scktsrvr.exe的源程序

使用DELPHI做多层开发的朋友们都应该对Scktsrvr.exe这个程序不陌生的,
Borland公司在DELPHI中给出了它的源代码。
这是一个900来行的程序,程序不算长,
现在我只选其中部分仔细读一读。
走的线路大致是,从服务器接到客户端连接,处理客户端的一个请求(这儿
选了客户端向服务器发出的‘取应用服务器列表‘请求)

服务器接受了客户端连接后,
因为ServerSocket采用的是阻塞模式,服务器执行了下面这个线程来
服务客户端:

//SCKTMAIN.PAS

PRocedure TSocketDispatcherThread.ClientExecute;
var
  Data: IDataBlock;
  msg: TMsg;
  Obj: ISendDataBlock;
  Event: THandle;
  WaitTime: DWord;
begin
  CoInitialize(nil);         //初始化COM
  try
   Synchronize(AddClient);       //在程序界面上显示客户信息,
    //用同步保证AddClient线程安全性
   FTransport := CreateServerTransport;
   try
    Event := FTransport.GetWaitEvent;
    PeekMessage(msg, 0, WM_USER, WM_USER, PM_NOREMOVE);
    GetInterface(ISendDataBlock, Obj);
    if FRegisteredOnly then
     FInterpreter := TDataBlockInterpreter.Create(Obj, SSockets) else 
     FInterpreter := TDataBlockInterpreter.Create(Obj, ‘);
    try
     Obj := nil;
     if FTimeout = 0 then
      WaitTime := INFINITE else
      WaitTime := 60000;
     while not Terminated and FTransport.Connected do
     try
      case MsgWaitForMultipleObjects(1, Event, False, WaitTime, QS_ALLEVENTS) of
       //MsgWaitForMultipleObjects保持线程同步之用,
       //本文暂不细说它.
       WAIT_OBJECT_0: //有数据来了
       begin
        WSAResetEvent(Event);
        Data := FTransport.Receive(False, 0);  //从客户端接收数据块
        if Assigned(Data) then
        begin
         FLastActivity := Now;
         FInterpreter.InterpretData(Data);//下面接着分析这儿
         Data := nil;
         FLastActivity := Now;
        end;
       end;
       WAIT_OBJECT_0 + 1:
        while PeekMessage(msg, 0, 0, 0, PM_REMOVE) do
         DispatchMessage(msg);
       WAIT_TIMEOUT:
        if (FTimeout > 0) and ((Now - FLastActivity) > FTimeout) then
         FTransport.Connected := False;
      end;
     except
      FTransport.Connected := False;
     end;
    finally
     FInterpreter.Free;
     FInterpreter := nil;
    end;
   finally
    FTransport := nil;
   end;
  finally
   CoUninitialize;
   Synchronize(RemoveClient);
  end;
end;
就这么舒舒服服的六十来行。
除开那些流程控制的语句,我们主要看到两个东西:
数据传输(FTransport) 和  数据解析(FInterpreter)。

在本文中,我更感兴趣的是它的应用协议的实现,
数据传输(FTransport)就先扔在一边,以后再看了.

现在我们就来看看它的数据解析部分。
。。。
FInterpreter := TDataBlockInterpreter.Create(Obj, SSockets) else 
FInterpreter := TDataBlockInterpreter.Create(Obj, ‘);
//这两种创建TDataBlockInterpreter类实例的方法的区别也不去管它.
。。。
 FInterpreter.InterpretData(Data);
。。。
就是这儿,就是这么一句。
它具体到底干了些什么呢??
================
。。。
 type
  TSocketDispatcherThread = class(TServerClientThread, ISendDataBlock)
  private
。。。
   FInterpreter: TDataBlockInterpreter;
   FTransport: ITransport;
。。。
 
================
FInterpreter这个对象引用就是这儿定义的。

procedure TDataBlockInterpreter.InterpretData(const Data: IDataBlock);
var
  Action: Integer;
begin
  Action := Data.Signature;//取出由客户端传来的数据块中请求标志值
     //客户端数据块的具体数据格式等会儿说.
  if (Action and asMask) = asError then DoException(Data);
  try
   case (Action and asMask) of   //根据客户端的请求标志值作相应的处理.
       //呵,就只有这么几个。一个大型的MIDAS系统
      //就全站在它们肩上.
    asInvoke: DoInvoke(Data);
    asGetID: DoGetIDsOfNames(Data);
    asCreateObject: DoCreateObject(Data);
    asFreeObject: DoFreeObject(Data);
    asGetServers: DoGetServerList(Data);
    asGetAppServers: DoGetAppServerList(Data);//取这个再进一步读一读.
   else
    if not DoCustomAction(Action and asMask, Data) then
     raise EInterpreterError.CreateResFmt(@SInvalidAction, [Action and asMask]);
   end;
  except
   on E: Exception do
   begin
    Data.Clear;
    WriteVariant(E.Message, Data);
    Data.Signature := ResultSig or asError;
    FSendDataBlock.Send(Data, False);
   end;
  end;
end;
==========================
顺着线一步步摸下去,
看它是怎么取APPSERVER列表返回客户端的。

很简单,就是通过GetMIDASAppServerList取本地的MIDAS应用服务
器列表,然后将其写在Data中,传回客户端就了事。
===========================
procedure TDataBlockInterpreter.DoGetAppServerList(const Data: IDataBlock);
var
  VList: OleVariant;
  List: TStringList;
  i: Integer;
begin
  Data.Clear;
  List := TStringList.Create;
  try
   GetMIDASAppServerList(List, FCheckRegValue);//取本机的应用服务器列表
     //想知道它是怎么做的吗?
     //源码上都有,自己看. 
   if List.Count > 0 then
   begin
    VList := VarArrayCreate([0, List.Count - 1], varOleStr);
    for i := 0 to List.Count - 1 do
     VList[i] := List[i];
   end else
    VList := NULL;
  finally
   List.Free;
  end;
  WriteVariant(VList, Data);
  Data.Signature := ResultSig or asGetAppServers;
  FSendDataBlock.Send(Data, False);//将应用服务器列表传回客户端
end;

========================================================
哦..前面还有一个地方没有说,就是这个神秘的Data的数据格式,它是以接口
形式提供的,
这个程序中用的是TDataBlock中实现的这个接口.
源码中可以看出,TDataBlock中包含了一个Stream,
function TDataBlock.GetSize: Integer;
begin
  Result := FStream.Size - BytesReserved;//BytesReserved的值这儿是8
end;
从这儿可以看出.数据块是从Stream的第8个字节算起,前面就是两个int型数的位置了.
function TDataBlock.GetSignature: Integer;
begin
  FStream.Position := 0;
  FStream.Read(Result, SizeOf(Result));
end;
呵, 没错, Stream的头四字节就是它的Signature.客户端的请求标志就是放在这儿.

function TDataBlock.GetStream: TStream;
var
  DataSize: Integer;
begin
  FStream.Position := 4;
  DataSize := FStream.Size - BytesReserved;
  FStream.Write(DataSize, SizeOf(DataSize));
  FStream.Position := 0;
  Result := FStream;
end;
这儿很明显,就是Data中包含数据的长度值.

需要提一下的是,如果你想改变一下传输数据块的格式,比如给它加密加压什么的,
中需要自己来实现IDataBlock接口,替掉TDataBlock就是了.

从这些源码中可以得到很多东西,无论是从优美的风格上,
无论是通讯程序开发还是MIDAS数据库以及DCOM学习或应用者,
它都是值得一读的源程序.
我觉得,更重要的是,它提供了一个严谨优美和实际的范例,更给出了
一个灵活实用的框架体系。

                       halfdream(哈欠) 于2001-10-14晚

http://blog.csdn.net/aroc_lo/article/details/9170229

时间: 2025-01-02 09:07:22

读一读Scktsrvr.exe的源程序的相关文章

Hadoop源码学习笔记(1) ——第二季开始——找到Main函数及读一读Configure类

Hadoop源码学习笔记(1) ——找到Main函数及读一读Configure类 前面在第一季中,我们简单地研究了下Hadoop是什么,怎么用.在这开源的大牛作品的诱惑下,接下来我们要研究一下它是如何实现的. 提前申明,本人是一直搞.net的,对java略为生疏,所以在学习该作品时,会时不时插入对java的学习,到时也会摆一些上来,包括一下设计模式之类的.欢迎高手指正. 整个学习过程,我们主要通过eclipse来学习,之前已经讲过如何在eclipse中搭建调试环境,这里就不多述了. 在之前源码初

[No000060]冷读热读:读书九问

兵无常势,水无常形,读书亦无法.彼之砒霜,我之佳肴.然读书无法却有道.你我都是使用同一颗大脑在读书.这颗大脑受制于那千千万万年以来,星辰起落,狩猎采集,演化大道. Q1:读物如何分级? 坏书.可用的书.力作.杰作与神作. 世人皆知精读略读,却不知冷读热读.你的大脑习惯用数字来锚定一切.既然都是书这种载体,价格.页数成了你的挑选标准.你总是习惯赋予神作与垃圾读物一样的阅读时间.须不知,人有好坏,书有高下.因此爱书之人均有自己的读物分级系统.我将图书分为:坏书.可用的书.力作.杰作与神作.多数图书,

杨绛先生送给年轻人的9句话,值得一读再读!

杨绛先生送给年轻人的9句话,值得一读再读! 1.你的问题主要在于读书不多而想得太多. 2.如要锻炼一个能做大事的人,必定要叫他吃苦受累,百不称心,才能养成坚忍的性格.一个人经过不同程度的锻炼,就获得不同程度的修养,不同程度的效益.好比香料,捣得愈碎,磨得愈细,香得愈浓烈. 3.有些人之所以不断成长,就绝对是有一种坚持下去的力量.好读书,肯下功夫,不仅读,还做笔记.人要成长,必有原因,背后的努力与积累一定数倍于普通人.所以,关键还在于自己. 4.少年贪玩,青年迷恋爱情,壮年汲汲于成名成家,暮年自安

SQL Server逻辑读-预读-物理读

SQL Server逻辑读-预读-物理读    SQL Server 存储数据的方式        1.页是最小的操作单元,也就是说从磁盘读取数据库的时候最少读取一页,每一页的大小是8KB,SQL SERVER对于页的读取是原子性,要么读完一页,要么完全不读,不会有中间状态 2.区是8个连续的页组成的,区是最小的分配单元,当需要空间时最少分配一个区的空间. 看图说话,两个表的结构完全一样,一个插入四条数据,另一个插入100条数据,结果大小都为0.008: SQL SERVER一页的总大小为:8K

数据库 一致性读&&当前读

今天小伙伴问了一个sql的问题: update t set status=2 where id in(select id from t where status=1) 这个sql,在并发的情况下,会不会有问题? 假设:下面的讨论,数据库的事务隔离级别是read_committed 其实这个可以很容易测试一下,得出结论:存在丢失更新的问题. 先来理解两个概念: 1. 一致性读 当前的数据库产品级别都实现了多版本一致性,即MVCC,那么有了MVCC,数据库实现了读写互不阻塞的效果. 但为了达到rea

ORACLE 物理读 逻辑读 一致性读 当前模式读总结浅析

在ORACLE数据库中有物理读(Physical Reads).逻辑读(Logical Reads).一致性读(Consistant Get).当前模式读(DB Block Gets)等诸多概念,如果不理解或混淆这些概念的话,对你深入理解一些知识无疑是一个障碍,但是这些概念确实挺让让人犯晕的.下面我们总结.学习一下这方面的知识点.捋一捋他们的关系和特点,希望对你有所帮助. 物理读(Physical Reads) 从磁盘读取数据块到内存的操作叫物理读,当SGA里的高速缓存(Cache Buffer

物理读,逻辑读,预读

在使用SET STATISTICS IO ON语句统计I/O时候,我们会看到类似下面的结果: 扫描计数 1,逻辑读取 2 次,物理读取 0 次,预读 0 次,lob 逻辑读取 0 次,lob 物理读取 0 次,lob 预读 0 次. 那么它们代表什么呢? 预读:用于估计信息,去硬盘读取数据到缓存. 物理读:查询计划生成好以后,如果缓存缺少所需要的数据,让缓存再次去读硬盘.如果内存里没有缓存数据或执行计划(sql语句改变执行计划不能重用,需要重新计算执行计划),那么SQLSERVER就要去硬盘读取

建议程序员都读一读的31篇论文系列笔记(1~2)

序:前几日网上偶尔看到"程序员必读论文系列",顺便搜了一下,发现有多个版本共31篇,不过看起来都不错,故准备花时间都读一下,可以拓宽下视野.来源论文题目主要参考 http://blog.csdn.net/turingbook/article/details/3946421 和 http://top.jobbole.com/17733/ .每读完一篇论文就写些笔记,或长或短,也就是这几篇文章的由来. 1. An Axiomatic Basis for Computer Programmi

读着读着《构建之法》(Build To Win) 越精神的白雪儿的思考

哲学家的宗旨是:我思,故我在 科学家的宗旨是:我发现,故我在 工程师的宗旨是:我构建,故我在 ——<工程学--无尽的前沿> 序言:珍惜角色“人”,注重实践“物” <构建之法>,精读三曲,感触良多. 曲一,语言诙谐幽默,思维独具匠心:曲二,提问勾画,思考获益:曲三,豁然开朗,又困惑不解.软件工程与“人”有不解之缘,“人”用百花齐放的实践构建软件工程.三曲之后,知识概念,不必硬背,只需循序渐进,逐步实践体验,但不得不提出如下五惑. 核心:提出困惑点,分享你我他 第 0 章  目录: 1