2

我正在尝试在 C# 中使用来自非托管 DLL 的 C 函数。

该函数的签名是:

const char*  CDECL get_lame_version       ( void );

我以这种方式导入函数:

[DllImport("libmp3lame.dll")]
static extern string get_lame_version();

如果我调用此函数,但在调用之前中断,然后按 F5,则会引发 AccessViolationException。

首先执行在调用之前中断:

这里执行在调用之前中断

然后我按 F5 有一个例外:

AccessViolationException

如果在调用之后执行中断,则不会引发异常:

在此处输入图像描述

所以我的问题是:我的代码有什么问题吗?如果不是,那是怎么回事?

编辑

下面是 get_lame_version 的定义:

/*! Get the LAME version string. */
/*!
  \param void
  \return a pointer to a string which describes the version of LAME.
*/
const char *
get_lame_version(void)
{                       /* primary to write screen reports */
    /* Here we can also add informations about compile time configurations */

#if   LAME_ALPHA_VERSION
    static /*@observer@ */ const char *const str =
        STR(LAME_MAJOR_VERSION) "." STR(LAME_MINOR_VERSION) " "
        "(alpha " STR(LAME_PATCH_VERSION) ", " __DATE__ " " __TIME__ ")";
#elif LAME_BETA_VERSION
    static /*@observer@ */ const char *const str =
        STR(LAME_MAJOR_VERSION) "." STR(LAME_MINOR_VERSION) " "
        "(beta " STR(LAME_PATCH_VERSION) ", " __DATE__ ")";
#elif LAME_RELEASE_VERSION && (LAME_PATCH_VERSION > 0)
    static /*@observer@ */ const char *const str =
        STR(LAME_MAJOR_VERSION) "." STR(LAME_MINOR_VERSION) "." STR(LAME_PATCH_VERSION);
#else
    static /*@observer@ */ const char *const str =
        STR(LAME_MAJOR_VERSION) "." STR(LAME_MINOR_VERSION);
#endif

    return str;
}
4

2 回答 2

1

pinvoke 签名错误。它应该是:

[DllImport("libmp3lame.dll", CallingConvention=CallingConvention.Cdecl)]
static extern IntPtr get_lame_version();

要调用它,您需要这样做;

string version = Marshal.PtrToStringAnsi(get_lame_version());

您不能依赖 p/invoke 编组器来编组字符串值,因为它不拥有该字符串。DLL 拥有该字符串。

更重要的是,您应该为 LAME DLL 函数指定调用约定,否则您最终会得到默认的 pinvoke 调用约定,stdcall. 对于没有参数的函数来说,这并不重要,但这是一个好习惯。

于 2012-04-25T11:40:33.537 回答
0

访问冲突是由非托管代码引起的,而不是您的 C# 代码。如果没有看到非托管代码,很难说问题出在哪里。

于 2012-04-25T10:00:24.200 回答