在 C# 中使用 QPDF

Posted

技术标签:

【中文标题】在 C# 中使用 QPDF【英文标题】:Using QPDF with C# 【发布时间】:2018-05-10 15:11:31 【问题描述】:

我正在尝试翻译这个 qpdf 命令:

qpdf --qdf --object-streams=disable input.pdf editable.pdf

进入我在使用 qpdf dll 时需要的等效方法调用(可从此处获得:https://sourceforge.net/projects/qpdf/)。

我通过 dumpbin 运行 qpdf dll 以获取函数名称,并通过查看包含用于 c++ 项目的头文件,我可以看到函数的参数。

例如,赋予上述 --object-streams 选项所需的函数将(据我所知)是这个函数:

void setObjectStreamMode(qpdf_object_stream_e);

从c++头文件变成:

[DllImport("qpdf21.dll")]
static extern void _ZN10QPDFWriter19setObjectStreamModeE20qpdf_object_stream_e(int stateEnum);

在 C# 文件中。

问题是当我使用上述函数时,我得到一个

AccessViolationException: 试图读取或写入受保护的内存

错误,这让我觉得我需要以某种方式创建一个 QPDF 对象,但我从未使用过面向对象的 pinvokes,所以我不知道如何使该对象在 c# 中可访问。

如果有人已经熟悉在 C# 甚至 C++ 中使用 dll,并且可以告诉我调用正确的函数来复制命令,我将不胜感激!

【问题讨论】:

解决内存问题后,我将删除编辑并回答问题。 发现了问题,我的清理密码错误。 【参考方案1】:

我已经弄明白了,下面的代码复制了命令,结果我查看了错误的头文件:

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern IntPtr qpdf_init();

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_cleanup(ref IntPtr qpdfData);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_read(IntPtr qpdfdata, [MarshalAs(UnmanagedType.LPStr)] string fileName, IntPtr password);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_set_object_stream_mode(IntPtr qpdf, int mode);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_set_qdf_mode(IntPtr qpdf, int value);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_init_write(IntPtr qpdf, [MarshalAs(UnmanagedType.LPStr)] string fileName);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_write(IntPtr qpdf);

static void Main(string[] args)

    //call init
    IntPtr qpdfData = qpdf_init();

    //call read (which gets and processes the input file)
    //0 result == good
    int result = qpdf_read(qpdfData, @"scan.pdf", IntPtr.Zero);

    //call init_write
    result = qpdf_init_write(qpdfData, @"scanEditable.pdf");

    //set write options
    //disable object stream mode
    qpdf_set_qdf_mode(qpdfData, 1);
    qpdf_set_object_stream_mode(qpdfData, 0);

    //call write
    result = qpdf_write(qpdfData);

    //call cleanup
    qpdf_cleanup(ref qpdfData);

【讨论】:

【参考方案2】:

看来您在这里找到了一个很好的答案。您已经发现了 C API,它旨在帮助通过 DLL 从 C++ 以外的语言使用 QPDF。 C API 主要记录在qpdf-c.h header file 中。您也可以在Using Other Languages section of the manual 中找到一些信息。 C API 没有公开 qpdf 的 C++ 库的全部功能。如果您发现遗漏的部分,请随时在github 创建问题。我尝试在添加新接口时更新 C API,但我不会对每个接口都这样做,并且 CLI 中的某些功能是直接在工具中实现的,不会映射到单个库函数。

如果 C API 对您的用例来说不够丰富,也可以使用声明为 extern "C" 的一些函数编写您自己的 C++ 类,导出它们,然后构建一个额外的 DLL,您可以按照自己的方式使用上面找到的。您可以查看 qpdf-c.cc 作为其工作原理的示例。

【讨论】:

以上是关于在 C# 中使用 QPDF的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Mac 10.8.3 上安装 qpdf?

Window下编译qtpdfium

在 C++ 中使用 C# 接口或在 C# 中使用 C++ 接口

如何在 C# 的泛型中使用扩展

如何在 C# 中使用 libcinder?

在 C# 中使用 GitLabCI