关于在dll中从一个from调另一个from中的控件属性方法的问题(很急呀) (100分)

  • 主题发起人 主题发起人 fjx_jwf
  • 开始时间 开始时间
F

fjx_jwf

Unregistered / Unconfirmed
GUEST, unregistred user!
我在一dll中先启动一个form1,上面有button1和label1两个控件
然后在button1中有如下代码:
appliction.create(tform2,form2);
form2.showmodal;
然后在form2中调用form1.label1.caption时报错
另外即便在form1内写如下语句也报错
如:showmessage(form1.label1.caption);与前面报同样的错
但如果写成showmessage(label1.caption);又是正确的;
我刚用dll,不是很熟悉,那位大哥帮帮忙呀
 
补充一下,form2也是在同一个dll中

没人理呀
 
是没人看懂还是太难呀,应该不是很难吧?
也就是在dll中有多个form,多个form之间的引用呀

急死了
 
引用与普通Form都是一样的,你报的具体是什么错误?
 
用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编译成二进制代码也如此。
 
pihome你的邮箱是多少我发过来,你帮我瞧一下
 
pihome@pihome.net
 
在form2中调用form1.label1.caption之前加一个
appliction.create(tform1,form1);
showmessage(form1.label1.caption);
form1.free;

 
to plhome:已经发出来了,请查收
 
我试了一下用chenshaizi的方法确实可行,但是当
application.createform(tform1,form1);
后就又显示了form1,也就是说屏幕上就有两个form1窗口很不好看呀
还有其它方法吗
 
补充一下:form1为mdichild窗口
 
to chenshaizi:我又试了一下,你的方法只可以读出,不能写入
application.createform(tform1,form1);
form1.label1.caption:='ok';//这句就不起作用了,相当于你这个form1只是前一个form1的副本
form1.free;
 
这样修改一下,走了,吃饭去了:)
procedure TForm2.Button1Click(Sender: TObject);
var
i: Integer;
begin
for i:=0 to Screen.FormCount-1 do
if Screen.Forms is TForm1 then
showmessage(TForm1(Screen.Forms).Button2.Caption);
end;

procedure TForm2.FormClose(Sender: TObject
var Action: TCloseAction);
begin
action:=cafree;
Form2:=Nil;
end;
 
还有其它方法吗?
我将它写成函数getform,然后在form2中定义一个
变量mfrm:tform1;
function tform2.getform:tform1;
var
i: Integer;
begin
for i:=0 to Screen.FormCount-1 do
if Screen.Forms is tform1 then
result:=tform1(Screen.Forms);
end;
然后在form1的create事件中写:mfrm:=getform;
在form2中的button1的click中写:
with mfrm do
begin
label1.caption:=label2.caption+label3.caption;
end;
仍旧要报错
 
实际上你最开始的程序中测试一下就会发现
if Assigned(Form1) then
ShowMessage(Form1.Button2.Caption)
//根本不会发生
可以这样处理一下:
Form1中
Form2:=TForm2.Create(Self);
Form2中
TForm2 = class(TForm)
public
constructor Create(AOwner: TComponent)
override;
end;

var
ParentForm: TForm1;

constructor TForm2.Create(AOwner: TComponent);
begin
inherited;

ParentForm:=TForm1(AOwner);
end;

然后
ShowMessage(ParentForm.Button2.Caption);
 
to pihome:你的方法非常成功
另外你能告诉我为什么在form1中
Form2:=TForm2.Create(Self);
form2.showmodal;
form2.free;//不加这句就完全正确,加上后在程序退出时就会报错,是什么原因?
非常感谢
 
是不是还保留了我第一次给你的修改建议?
procedure TForm2.FormClose(Sender: TObject
var Action: TCloseAction);
begin
action:=cafree;
Form2:=Nil
//把这句去掉:)
end;
 
我没保留action:=cafree;
Form2:=Nil;
因为form2是normal窗口
我发现问题出在:如果在主程序exe中调dll中的form1,然后在form1中调出form2后如果后面没有form2.free就完全正确,加上form2.free后主程序exe关闭时就会报错
 

Similar threads

S
回复
0
查看
3K
SUNSTONE的Delphi笔记
S
S
回复
0
查看
2K
SUNSTONE的Delphi笔记
S
D
回复
0
查看
1K
DelphiTeacher的专栏
D
后退
顶部