站内搜索

一个实际的OLE服务器的开发

 先行知识:Delphi/COM/OLEAutomation/SQLServer

   难度:★★☆☆☆

  在前几篇文章中我们已经讨论过关于VCL和OLE的知识。在这篇文章中我们将完成一个比较有实际意义的OLEAutomation服务器程序,最后我们把他们封装为Delphi中使用的VCL组件。

  首先我们来做一个实际的程序,在它没有变为服务器之前,这是个用来管理客户购买记录的程序(它自己与SQLServer连接),它可以录入和删除客户的购买记录并直观的显示出来,所有的数据都存放在SQLServer中。我们将它做为OLEAutomation出于这样一种考虑,假设我们是一家大型的供货公司,我们可能有很多系统需要使用这个客户购买记录程序并用它处理SQLServer中相应的数据,但我们不愿意每次都重复的编写同样的处理代码,我们更希望能把这个处理程序独立出来,并向其它程序提供服务。那么在下面的工作中我们完成了这个服务器程序,界面如下:(注意,这仅仅是一个例子,我们不评价其数据库设计的好坏J)
  
  我们不过多的讨论这个程序的代码(因为这和开发一般的程序没有任何不同,你可以按照最后的地址给我来信索取这篇文章的全部代码)。然后我们来把它变为一个服务器。选择FileàNewàOthersàActiveXàAutomationObject。接下来delphi为我们定义了类型库和实现文件,我们要做的只是在类型库中添加相应的我们要用到的服务器属性和事件。我们简单的给出定义这个OLEAutomation功能的接口(来自类型库所产生的ObjectPascal代码):
  
  ICustFormOLE=interface(IDispatch)
  
  ['{D7AE75F9-F838-4702-A8EB-EAD0EED242DE}']
  
  functionGet_CustName:WideString;safecall;
  
  procedureSet_CustName(constValue:WideString);safecall;
  
  functionGet_ProductName:WideString;safecall;
  
  procedureSet_ProductName(constValue:WideString);safecall;
  
  functionGet_ProductNum:Integer;safecall;
  
  procedureSet_ProductNum(Value:Integer);safecall;
  
  functionGet_Remark:WideString;safecall;
  
  procedureSet_Remark(constValue:WideString);safecall;
  
  //下面的方法和属性都对应着原程序中相应的方法和属性
  
  procedureAddToData;safecall;
  
  procedureDelData;safecall;
  
  propertyCustName:WideStringreadGet_CustNamewriteSet_CustName;
  
  propertyProductName:WideStringreadGet_ProductNamewriteSet_ProductName;
  
  propertyProductNum:IntegerreadGet_ProductNumwriteSet_ProductNum;
  
  propertyRemark:WideStringreadGet_RemarkwriteSet_Remark;
  
  end;

 ICustFormOLEDisp=dispinterface
  
  ['{D7AE75F9-F838-4702-A8EB-EAD0EED242DE}']
  
  propertyCustName:WideStringdispid201;
  
  propertyProductName:WideStringdispid202;
  
  propertyProductNum:Integerdispid203;
  
  propertyRemark:WideStringdispid204;
  
  procedureAddToData;dispid205;
  
  procedureDelData;dispid206;
  
  end;
  
  我们现在回到接口的实现文件,注意代码中的注释,事实上这段代码相当的简单:
  
  unitCustOLEImpUnit;
  
  {$WARNSYMBOL_PLATFORMOFF}
  
  interface
  
  uses
  
  ComObj,ActiveX,CustViewOLE_TLB,StdVcl,windows;
  
  type
  
  TCustFormOLE=class(TAutoObject,ICustFormOLE)
  
  //注意这里实现了我们在前面定义的ICustFormOLE接口
  
  protected
  
  functionGet_CustName:WideString;safecall;
  
  functionGet_ProductName:WideString;safecall;
  
  functionGet_ProductNum:Integer;safecall;
  
  functionGet_Remark:WideString;safecall;
  
  procedureAddToData;safecall;
  
  procedureDelData;safecall;
  
  procedureSet_CustName(constValue:WideString);safecall;
  
  procedureSet_ProductName(constValue:WideString);safecall;
  
  procedureSet_ProductNum(Value:Integer);safecall;
  
  procedureSet_Remark(constValue:WideString);safecall;
  
  end;
  
  implementation
  
  usesComServ,CustFormUnit;
  
  functionTCustFormOLE.Get_CustName:WideString;
  
  begin
  
  result:=CustForm.CustomEdit.Text;
  
  //可以看到,我们只是用了最初程序窗体的控件和属性,这里的接口实现相当于
  
  //只是简单的封状了我们的原始程序,下面的代码情况类似。
 end;
  
  functionTCustFormOLE.Get_ProductName:WideString;
  
  begin
  
  result:=CustForm.ProductEdit.Text;
  
  end;
  
  functionTCustFormOLE.Get_ProductNum:Integer;
  
  begin
  
  result:=CustForm.ProNumEdit.Value;
  
  end;
  
  functionTCustFormOLE.Get_Remark:WideString;
  
  begin
  
  result:=CustForm.Memo1.Lines.Text;
  
  end;

  procedureTCustFormOLE.AddToData;
  
  begin
  
  CustForm.AddButton.Click;
  
  end;
  
  procedureTCustFormOLE.DelData;
  
  begin

  CustForm.DelButton.Click;
  
  end;
  
  procedureTCustFormOLE.Set_CustName(constValue:WideString);
  
  begin
  
  CustForm.CustomEdit.Text:=Value;

  end;
  
  procedureTCustFormOLE.Set_ProductName(constValue:WideString);
  
  var
  
  i:integer;
  
  begin
  
  i:=CustForm.ProductEdit.Items.IndexOf(Value);
  
  ifi<>-1then
  
  CustForm.ProductEdit.ItemIndex:=i
  
  else
  begin
  
  messagebox(CustForm.Handle,'你在客户程序指定的商品类型并不存在!','CustProOLE常规错误',MB_ICONWARNING);
  
  CustForm.ProductEdit.ItemIndex:=0;
  
  end;
  
  end;
  
  procedureTCustFormOLE.Set_ProductNum(Value:Integer);
  
  begin
  
  CustForm.ProNumEdit.Value:=Value;
  
  end;
  
  procedureTCustFormOLE.Set_Remark(constValue:WideString);
  
  begin
  
  CustForm.Memo1.Lines.Text:=Value;
  
  end;

  initialization
  
  TAutoObjectFactory.Create(ComServer,TCustFormOLE,Class_CustFormOLE,
  
  ciMultiInstance,tmApartment);
  
  end.

  现在我们就可以实际的测试和使用这个服务器了,我们可以新建立一个工程,选择Project-->ImportTypeLibrary…可以发现这里已经有我们刚才建立的服务器信息了(当然前提是你已经运行过服务器程序),然后CreateUnit将相应的类型文件所生成的pascal文件加入我们的工程中,一但我们启动了服务器我们就可以很轻松的使用接口中的属性和方法了:

  functionTForm1.GetDefaultInterface:ICustFormOLE;
  
  begin
  
  ifnotassigned(FInterface)then
  
  FInterface:=CoCustFormOLE.Create;//注意这里,你可以在类型库文件产生的pascal文件中找到CoCustFormOLE的含义
  
  result:=FInterface;
  
  end;
  
  由于篇幅原因,我们不能给出测试程序的全部代码(事实上有了服务器程序,我们的测试客户程序想要处理SQLServer中的相应数据就相当的简单了。),可以照后文的地址向我索取(说明一下,本文中的数据库用到SQLServer,所以我发给你们的程序中你们需要还原其中的数据库备份到你们的SQLServer,并修改相应的连接字符串,否则程序不能运行)。
  
  在本文的最后,我们介绍一种更简单的使用我们刚才所开发的服务器的方法,那就是把它封装为delphi中的组件,选择Project-->ImportTypeLibrary…中我们开发的服务器,然后Install将它安装到一个已经存在的包或你新建的组件包中,delphi将为我们做很多工作,最后你可以从你指定的面板找到安装的新的组件,现在你就可以象使用普通VCL组件一样使用我们开发的服务器了。(注意,delphi为我们定义了一个继承自ToleContol的类,这一切复杂的工作都是由delphi在背后为我们完成的,如果你有兴趣,建议研究一下这个组件中delphi自动为我们生成的大量代码)。
  
  索取地址:hk.barton@sohu.com

  • 上一篇:GSM规范中的部分编码转换
  • 下一篇:Delphi学习:图像放大漫游攻略