一个实际的OLE服务器的开发和使用

原创 2003年11月23日 19:16:00

一个实际的OLE服务器的开发和使用

先行知识:Delphi /COM/OLE Automation/ SQL Server

难度:★★☆☆☆

 <?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

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

首先我们来做一个实际的程序,在它没有变为服务器之前,这是个用来管理客户购买记录的程序(它自己与SQL Server连接),它可以录入和删除客户的购买记录并直观的显示出来,所有的数据都存放在SQL Server中。我们将它做为OLE Automation出于这样一种考虑,假设我们是一家大型的供货公司,我们可能有很多系统需要使用这个客户购买记录程序并用它处理SQL Server中相应的数据,但我们不愿意每次都重复的编写同样的处理代码,我们更希望能把这个处理程序独立出来,并向其它程序提供服务。那么在下面的工作中我们完成了这个服务器程序,界面如下:(注意,这仅仅是一个例子,我们不评价其数据库设计的好坏J

<?xml:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" />r_OLEDocPro.jpg

我们不过多的讨论这个程序的代码(因为这和开发一般的程序没有任何不同,你可以按照最后的地址给我来信索取这篇文章的全部代码)。然后我们来把它变为一个服务器。选择FileàNewàOthersàActiveXàAutomation Object。接下来delphi为我们定义了类型库和实现文件,我们要做的只是在类型库中添加相应的我们要用到的服务器属性和事件。我们简单的给出定义这个OLE Automation功能的接口(来自类型库所产生的Object Pascal代码):

  ICustFormOLE = interface(IDispatch)

    ['{D7AE75F9-F838-4702-A8EB-EAD0EED242DE}']

    function Get_CustName: WideString; safecall;

    procedure Set_CustName(const Value: WideString); safecall;

    function Get_ProductName: WideString; safecall;

    procedure Set_ProductName(const Value: WideString); safecall;

    function Get_ProductNum: Integer; safecall;

    procedure Set_ProductNum(Value: Integer); safecall;

    function Get_Remark: WideString; safecall;

    procedure Set_Remark(const Value: WideString); safecall;

    //下面的方法和属性都对应着原程序中相应的方法和属性

    procedure AddToData; safecall;

    procedure DelData; safecall;

    property CustName: WideString read Get_CustName write Set_CustName;

    property ProductName: WideString read Get_ProductName write Set_ProductName;

    property ProductNum: Integer read Get_ProductNum write Set_ProductNum;

    property Remark: WideString read Get_Remark write Set_Remark;

  end;

 

  ICustFormOLEDisp = dispinterface

    ['{D7AE75F9-F838-4702-A8EB-EAD0EED242DE}']

    property CustName: WideString dispid 201;

    property ProductName: WideString dispid 202;

    property ProductNum: Integer dispid 203;

    property Remark: WideString dispid 204;

    procedure AddToData; dispid 205;

    procedure DelData; dispid 206;

  end;

我们现在回到接口的实现文件,注意代码中的注释,事实上这段代码相当的简单:

unit CustOLEImpUnit;

 

{$WARN SYMBOL_PLATFORM OFF}

 

interface

 

uses

  ComObj, ActiveX, CustViewOLE_TLB, StdVcl,windows;

 

type

  TCustFormOLE = class(TAutoObject, ICustFormOLE)

//注意这里实现了我们在前面定义的ICustFormOLE接口

  protected

    function Get_CustName: WideString; safecall;

    function Get_ProductName: WideString; safecall;

    function Get_ProductNum: Integer; safecall;

    function Get_Remark: WideString; safecall;

    procedure AddToData; safecall;

    procedure DelData; safecall;

    procedure Set_CustName(const Value: WideString); safecall;

    procedure Set_ProductName(const Value: WideString); safecall;

    procedure Set_ProductNum(Value: Integer); safecall;

    procedure Set_Remark(const Value: WideString); safecall;

  end;

 

implementation

 

uses ComServ,CustFormUnit;

 

function TCustFormOLE.Get_CustName: WideString;

begin

 result:=CustForm.CustomEdit.Text;

 //可以看到,我们只是用了最初程序窗体的控件和属性,这里的接口实现相当于

 //只是简单的封状了我们的原始程序,下面的代码情况类似。

end;

 

function TCustFormOLE.Get_ProductName: WideString;

begin

 result:=CustForm.ProductEdit.Text;

end;

 

function TCustFormOLE.Get_ProductNum: Integer;

begin

 result:=CustForm.ProNumEdit.Value;

end;

 

function TCustFormOLE.Get_Remark: WideString;

begin

 result:=CustForm.Memo1.Lines.Text;

end;

 

procedure TCustFormOLE.AddToData;

begin

 CustForm.AddButton.Click;

end;

 

procedure TCustFormOLE.DelData;

begin

 CustForm.DelButton.Click;

end;

 

procedure TCustFormOLE.Set_CustName(const Value: WideString);

begin

 CustForm.CustomEdit.Text:=Value;

end;

 

procedure TCustFormOLE.Set_ProductName(const Value: WideString);

var

 i:integer;

begin

 i:=CustForm.ProductEdit.Items.IndexOf(Value);

 if i<>-1 then

  CustForm.ProductEdit.ItemIndex:=i

 else

 begin

  messagebox(CustForm.Handle,'你在客户程序指定的商品类型并不存在!','CustProOLE常规错误',MB_ICONWARNING);

  CustForm.ProductEdit.ItemIndex:=0;

 end;

end;

 

procedure TCustFormOLE.Set_ProductNum(Value: Integer);

begin

 CustForm.ProNumEdit.Value:=Value;

end;

 

procedure TCustFormOLE.Set_Remark(const Value: WideString);

begin

 CustForm.Memo1.Lines.Text:=Value;

end;

 

initialization

  TAutoObjectFactory.Create(ComServer, TCustFormOLE, Class_CustFormOLE,

    ciMultiInstance, tmApartment);

end.

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

function TForm1.GetDefaultInterface:ICustFormOLE;

begin

 if not assigned(FInterface) then

  FInterface:=CoCustFormOLE.Create;//注意这里,你可以在类型库文件产生的pascal文件中找到CoCustFormOLE的含义

 result:=FInterface;

end;

    由于篇幅原因,我们不能给出测试程序的全部代码(事实上有了服务器程序,我们的测试客户程序想要处理SQL Server中的相应数据就相当的简单了。),可以照后文的地址向我索取(说明一下,本文中的数据库用到SQL Server,所以我发给你们的程序中你们需要还原其中的数据库备份到你们的SQL Server,并修改相应的连接字符串,否则程序不能运行)。

在本文的最后,我们介绍一种更简单的使用我们刚才所开发的服务器的方法,那就是把它封装为delphi中的组件,选择Project-->Import Type Library…中我们开发的服务器,然后Install将它安装到一个已经存在的包或你新建的组件包中,delphi将为我们做很多工作,最后你可以从你指定的面板找到安装的新的组件,现在你就可以象使用普通VCL组件一样使用我们开发的服务器了。(注意,delphi为我们定义了一个继承自ToleContol的类,这一切复杂的工作都是由delphi在背后为我们完成的,如果你有兴趣,建议研究一下这个组件中delphi自动为我们生成的大量代码)。

索取地址:hk.barton@sohu.com

git在实际开发中的运用-基本操作

今天,来分享下我在开发中使用到的一些git基本命令,同时也帮助刚入门的小伙伴们加深一些印象。 git的特点就不多说了,分布式、版本可控是其中比较有特色的亮点。 我们将Git的内容分成两个大部分:基础操...
  • u013718120
  • u013718120
  • 2015年09月17日 00:04
  • 1357

OLE复合文档编程参考

/////////////////////////////////////////////////结构化存储和OLE对象////////////////////////////////////////...
  • tangyanzhi1111
  • tangyanzhi1111
  • 2013年06月07日 13:51
  • 3598

OLe复合文档介绍

/////////////////////////////////////////////////结构化存储和OLE对象////////////////////////////////////////...
  • dotneterbj
  • dotneterbj
  • 2014年03月10日 10:42
  • 1327

将现有程序改造为OLE服务器供包容器(word,ppt)使用

http://blog.csdn.net/leohels/article/details/7580873 我们手头上经常会有现成的应用程序,但它们往往不具有OLE可视化编辑服务器的特性,可以通过简单的...
  • Augusdi
  • Augusdi
  • 2013年10月24日 15:15
  • 1947

java多线程简单操作和实际项目开发如何写

先上代码: import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import j...
  • u010235716
  • u010235716
  • 2016年06月15日 14:08
  • 1530

MFC中使用OLE/COM操作EXCEL的方法

使用OLE的方法操作EXCEL,首先计算机必须安装excel,这样才会有接口暴露出来。本次使用大神封装好的类。 excel作为OLE/COM库插件,定义好了各类交互接口,而且这些接口是跨语言的,可以导...
  • lht501692913
  • lht501692913
  • 2015年12月17日 19:25
  • 2307

从OLE到COM,再到ActiveX,再到.NET

微软从OLE到COM,再到ActiveX,再到.NET的发展历史的简介
  • just0kk
  • just0kk
  • 2016年03月02日 21:31
  • 755

runtime从入门到精通(六)—— runtime在实际开发中的应用

上一篇文章,我们学习了runtime的消息发送和消息转发机制(查看链接: runtime从入门到精通(五)—— 消息发送和消息转发 ),倒到此为止,有关runtime的理论知识介绍就先告于段落,小伙伴...
  • coyote1994
  • coyote1994
  • 2016年09月08日 12:08
  • 2109

OLE、ActiveX 、COM和普通DLL的简单理解

COM: COM组件是以WIN32动态链接库(DLL)或可执行文件(EXE)形式发布的可执行代码组成。 COM组件是遵循COM规范编写的 COM组件是一些小的二进制可执行文件 COM组件可以给应用程序...
  • Liekkas
  • Liekkas
  • 2015年04月27日 11:22
  • 1113

以OLE方式读写EXCEL的C++类【转载】

◆C++中的【L"https://msdn.microsoft.com/library"】。 这个是C++标准规定的写法。详见以下链接: http://bbs.csdn.net/topics/3906...
  • u013600225
  • u013600225
  • 2016年03月20日 08:21
  • 1296
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:一个实际的OLE服务器的开发和使用
举报原因:
原因补充:

(最多只允许输入30个字)