delphi 5 写的DLL在VC中调用的问题(很急,请有空的帮忙看看),100分赠送了!

mybios 2006-04-03 06:22:22
delphi写的DLL在VC中调用后,VC传入的PChar出现写冲突的错误,如何解决?
delphi中导出函数定义如下:
Procedure GetGuestFiscalInfo(RoomNo: PChar; var PrimaryAccount, GuestName: PChar;
var Amount: Double; var FiscalComments: PChar;
var ErrorCode, ErrorMessage: PChar);
然后VC中调用之
char szGuestName[256];
double fAmount;
char szFiscalComments[256];
char szErrorCode[3];
char szErrorMessage[256];
char m_szPrimaryAccount[11];
ext.m_GetGuestFiscalInfo("1", m_szPrimaryAccount
, szGuestName
, &fAmount
, szFiscalComments
, szErrorCode
, szErrorMessage); // 此句调用DLL的导出函数
运行过程中很正常,最后将会往szErrorCode和szErrorMessage这两个pChar写错误信息,但是就是在写入的时候出现了project faulted with message : 'access violation at xxxxx.exe: write of address 0xfdcdcdcd'.的错误啊。
各位大大请帮忙解决一下
...全文
208 点赞 收藏 5
写回复
5 条回复
切换为时间正序
当前发帖距今超过3年,不再开放新的回复
发表回复
mybios 2006-04-05
好的。谢谢madyak了。经过去掉多余var后成功接上了。
回复
madyak 2006-04-04
应该是有问题的
Procedure GetGuestFiscalInfo(RoomNo: PChar; PrimaryAccount, GuestName: PChar;
var Amount: Double; FiscalComments: PChar;
ErrorCode, ErrorMessage: PChar); stdcall;
应该把多余的Var去掉,用Var声明后是引用,一般对应C里面的&,多数情况也可以采用指针的方法翻译。PChar本身就是指针,如果再加上个Var,相当于用二级指针了,所以二者的声明应该是没有对应上,如果C的不改的话,你需要把DELPHI中多余的var去掉,只保留var Amount: Double
如果DELPHI的不改,C中可能需要指针和引用复合声明参数
回复
mybios 2006-04-04
to madyak:我传进去的是一个指针,指针跟引用在传递过程中是一样的。char szErrorMessage[256];这个szErrorMessage就是一个char*。

to coreblood:您给的资料好像不是我描述的问题的解决办法啊。


delphi中声明dll导出函数如下:
Procedure GetGuestFiscalInfo(RoomNo: PChar; var PrimaryAccount, GuestName: PChar;
var Amount: Double; var FiscalComments: PChar;
var ErrorCode, ErrorMessage: PChar); stdcall;
里面有这么一段代码,就是这段代码出错的
StrPCopy(ErrorCode,'01');
StrPCopy(ErrorMessage,'无效的房号');


转换成C++的导入函数如下,有没有错?:
typedef void (__stdcall GetGuestFiscalInfo)(char * RoomNo, char * PrimaryAccount, char * GuestName, double * Amount, char * FiscalComments, char * ErrorCode, char * ErrorMessage);
回复
coreblood 2006-04-03
同意
请参考MSDN中关于调用DLL的文档

下面是关于VB中的一些说明


将字符串传递到 DLL 过程


通常,字符串应该使用 ByVal 方式传递到 APIs。Visual Basic 使用被称为 BSTR 的 String 数据类型,它是由自动化(以前被称为 OLE自动化)定义的数据类型。一个 BSTR 由头部和字符串组成,头部包含了字符串的长度信息,字符串中可以包含嵌入的 null 值。BSTR 是以指针的形式进行传递的,因而 DLL 过程能够修改字符串。(指针是一个变量,包含另外一个变量的内存地址,而不是数据。) BSTR 是 Unicode 的,即每个字符需要两个字节。BSTR 通常以两字节的 null 字符结束。

图 1.2 BSTR 类型(每一个框表示两个字节)

DLL 中的大部分过程(以及 Window API 中的所有过程)能够识别 LPSTR 类型,这是指向标准的以 null 结束的 C 语言字符串的指针,它也被称为 ASCIIZ 字符串。LPSTR 没有前缀。下图显示了一个指向 ASCIIZ 字符串的 LPSTR。

图 1.3 LPSTR 类型

如果 DLL 过程需要一个 LPSTR(指向以 null 结束的字符串的指针)作为参数,可以将 BSTR 以使用值方式传递给它。因为指向 BSTR 的指针实际指向以 null 值结束的字符串的第一个数据字节,对于 DLL 过程来说,它就是一个 LPSTR。

