我正在尝试在 .NET 中生成一个电子表格,当我的经理不在办公室时,他会在他的 iPad 上打开它。
电子表格在 Windows PC 上可以正常打开,但是当尝试在 iPad 上打开时它显示“读取文档时发生错误”(非常有用!)
通过使用 OpenXML SDK 生产力工具上的“比较”功能与在 iPad 上确实打开的文档,并通过在记事本中手动编辑错误文档的 XML 文件,我缩小了范围它向下到文件 xl/_rels/workbook.xml.rels,它存储工作簿中各部分的关系。
这是我用来生成 WorkbookPart 和引用的代码
WorkbookPart workbookPart1 = document.AddWorkbookPart();
WorkbookStylesPart workbookStylesPart1 = workbookPart1.AddNewPart<WorkbookStylesPart>("rId3");
ThemePart themePart1 = workbookPart1.AddNewPart<ThemePart>("rId2");
WorksheetPart worksheetPart1 = workbookPart1.AddNewPart<WorksheetPart>("rId1");
我的代码生成以下输出,在 iPad 上无法打开。
<?xml version="1.0" encoding="utf-8" ?>
<Relationships xmlns="http://schemas.openxmlformats.org/package/2006/relationships">
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles" Target="/xl/styles.xml" Id="rId3" />
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/theme" Target="/xl/theme/theme.xml" Id="rId2" />
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet" Target="/xl/worksheets/sheet.xml" Id="rId1" />
</Relationships>
如果我更改 Target 属性的值以使用相对引用路径,提供以下输出,那么它确实会在 iPad 上打开。
<?xml version="1.0" encoding="utf-8" ?>
<Relationships xmlns="http://schemas.openxmlformats.org/package/2006/relationships">
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles" Target="styles.xml" Id="rId3" />
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/theme" Target="theme/theme.xml" Id="rId2" />
<Relationship Type="http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet" Target="worksheets/sheet.xml" Id="rId1" />
</Relationships>
所以问题是:
如何更改我的 .NET 代码,以便它输出具有相对路径的第二个版本的 XML。
感谢所有帮助!
最佳答案
我花了很多时间对此进行研究,并认为我会分享我的结果。 OpenXML 似乎在做两件事。 1. content_types.xml 文件缺少工作簿条目 2. xl/_rels/workbook.xml.rels 文件使用完全相对路径。
Excel 本身可以很好地打开文件,但我在 iPad 上尝试了各种应用程序,但都失败了。所以我不得不使用以下代码自己手动修复文件。它假定文件的全部内容作为流传入并使用 DotNetZip 打开和操作。希望这段代码对其他人有帮助!
private Stream ApplyOpenXmlFix(Stream input)
{
const string RELS_FILE = @"xl/_rels/workbook.xml.rels";
const string RELATIONSHIP_ELEMENT = "Relationship";
const string CONTENT_TYPE_FILE = @"[Content_Types].xml";
const string XL_WORKBOOK_XML = "/xl/workbook.xml";
const string TARGET_ATTRIBUTE = "Target";
const string SUPERFLUOUS_PATH = "/xl/";
const string OVERRIDE_ELEMENT = "Override";
const string PARTNAME_ATTRIBUTE = "PartName";
const string CONTENTTYPE_ATTRIBUTE = "ContentType";
const string CONTENTTYPE_VALUE = "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet.main+xml";
XNamespace contentTypesNamespace = "http://schemas.openxmlformats.org/package/2006/content-types";
XNamespace relsNamespace = "http://schemas.openxmlformats.org/package/2006/relationships";
XDocument xlDocument;
MemoryStream memWriter;
try
{
input.Seek(0, SeekOrigin.Begin);
ZipFile zip = ZipFile.Read(input);
//First we fix the workbook relations file
var workbookRelations = zip.Entries.Where(e => e.FileName == RELS_FILE).Single();
xlDocument = XDocument.Load(workbookRelations.OpenReader());
//Remove the /xl/ relative path from all target attributes
foreach (var relationship in xlDocument.Root.Elements(relsNamespace + RELATIONSHIP_ELEMENT))
{
var target = relationship.Attribute(TARGET_ATTRIBUTE);
if (target != null && target.Value.StartsWith(SUPERFLUOUS_PATH))
{
target.Value = target.Value.Substring(SUPERFLUOUS_PATH.Length);
}
}
//Replace the content in the source zip file
memWriter = new MemoryStream();
xlDocument.Save(memWriter, SaveOptions.DisableFormatting);
memWriter.Seek(0, SeekOrigin.Begin);
zip.UpdateEntry(RELS_FILE, memWriter);
//Now we fix the content types XML file
var contentTypeEntry = zip.Entries.Where(e => e.FileName == CONTENT_TYPE_FILE).Single();
xlDocument = XDocument.Load(contentTypeEntry.OpenReader());
if (!xlDocument.Root.Elements().Any(e =>
e.Name == contentTypesNamespace + OVERRIDE_ELEMENT &&
e.Attribute(PARTNAME_ATTRIBUTE) != null &&
e.Attribute(PARTNAME_ATTRIBUTE).Value == XL_WORKBOOK_XML))
{
//Add in the missing element
var overrideElement = new XElement(
contentTypesNamespace + OVERRIDE_ELEMENT,
new XAttribute(PARTNAME_ATTRIBUTE, XL_WORKBOOK_XML),
new XAttribute(CONTENTTYPE_ATTRIBUTE, CONTENTTYPE_VALUE));
xlDocument.Root.Add(overrideElement);
//Replace the content
memWriter = new MemoryStream();
xlDocument.Save(memWriter, SaveOptions.DisableFormatting);
memWriter.Seek(0, SeekOrigin.Begin);
zip.UpdateEntry(CONTENT_TYPE_FILE, memWriter);
}
Stream output = new MemoryStream();
//Save file
zip.Save(output);
return output;
}
catch
{
//Just in case it fails, return the original document
return input;
}
}
关于.net - 在 .NET 中创建的 OpenXML 电子表格无法在 iPad 中打开,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10929054/
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou
我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案
我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur
我正在尝试找出如何为我的Ruby项目创建一种“无类DSL”,类似于在Cucumber步骤定义文件中定义步骤定义或在Sinatra应用程序中定义路由。例如,我想要一个文件,其中调用了我的所有DSL函数:#sample.rbwhen_string_matches/hello(.+)/do|name|call_another_method(name)end我认为用我的项目特有的一堆方法污染全局(内核)命名空间是一种不好的做法。因此方法when_string_matches和call_another_method将在我的库中定义,并且sample.rb文件将以某种方式在我的DSL方法的上下文中
有这些railscast。http://railscasts.com/episodes/218-making-generators-in-rails-3有了这个,你就会知道如何创建样式表和脚手架生成器。http://railscasts.com/episodes/216-generators-in-rails-3通过这个,您可以了解如何添加一些文件来修改脚手架View。我想把两者结合起来。我想创建一个生成器,它也可以创建脚手架View。有点像RyanBates漂亮的生成器或web_app_themegem(https://github.com/pilu/web-app-theme)。我
这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Rubysyntaxquestion:Rational(a,b)andRational.new!(a,b)我正在阅读ruby镐书,我对创建有理数的语法感到困惑。Rational(3,4)*Rational(1,2)产生=>3/8为什么Rational不需要new方法(我还注意到例如我可以在没有new方法的情况下创建字符串)?
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里
假设我有一个在Ruby中看起来像这样的哈希:{:ie0=>"Hi",:ex0=>"Hey",:eg0=>"Howdy",:ie1=>"Hello",:ex1=>"Greetings",:eg1=>"Goodday"}有什么好的方法可以将它变成如下内容:{"0"=>{"ie"=>"Hi","ex"=>"Hey","eg"=>"Howdy"},"1"=>{"ie"=>"Hello","ex"=>"Greetings","eg"=>"Goodday"}} 最佳答案 您要求一个好的方法来做到这一点,所以答案是:一种您或同事可以在六个月后理解
我很好奇.NET将如何影响Python和Ruby应用程序。用IronPython/IronRuby编写的应用程序是否会非常特定于.NET环境,以至于它们实际上将变得特定于平台?如果他们不使用任何.NET功能,那么IronPython/IronRuby相对于非.NET同类产品的优势是什么? 最佳答案 我不能说任何关于IronRuby的东西,但是大多数Python实现(如IronPython、Jython和PyPy)都试图尽可能忠实于CPython实现。不过,IronPython正在迅速成为这方面的佼佼者之一,并且在PlanetPyth