请教:为何在DELPHI中调用DLL的过程后即自动退出程序? ( 积分: 100 )

  • 主题发起人 主题发起人 michaelqz
  • 开始时间 开始时间
M

michaelqz

Unregistered / Unconfirmed
GUEST, unregistred user!
Tdlltype = procedure(AHandle:THandle); stdcall;

var
DllHandle:THandle;
ShowOSInfo:Tdlltype;
begin
DllHandle:=LoadLibrary('chz.dll');
try
if DllHandle=0 then exit;
@ShowOSInfo:=GetProcAddress(DllHandle,'Chz_input');
if Assigned(@ShowOSInfo) then ShowOSInfo(Application.Handle);
finally
FreeLibrary(DllHandle);
end;
end;

以上是主程序

procedure Chz_input(AHandle: THandle);
//var
// old:LongWord;
begin
// old:=Application.Handle;
// Application.Handle:=AHandle;
ShowMessage('aaa');
// Application.
// Application.Handle:=old;
// Exit;
// OneMdi(Tchz_input);
end;

exports
Chz_input;

以上是DLL中的程序,不管Application.Handle:=AHandle;有没有运行,当调用完DLL中的过程后,所有程序即退出.请问有哪位高手知道怎么解决,万分感谢!
原帖来自于网易社区:http://club.163.com/viewArticleByWWW.m?boardId=program&articleId=program_11115c4017e1ff4_0
 
dll中的输出方法加stdcall
procedure Chz_input(AHandle: THandle); stdcall;
如果还不行,尝试传递Application对象进去。
procedure Chz_input(AApp: TApplication); stdcall;
begin
....
Application := AApp;
....
end;


另外,还要定义dll入回函数。
参考下面的资料
下文作者: 阿朱。
-------------------------
用Delphi制作DLL
一 Dll的制作一般步骤
二 参数传递
三 DLL的初始化和退出清理[如果需要初始化和退出清理]
四 全局变量的使用
五 调用静态载入
六 调用动态载入
七 在DLL建立一个TForM
八 在DLL中建立一个TMDIChildForM
九 示例:
十 Delphi制作的Dll与其他语言的混合编程中常遇问题:
十一 相关资料

一 Dll的制作一般分为以下几步:
1 在一个DLL工程里写一个过程或函数
2 写一个Exports关键字,在其下写过程的名称。不用写参数和调用后缀。
二 参数传递
1 参数类型最好与window C++的参数类型一致。不要用DELPHI的数据类型。
2 最好有返回值[即使是一个过程],来报出调用成功或失败,或状态。成功或失败的返回值最好为1[成功]或0[失败].一句话,与windows c++兼容。
3 用stdcall声明后缀。
4 最好大小写敏感。
5 无须用far调用后缀,那只是为了与windows 16位程序兼容。

三 DLL的初始化和退出清理[如果需要初始化和退出清理]
1 DLLProc[SysUtils单元的一个Pointer]是DLL的入口。在此你可用你的函数替换了它的入口。但你的函数必须符合以下要求[其实就是一个回调函数]。如下:
procedure DllEnterPoint(dwReason: DWORD);far;stdcall;
dwReason参数有四种类型:
DLL_PROCESS_ATTACH:进程进入时
DLL_PROCESS_DETACH进程退出时
DLL_THREAD_ATTACH 线程进入时
DLL_THREAD_DETACH 线程退出时
在初始化部分写:
DLLProc := @DLLEnterPoint;
DllEnterPoint(DLL_PROCESS_ATTACH);
2 如Form上有TdcomConnection组件,就Uses Activex,在初始化时写一句CoInitialize (nil);
3 在退出时一定保证DcomConnection.Connected := False,并且数据集已关闭。否则报地址错。

四 全局变量的使用
在widnows 32位程序中,两个应用程序的地址空间是相互没有联系的。虽然DLL在内存中是一份,但变量是在各进程的地址空间中,因此你不能借助dll的全局变量来达到两个应用程序间的数据传递,除非你用内存映像文件。

五 调用静态载入
1 客户端函数声名:
1)大小写敏感。
2)与DLL中的声明一样。
如: showform(form:Tform);Far;external'yproject_dll.dll';
3)调用时传过去的参数类型最好也与windows c++一样。
4)调用时DLL必须在windows搜索路径中,顺序是:当前目录;Path路径;windows;widows/system;windows/ssystem32;

六 调用动态载入
1 建立一种过程类型[如果你对过程类型的变量只是一个指针的本质清楚的话,你就知道是怎么回事了]。如:
type
mypointer=procedure(form:Tform);Far;external;
var
Hinst:Thandle;
showform:mypointer;
begin
Hinst:=loadlibrary('yproject_dll');//Load一个Dll,按文件名找。
showform:=getprocaddress(Hinst,'showform');//按函数名找,大小写敏感。如果你知道自动化对象的本质就清楚了。
showform(application.mainform);//找到函数入口指针就调用。
Freelibrary(Hinst);
end;

七 在DLL建立一个TForM
1 把你的Form Uses到Dll中,你的Form用到的关联的单元也要Uses进来[这是最麻烦的一点,因为你的Form或许Uses了许多特殊的单元或函数]
2 传递一个Application参数,用它建立Form.

八 在DLL中建立一个TMDIChildForM
1 Dll中的MDIForm.FormStyle不用为fmMDIChild.
2 在CreateForm后写以下两句:
function ShowForm(mainForm:TForm):integer;stdcall
var
Form1: TForm1;
ptr:PLongInt;
begin
ptr:=@(Application.MainForm);//先把dll的MainForm句柄保存起来,也无须释放,只不过是替换一下
ptr^:=LongInt(mainForm);//用主调程序的mainForm替换DLL的MainForm。MainForm是特殊的WINDOW,它专门管理Application中的Forms资源.
//为什么不直接Application.MainForm := mainForm,因为Application.MainForm是只读属性
Form1:=TForm1.Create(mainForm);//用参数建立
end;
备注:参数是主调程序的Application.MainForm

