delphi .dfm文件汉字不显示_搞定Delphi模块化之Package

Package的优点

  • 应用程序可以被高度的模块化,而且可以逐渐交付完成的功能给客户
  • 维护方便,可以只更新单一的模块功能
  • 提升程序的载入速度

Package的缺点

  • 有些情形下使用Package只能间接的方式取得资料(变量, 类 …).
  • Package Name 不能重复.
  • Contains 中的 Unit Name 不能在「所有」的Package中重复出现,只能出现一次
  • PackageA有使用到PackageB必须要在Requires中引用 但是PackageA及PackageB「不能彼此循环引用.」

Package种类

当用户运行应用程序时,运行时程序包提供功能。设计时程序包用于在IDE中安装组件并为自定义组件创建特殊的属性编辑器。单个包可以在设计时和运行时均起作用,并且设计时包经常通过在其require子句中引用运行时包来工作。

  • 设计期包(Designtime only) -用来在DELPHI的IDE环境安装控件和为控件建立特殊的属性编辑器。设计期包允许包含控件、属性和控件编辑器等等,
  • 运行期包(Runtime only)-当运行程序时提供VCL和库函数的支持,操作上很类似标准的动态链接库。Install按钮无法使用。
  • 设计和运行期包(Designtime and Runtime ):设计与运行时都能用

    18835e828b2e5821e339cb40baa83958.png

Package文件说明

BPL  英文全称 Borland Package library ,是一种特殊的DLL文件,用于代码重用和减少可执行文件。编译bpl时,仅需要添加相应功能的pas文件,如果有窗体,则需要添加dfm文件。既然是DLL文件,那就是在运行时所需要的文件。BPL相当于C++中的DLL

DCP  英文全称:delphi compiled package,是 package 编译时跟 bpl 一起产生出来的,记录着 package 中公开的 class、procedure、function、variable、const.... 等等的名称和相对位址。如果 某个控件包 A 引用了 控件包 B,当 控件包 A 编译时,需要 控件包 B.dcp,若 控件包 B 有修改,更改了公开的介面,则 控件包 A 必须在 控件包 B 编译之后重新编译,以引用新的 B.dcp。否则,当 控件包 A 执行时,执行到引用自 控件包 B 的内容时,就会出现错误。DCP相当于C++中的Lib,编译时需要。

DCU  英文全称:Delphi Compiled Unit File,是delphi单元文件.pas文件编译后产生的文件,感觉没有太大用处。

Package加载方式

Package中的代码

unit Unit2;

interface

uses Vcl.Dialogs;

//函数案例
function add(Num1, Num2: Integer): Integer; stdcall;

//过程案例
procedure ShowMsg(Str: String); stdcall;

type
//类的案例
  TUser = class
  public
    function ShowString(): string;
  end;

  // 需要像DLL一样声明导出函数的列表,如果是静态导入此项可以省略
exports add, ShowMsg;

implementation
procedure ShowMsg(Str: String);
beginshowmessage(Str);
end;

function add(Num1, Num2: Integer): Integer;
begin
  Result := Num1 + Num2;
end;

{ TUser }

function TUser.ShowString: string;
begin

  Result := 'HelloWorld';
end;

end.

静态加载

一般大家在用Delphi時都是使用『静态载入』, 像VCL的Package就是这种方式, 这种方式的好处是设计者不用去理会Package 的载入和释放, 其实设计者根本感觉不到设用这项技术; 当然也可以手动将Package加入到项目中『project->Options->Packages->Build with runtime packages中加入Package Name彼此的分隔符是分号』

99ce977001b04d8bc518243063303b20.png

b875b7ec199e9810d64afb6ee58d6367.png

动态载入代码

基本上是无痛使用,只要路径配置没有问题,基本上和使用普通单元没有区别

implementation

uses  Unit2, Unit3;
{$R *.dfm}

procedure TForm1.Button1Click(Sender: TObject);

beginshowmessage(TUser.create().showString());
  var
  From3 := TForm3.create(nil);
  From3.visible := true;

end;

动态加载

动态加载和静态加载相反,无论是载入还是释放都要自己来处理,看起来好像是动态载入,这种方式个人感觉相当麻烦,虽然本质上和dll的动态加载一样,但是因为在导入的元素中多了类的概念,所以还需要使用反射的方式创建类的对象才能实现类成员的引用

implementation

uses rtti, System.StrUtils;
{$R *.dfm}

procedure TForm1.Button2Click(Sender: TObject);
var
  // 声明和Package导出列表中一致结构的过程
  add01: procedure(Msg: String); stdcall;
  // 声明和Package导出列表中一致结构的函数
  add02: function(Num1: Integer; Num2: Integer): Integer; stdcall;
begin
  // 载入bpl格式的Package
  var
  PackageHandle := LoadPackage('Package1.bpl');

  if PackageHandle <> 0 then
  begin
    // 载入成功之后获取对应函数、过程的指针
    @add01 := GetProcAddress(PackageHandle, 'ShowMsg');
    @add02 := GetProcAddress(PackageHandle, 'add');
    if @add01 <> nil then
    begin
      // 调用
      add01('HelloWorld');
      showmessage(add02(1, 2).Tostring);
    end;

  end;
  // 对于类我们需要先创建类的对象然后才可以实现类中函数的调用
  var
    // 创建运行期上下问对象
  rc := TRttiContext.create;
  var
    // 载入对应单元中的类,注意此处需要写单元名+类名
  ClassType := rc.FindType('Unit2.TUser');

  var
    // 获取元类实例(对象)
  Instance := ClassType.AsInstance;
  var
    // 获取该实例的元信息类型
  QRClass := Instance.MetaclassType;
  var
    // 获取用于创建TUser类型的构造方法
  CreateMethod := Instance.GetMethod('Create');
  var
    // 利用获取到的构造方法对象,创建TUser类对象
  User := CreateMethod.Invoke(QRClass, []);

  var
    // 函数调用
  rs := ClassType.GetMethod('ShowString').Invoke(User, []);
  // 显示返回值
  showmessage(rs.asstring);
  //卸载包
  UnloadPackage(PackageHandle);
end;

从上面动态加载的代码可以看出涉及到反射相关的知识,个人感觉这种方式在使用起来不太方便,当然如果对反射比较熟悉的话那就没问题了

动态载入参考代码

我在搜索Package相关内容的使用看到下面这段代码,它也可以实现创建类的对象,只是中间出现的类型的强制转换,个人不是特别推荐,只是记录一下作为笔记参考

function CreateFormByClassName(ClassName: string): integer;
var
  AClass: TPersistentClass;
  AForm: TCustomForm;
begin
  Result := mrNone;
  AClass := GetClass(ClassName);
  if AClass <> nil then
  begin
    AForm := TComponentClass(AClass).Create(Application) as TCustomForm;
    Result := AForm.ShowModal;
  end;

官方参考文档

官方文档是英文的,我也是翻看+翻译读了很久挑了几篇有用的

  • http://docwiki.embarcadero.com/RADStudio/Sydney/en/Packages_(Delphi)
  • http://docwiki.embarcadero.com/RADStudio/Sydney/en/Compiling_Packages
  • http://docwiki.embarcadero.com/RADStudio/Sydney/en/Loading_Packages_in_an_Application
  • http://docwiki.embarcadero.com/RADStudio/Sydney/en/Add_Runtime_Package

在线观看

30d5bfcb8639ca907b3add3b7aac2b15.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值