2

我当然希望这不是一个已经回答的问题或一个愚蠢的问题。最近我一直在用几种仪器编程。尝试在它们之间进行通信以创建测试程序。但是,当我尝试调用从仪器 DLL 文件中“屏蔽”出来的函数时,我遇到了一些特定仪器的问题。当我使用交互式 python shell 时,它可以完美运行(尽管它有很多单词破坏)。但是当我以面向对象的方式实现这些功能时,程序失败了,实际上它并没有失败,它只是什么都不做。这是第一个调用的方法:(导入了 ctypes 和 ctypes.util)

    def init_hardware(self):
    """ Inits the instrument """
    self.write_log("Initialising the automatic tuner")
    version_string = create_string_buffer(80)
    self.error_string = create_string_buffer(80)
    self.name = "Maury MT982EU"
    self.write_log("Tuner DLL path: %s", find_library('MLibTuners'))
    self.maury = WinDLL('MlibTuners')
    self.maury.get_tuner_driver_version(version_string)
    if (version_string.value == ""):
        self.write_log("IMPORTANT: Error obtaining the driver version")
    else:
        self.write_log("Version number of the DLL: %s" % version_string.value)
    self.ThreeTypeLong = c_long * 3

现在效果很好,一切都很完美,我得到了完美的日志条目。但是当我尝试在程序中进一步运行一个方法时,调用:

def add_tuner_and_controller(self, name, serial_number, tuner_number=0):
    """ Adds the tuner to the driver object, controller is inside the tuner """
    self.write_log("Adding tuner %d and the built-in controller" % tuner_number)
    TempType = self.ThreeTypeLong()
    self.maury.add_controller(c_short(tuner_number), c_char_p(self.file_path), c_char_p(name), c_int(0), c_int(0), 
                              c_long(0), c_short(serial_number), self.error_string)
    self.maury.add_tuner(c_short(tuner_number), c_char_p(name), c_short(serial_number), c_short(0),
                            c_short(1), pointer(c_double()), TempType, pointer(c_double()), pointer(c_double()),
                            pointer(c_double()), self.error_string)

程序突然停止工作/继续运行,调用“self.maury”行时没有任何反应。当我将所有内容都放在 init_hardware 方法中时,它可以完美运行,所以我猜测存在轻微的内存“错误”或面向目标的结构。我真的希望它保持这种状态,无论如何以这种方式隔离功能?还是我必须将自己限制在一大段代码中?


编辑:
文档信息
[图例:星号表示指针,括号表示数组]

add_tuner 函数在调谐器驱动程序对象中添加或更新一个调谐器。

short add_tuner(short tuner_number, char model[ ], short serial_number, short ctlr_num, short ctlr_port, short *no_of_motors, long max_range[ ], double *fmin, double *fmax, double *fcrossover, char error_string[ ])

输出no_motors, max_range (array of three numbers), fmin, fmax, fcrossover,error_string (80+ characters long), function-return->Error flag



add_controller 函数在调谐器驱动对象中添加或更新一个控制器

short add_controller(short controller_number, char driver[ ], char model[ ], int timeout, int address, long delay_ms, char error_string[ ])

输出error_string, function-return->Error flag

4

3 回答 3

1

我不确定您的确切问题,但这里有几个一般提示:

对于那些在构造函数之外调用的函数,我强烈建议也将它们设置argtypes在构造函数中。一旦声明了argtypes,就不需要将所有参数强制转换为c_shortc_double等。此外,如果您不小心将不正确的参数传递给 C 函数,Python 将引发运行时错误,而不是在 DLL 中崩溃。

另一个小细节,但您应该在调谐器和控制器中使用x = 0; byref(x)或者代替指针(c_double())。POINTER(c_double)()

我最近也在 Python 2.6 中编写了一些 ctypes 类,我还没有看到像你描述的那样的任何问题。由于显然也没有任何 Python 错误报告,因此我坚信只有一分钟的细节我们都忽略了您的方法中有问题。

于 2009-07-23T11:29:01.680 回答
0

add-controller 或 add-tuner 中的任何参数实际上是否返回值?

强烈建议您定义函数的原型,而不是使用所有参数的强制转换直接调用它们。

我确定您已经阅读过此页面,但您要查看的部分是函数原型。使代码更清晰,更易于跟踪/调试。

另外——正如 Mark Rushakoff 也提到的——在你的电话中使用 pointer(c_double()) 和 like 是非常讨厌的。使用 POINTER() 时我的运气要好得多,并再次建议您将值预先声明为变量,并在函数调用中传递该变量。然后至少你可以在之后检查它的值是否有奇怪的行为。

编辑:所以你的原型和调用看起来像这样:

prototype = WINFUNCTYPE(
    c_int,              # Return value (correct? Guess)
    c_short,            # tuner_number
    c_char_p,           # file_path
    c_char_p,           # name
    c_int,              # 0?
    c_int,              # 0?
    c_long,             # 0?
    c_short,            # serial_number
    c_char_p,           # error_string
)
# 1 input, 2 output, 4 input default to zero (I think; check doc page)
paramflags = (1, 'TunerNumber' ), (1, 'FilePath' ), (1, 'Name' ), ...
AddController = prototype(('add_controller', WinDLL.MlibTuners), paramflags)

那么你的电话就干净多了:

arg1 = 0
arg2 = 0
arg3 = 0
AddController(tuner_number, self.file_path, name, arg1, arg2, arg3, 
    serial_number, self.error_string)
于 2009-07-23T12:55:15.333 回答
0

我发现调用导出的 DLL 中的函数的唯一方法是通过每个方法中的参数来使用 DLL。(程序导出 dll 并在每个调用的方法中将其作为参数发送)。它看起来很丑,但这是我发现为我工作的唯一方式。我什至尝试将 DLL 作为类属性导出。我正在使用的系统非常庞大,所以我猜某处有一些 boboo 代码导致它失败。感谢所有的反馈和提示!

/马自达

于 2009-08-04T08:08:56.183 回答