Python ctypes - dll 函数接受结构体崩溃
我需要在Windows XP系统下访问一个POS终端。我使用的是Python 2.7。
我加载的DLL中有一个关键的函数负责处理支付,它需要两个指向结构体的指针,但在调用时出现崩溃,返回值是1(通信错误),而且没有其他的错误信息。
需要注意的是,当调用支付函数时,并不是所有的POSData结构体中的元素都有值。还有其他我尝试过的函数(比如GetVersion)是可以正常工作的。
以下是相关的规格说明和我的代码:
typedef struct
{
char IPAddress[16]; //xxx.xxx.xxx.xxx
int Port;
} TETHParameters;
typedef struct
{
char TerminalId[8+1];
char AcquirerId[11+1];
char TransactionType[3+1];
char TransactionResult[2+1];
char KODescription[24+1];
char CardType[1+1];
char STAN[6+1];
char PAN[19+1];
char AuthorizationCode[6+1];
char OperationNumber[6+1];
char DataTrs[7+1];
} TPOSData;
typedef struct
{
char Amount[8+1];
char ECRId[8+1];
char PaymentType[1+1];
char TerminalId[8+1];
} TECRData;
__declspec(dllexport) void IAE17_GetVersion(char *Version);
__declspec(dllexport) void IAE17_InitEth(TETHParameters *ETHParameters);
__declspec(dllexport) void IAE17_Free(void);
__declspec(dllexport) int IAE17_Payment(TECRData *ECRData, TPOSData *POSData);
from ctypes import *
#da python 3.x sara' configparser
import ConfigParser
import logging
from time import localtime, strftime
#STRUTTURE DATI
class TETHParameters(Structure):
_fields_ = [("IPAddress" , c_char_p), ("Port" , c_int )]
class TECRData(Structure):
_fields_ = [("Amount" , c_char_p),
("ECRId", c_char_p),
("PaymentType", c_char_p),
("TerminalId", c_char_p),
("Contract", c_char_p),
("PreauthorizationCode", c_char_p),
("STAN", c_char_p),
("Ticket2Ecr", c_char_p)]
class TPOSData(Structure):
_fields_ = [
("TerminalId" , c_char_p),
("AcquirerId" , c_char_p),
("TransactionType" , c_char_p),
("TransactionResult" , c_char_p),
("KODescription" , c_char_p),
("CardType" , c_char_p),
("STAN" , c_char_p),
("POSBalance" , c_char_p),
("BankBalance" , c_char_p),
("PAN" , c_char_p),
("AuthorizationCode" , c_char_p),
("OperationNumber" , c_char_p),
("AmountAuth" , c_char_p),
("PreauthorizationCode" , c_char_p),
("ActionCode" , c_char_p),
("DataTrs" , c_char_p),
("AmountEcho" , c_char_p),
("Ticket" , c_char_p)
]
ECRData = TECRData( ECRId = c_char_p( '012345678' ),
Amount = c_char_p( '00000000') ,
TerminalID = c_char_p( '01234567' ),
PaymentType = c_char_p ("0")
)
POSData = TPOSData( KODescription = c_char_p(' '),
TerminalId = c_char_p(' '),
AcquirerId = c_char_p(' '),
TransactionType = c_char_p(' '),
TransactionResult = c_char_p(' '),
CardType = c_char_p(' '),
STAN = c_char_p(' '),
PAN = c_char_p(' '),
AuthorizationCode = c_char_p(' '),
OperationNumber = c_char_p(' '),
DataTrs = c_char_p(' ')
)
ETHParameters = TETHParameters( IPAddress = c_char_p( '192.168.127.190' ) , Port = c_int(45119))
iae17 = windll.LoadLibrary('iae17')
iae17.IAE17_InitEth( byref( ETHParameters) )
result = iae17.IAE17_Payment( byref(ECRData), byref(POSData))
print result
1 个回答
5
c_char_p
是 C 语言中 char *
的直接对应。也就是说,当你的 C 结构是
typedef struct
{
char TerminalId[8+1];
char AcquirerId[11+1];
char TransactionType[3+1];
&c
而你在 ctypes 中创建的结构,实际上是等同于
typedef struct
{
char* TerminalId;
char* AcquirerId;
char* TransactionType;
&c
这两者之间的差别非常大。你为什么不使用 ctypes 的 数组,而是用“指针”?