九 示例:
DLL源代码:
library Project2;

uses
SysUtils,
Classes,
Dialogs,
Forms,
Unit2 in 'Unit2.pas' {Form2};

{$R *.RES}
var
ccc: Pchar;

procedure OpenForm(mainForm:TForm);stdcall;
var
Form1: TForm1;
ptr:PLongInt;
begin
ptr:=@(Application.MainForm);
ptr^:=LongInt(mainForm);
Form1:=TForm1.Create(mainForm);
end;

procedure InputCCC(Text: Pchar);stdcall;
begin
ccc := Text;
end;

procedure ShowCCC;stdcall;
begin
ShowMessage(String(ccc));
end;

exports
OpenForm;
InputCCC,
ShowCCC;
begin
end.

调用方源代码:
unit Unit1;

interface

uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls;

type
TForm1 = class(TForm)
Button1: TButton;
Button2: TButton;
Edit1: TEdit;
procedure Button1Click(Sender: TObject);
procedure Button2Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;

var
Form1: TForm1;

implementation

{$R *.DFM}
procedure OpenForm(mainForm:TForm);stdcall;External'project2.dll';
procedure ShowCCC;stdcall;External'project2.dll';
procedure InputCCC(Text: Pchar);stdcall;External'project2.dll';

procedure TForm1.Button1Click(Sender: TObject);
var
Text: Pchar;
begin
Text := Pchar(Edit1.Text);
// OpenForm(Application.MainForm);//为了调MDICHILD
InputCCC(Text);//为了实验DLL中的全局变量是否在各个应用程序间共享
end;

procedure TForm1.Button2Click(Sender: TObject);
begin
ShowCCC;//这里表明WINDOWS 32位应用程序DLL中的全局变量也是在应用程序地址空间中,16位应用程序或许不同,没有做实验。
end;

十 Delphi制作的Dll与其他语言的混合编程中常遇问题:
1 与PowerBuilder混合编程
在定义不定长动态数组方面在函数退出清理堆栈时老出现不可重现的地址错,原因未明,大概与PB的编译器原理有关,即使PB编译成二进制代码也如此。

//----------------------------------------------------------------------------------------------------

用DLL方式封装MDI子窗体。 选择自 lostpen 的 Blog
关键字 用DLL方式封装MDI子窗体。
出处

用DLL方式封装MDI子窗体是一种常用的软件开发技术,它的优点:

开发人员可以负责某一个模块的编写包括(界面+逻辑),可以互不干扰,模块开发完成后,主程序统一调用。
易于程序升级,当程序升级时,不用编译主exe文件,更新某个DLL就可以升级。
可以根据客户不同的需求、价钱给他们不同的模块。(奸商都愿意用这招)
最近用到这个技术,因为小弟没有用过,在网上google了一下,对它有了了解,我用的是基于COM的DLL,下面总结如下:

想在DLL中封装MDI子窗体,要注意以下几点:

你的DLL中的接口要有两个参数:第一,主程序的Application对象;第二,主程序窗体对象;
你要重写DLL入口函数
如果是基于COM的,要注意:因为COM没有TApplication和TForm类型,你在传入时需要强转为Integer。
需要注意的就这么多,下面结合例子看看(假设你已经建立了工程):

TComDllTest1 = class(TAutoObject, IComDllTest1)
Private
FMsgStr: WideString;
function GetMDIForm(AForm: TForm): boolean;
protected
{ Protected declarations }
function Get_MsgStr: WideString; safecall;
procedure Set_MsgStr(const Value: WideString); safecall;
procedure ShowMsgStr; safecall;
procedure CreateForm(AHandle: SYSUINT); safecall;
procedure CreateMdiForm(var AApp; AForm: SYSINT); safecall; //创建MDI窗体 SYSINT为unsigned int
procedure CreateComTest(AForm: SYSUINT); safecall;
public
constructor Create(AForm: SYSINT);
end;

这是CreateMdiForm方法的实现:

procedure TComDllTest1.CreateMdiForm(var AApp; AForm: SYSINT);
var
app: TApplication;
af: TForm;
begin
app := TApplication(AApp); //将传进来的Application对象强转
af := TForm(AForm); //将传进来的Form对象强转

Application := app; //将主程序的application付给COM工程的application对象

if not GetMDIForm(Form1) then //GetMDIForm 是判断窗体是否已经创建
begin
Form1 := TForm1.Create(af); //创建子窗体
Form1.FormStyle := fsMDIChild;
Form1.Show;
end;
end;

最后一步也是最关键的就是重写DLL入口函数,如果不重写,关闭主窗体时会报地址错误,我就吃过这亏(我汗......),在DLL工程文件中写:

procedure DLLUnloadProc(Reason: Integer); register;
begin
if (Reason = DLL_PROCESS_DETACH) or (Reason = DLL_THREAD_DETACH) then Application := DLLApp; //DLLApp是在DLL工程文件中定义的全局TApplication对象
//用来保存Application对象
end;

在DLL初始化的位置加入:

DLLApp:=Application; //保留Application
DLLProc := @DLLUnloadProc; //将重写后的入口函数地址付给DLLProc

这样就实现了DLL封装MDI子窗体,当然你要在子窗体上进行比如:数据库的操作,还要有更丰富的界面,肯定会遇到各种问题。以后总结了再说吧!^_^



作者Blog:http://blog.csdn.net/lostpen/
 
谢谢!通过了!
 

Similar threads

后退
顶部