例如,sndPlaySound 函数接受一个数字声音 (.wav) 文件名,然后演奏该文件。

Private Declare Function sndPlaySound Lib "winmm.dll" _
Alias "sndPlaySoundA" (ByVal lpszSoundName As String, _
ByVal uFlags As Long) As Long

因为该过程的字符串参数被声明为 ByVal,Visual Basic 将传递一个 BSTR,该 BSTR 指向第一个数据字节:

Dim SoundFile As String, ReturnLength As Long
SoundFile = Dir("c:\Windows\System\" & "*.wav")
Result = sndPlaySound(SoundFile, 1)

通常,如果 DLL 过程需要 LPSTR 参数,那么使用 ByVal 关键字。如果 DLL 需要得到指向 LPSTR 的指针,则使用引用方式传递 Visual Basic 字符串。

如果要将二进制数据传递到 DLL 过程,可以将变量作为 Byte 数据类型的数组传递,不要将其作为 String 变量。字符串是假定用来包含字符的,如果将二进制数据作为 String 变量传递,外部程序可能无法正确读入数据。

假设声明了一个字符串变量,但没有初始化它,如果将其以使用值方式传递到 DLL,该字符串变量将作为 NULL 传递,而不是作为空字符串 ("")。为了消除代码中的混淆,如果要将 NULL 传递到 LPSTR 参数,请使用 vbNullString 常数。

将字符串传递到使用自动化的 DLL
某些 DLL 是专门使用 BSTR 等自动化数据类型的,它们利用了自动化提供的若干过程。

因为 Visual Basic 使用自动化数据类型作为自己的数据类型,所以能够使用引用方式将 Visual Basic 参数传递到需要自动化数据类型的任何 DLL。因此,如果 DLL 过程需要以 Visual Basic 字符串作为参数,就不必用 ByVal 关键字来声明参数,除非该过程确实需要以使用值方式传递字符串。

某些 DLL 过程可以返回字符串到调用它的过程。除非 DLL 函数是专门为自动化数据类型而编写的,否则它将不能返回字符串。如果确实能够返回字符串,该 DLL 可能会提供对过程进行描述的类型库。请参考该 DLL 的有关文档。

详细信息 关于自动化数据类型,请参阅 Microsoft Press 出版的 OLE 2 PROGRAMMER'S REFERENCE。

修改字符串参数的过程
DLL 过程能够修改作为参数输入的字符串变量的数据。不过,如果修改后的数据超过了原来的长度,过程的修改将越界(越过字符串的结尾),这可能会毁坏其它的数据。

要避免这个问题,一种办法是使字符串参数足够长,从而使 DLL 过程无法超出字符串的尾部。例如,GetWindowsDirectory 过程在第一个参数中返回了 Windows 目录的路径:

Declare Function GetWindowsDirectory Lib "kernel32" _
Alias "GetWindowsDirectoryA" (ByVal lpBuffer As _
String, ByVal nSize As Long) As Long

在调用该过程时,为了安全起见,先使用 String 函数在字符串中填充 255 个空字符(二进制的 0),只要返回的路径少于 255 个字符,就不会出问题:

Path = String(255, vbNullChar)
ReturnLength = GetWindowsDirectory(Path, Len(Path))
Path = Left(Path, ReturnLength)

另一个办法是将字符串定义为定长的:

Dim Path As String * 255
ReturnLength = GetWindowsDirectory(Path, Len(Path))

上述方法的目的只有一个:创建一个固定长度的字符串,使之能够包含过程可能产生的最长的字符串。

注意 Windows API 的 DLL 过程通常不需要超过 255 个字符的字符串缓冲区。尽管这对于其它的许多库也是成立的,为了保险起见,最好参考相应过程的文档。

当 DLL 过程需要内存缓冲区时,既可以使用适合的数据类型,也可以使用字节数据类型的数组。
回复
madyak 2006-04-03
Procedure GetGuestFiscalInfo(RoomNo: PChar; var PrimaryAccount, GuestName: PChar;
var Amount: Double; var FiscalComments: PChar;
var ErrorCode, ErrorMessage: PChar);stdcall;//要加stdcall指示字

//你的声明中,很多是指针加引用,一起上的,C的调用中最好与之相呼应


ext.m_GetGuestFiscalInfo("1", &m_szPrimaryAccount
, &szGuestName
, &fAmount
, &szFiscalComments
, &szErrorCode
, &szErrorMessage);
回复
相关推荐
发帖
语言基础/算法/系统设计
创建于2007-08-02

1.6w+

社区成员

Delphi 语言基础/算法/系统设计
申请成为版主
帖子事件
创建了帖子
2006-04-03 06:22
社区公告
暂无公告