打开 XML SDK - 使用 MemoryStream 作为 docx 持有者从模板 (dotx) 创建 docx

Open XML SDK - Create docx from template (dotx) using MemoryStream as docx holder

我正在尝试使用 OpenXML 2.5 库从 word 模板 (.dotx) 创建 .docx 文件。更确切地说,我正在尝试遵循 this ,但我想使用 MemoryStream 而不是文件路径打开文档。

所以在上面的link中,方法是这样的:将模板文件复制到磁盘上的一个新文件中。以 WordProcessingDocument 形式打开新文件,将其类型更改为文档并添加对模板的引用(我不确定为什么需要这样做...)

我需要的是。从数据库中获取我的模板作为字节数组。将数组复制到流中并将其作为 WordProcessingDocument 打开。然后做和上面一样的事情。 我不想使用磁盘上的临时文件等等。我不明白的是为什么我应该在新文档中引用模板文件(实际上,检查 docx 存档中的 xml 文件,它只是添加了一个包含模板名称的 xml 节点...为什么?)

现在,我的问题是,如果我使用 MemoryStream 而不是文件的物理路径,生成的 docx 已损坏 。我收到“Microsoft Office 无法打开此文件,因为某些部分丢失或无效”。 如果我使用临时磁盘文件,它就可以工作。 为什么会这样?

简化后,我的代码如下所示:

byte[] result = null;
byte[] templateBytes = System.IO.File.ReadAllBytes("template.dotx");
using (MemoryStream templateStream = new MemoryStream())
{
    templateStream.Write(templateBytes, 0, (int)templateBytes.Length);

    using (WordprocessingDocument doc = WordprocessingDocument.Open(templateStream, true))
    {
        doc.ChangeDocumentType(WordprocessingDocumentType.Document);
        var mainPart = doc.MainDocumentPart;

        //Here I should create an AttachedTemplate object and assign a relationship id to it, which also implies
        //setting a "reference" to the template ? I'm not using files so there's nothing to "reference" anyway
        // If it's doing more than just putting a visual reference to the template file, then why limit the way of referencing the template
        // only by using the Uri to the file? Please take a look at the above link to see what I'm referring to.

        //Write some text in the file ....
        mainPart.Document.Save();

        File.WriteAllBytes("Result.docx", templateStream.ToArray());
    }
}

这让我得到了一个损坏的 docx 文件。如果我从 MemoryStream 切换到从磁盘打开新的 docx 文件(使用 WordProcessingDocument class 的其他构造函数),它就可以工作。 我没有做错什么(比如在处理内存流后试图将结果写入文件),所以它应该给我相同的结果。

我的代码有什么问题?

谢谢

如果您从内部 using 块中取出代码的最后一行 File.WriteAllBytes,它应该可以正常工作。只需将该行放在外部 using 块的右大括号之前。