我正在使用 JAXB 的partial-unmarshalling 示例,但我无法解码不在根级别的 XML 元素(因为它们不没有 @XmlRootElement 标签)。在我的示例中,我尝试读取 shipTo-Element 而不是 purchaseOrder-Element。
通常我会使用 JAXBElement unmarshal(Source source,Class declaredType) 但由于该示例使用的是 UnmarshallerHandler 和 XMLFilterImpl,我不知道在哪里告诉 Jaxb 它应该使用哪个类。
我的错误消息是:由:javax.xml.bind.UnmarshalException:意外元素(uri:“”,本地:“shipTo”)引起的。预期的元素是 <{}comment>、<{}purchaseorder>、<{}purchaseorders>
我在谷歌上搜索了很多,但还没有找到任何有用的东西。
这是来自 JaxB 网页的示例代码:
Main.java
public class Main {
public static void main( String[] args ) throws Exception {
// create JAXBContext for the primer.xsd
JAXBContext context = JAXBContext.newInstance("primer");
// create a new XML parser
SAXParserFactory factory = SAXParserFactory.newInstance();
factory.setNamespaceAware(true);
XMLReader reader = factory.newSAXParser().getXMLReader();
// prepare a Splitter
Splitter splitter = new Splitter(context);
// connect two components
reader.setContentHandler(splitter);
for( int i=0; i<args.length; i++ ) {
// parse all the documents specified via the command line.
// note that XMLReader expects an URL, not a file name.
// so we need conversion.
reader.parse(new File(args[i]).toURL().toExternalForm());
}
}
Splitter.java
public class Splitter extends XMLFilterImpl {
public Splitter( JAXBContext context ) {
this.context = context;
}
/**
* We will create unmarshallers from this context.
*/
private final JAXBContext context;
public void startElement(String namespaceURI, String localName, String qName, Attributes atts)
throws SAXException {
if( depth!= 0 ) {
// we are in the middle of forwarding events.
// continue to do so.
depth++;
super.startElement(namespaceURI, localName, qName, atts);
return;
}
if( namespaceURI.equals("") && localName.equals("purchaseOrder") ) {
// start a new unmarshaller
Unmarshaller unmarshaller;
try {
unmarshaller = context.createUnmarshaller();
} catch( JAXBException e ) {
// there's no way to recover from this error.
// we will abort the processing.
throw new SAXException(e);
}
unmarshallerHandler = unmarshaller.getUnmarshallerHandler();
// set it as the content handler so that it will receive
// SAX events from now on.
setContentHandler(unmarshallerHandler);
// fire SAX events to emulate the start of a new document.
unmarshallerHandler.startDocument();
unmarshallerHandler.setDocumentLocator(locator);
Enumeration e = namespaces.getPrefixes();
while( e.hasMoreElements() ) {
String prefix = (String)e.nextElement();
String uri = namespaces.getURI(prefix);
unmarshallerHandler.startPrefixMapping(prefix,uri);
}
String defaultURI = namespaces.getURI("");
if( defaultURI!=null )
unmarshallerHandler.startPrefixMapping("",defaultURI);
super.startElement(namespaceURI, localName, qName, atts);
// count the depth of elements and we will know when to stop.
depth=1;
}
}
public void endElement(String namespaceURI, String localName, String qName) throws SAXException {
// forward this event
super.endElement(namespaceURI, localName, qName);
if( depth!=0 ) {
depth--;
if( depth==0 ) {
// just finished sending one chunk.
// emulate the end of a document.
Enumeration e = namespaces.getPrefixes();
while( e.hasMoreElements() ) {
String prefix = (String)e.nextElement();
unmarshallerHandler.endPrefixMapping(prefix);
}
String defaultURI = namespaces.getURI("");
if( defaultURI!=null )
unmarshallerHandler.endPrefixMapping("");
unmarshallerHandler.endDocument();
// stop forwarding events by setting a dummy handler.
// XMLFilter doesn't accept null, so we have to give it something,
// hence a DefaultHandler, which does nothing.
setContentHandler(new DefaultHandler());
// then retrieve the fully unmarshalled object
try {
JAXBElement<PurchaseOrderType> result =
(JAXBElement<PurchaseOrderType>)unmarshallerHandler.getResult();
// process this new purchase order
process(result.getValue());
} catch( JAXBException je ) {
// error was found during the unmarshalling.
// you can either abort the processing by throwing a SAXException,
// or you can continue processing by returning from this method.
System.err.println("unable to process an order at line "+
locator.getLineNumber() );
return;
}
unmarshallerHandler = null;
}
}
}
public void process( PurchaseOrderType order ) {
System.out.println("this order will be shipped to "
+ order.getShipTo().getName() );
}
/**
* Remembers the depth of the elements as we forward
* SAX events to a JAXB unmarshaller.
*/
private int depth;
/**
* Reference to the unmarshaller which is unmarshalling
* an object.
*/
private UnmarshallerHandler unmarshallerHandler;
/**
* Keeps a reference to the locator object so that we can later
* pass it to a JAXB unmarshaller.
*/
private Locator locator;
public void setDocumentLocator(Locator locator) {
super.setDocumentLocator(locator);
this.locator = locator;
}
/**
* Used to keep track of in-scope namespace bindings.
*
* For JAXB unmarshaller to correctly unmarshal documents, it needs
* to know all the effective namespace declarations.
*/
private NamespaceSupport namespaces = new NamespaceSupport();
public void startPrefixMapping(String prefix, String uri) throws SAXException {
namespaces.pushContext();
namespaces.declarePrefix(prefix,uri);
super.startPrefixMapping(prefix, uri);
}
public void endPrefixMapping(String prefix) throws SAXException {
namespaces.popContext();
super.endPrefixMapping(prefix);
}
Primer.xsd
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<xsd:annotation>
<xsd:documentation xml:lang="en">
Purchase order schema for Example.com.
Copyright 2000 Example.com. All rights reserved.
</xsd:documentation>
</xsd:annotation>
<xsd:element name="purchaseOrders">
<xsd:complexType>
<xsd:sequence>
<xsd:element ref="purchaseOrder" minOccurs="0" maxOccurs="unbounded"/>
</xsd:sequence>
</xsd:complexType>
</xsd:element>
<xsd:element name="purchaseOrder" type="PurchaseOrderType"/>
<xsd:element name="comment" type="xsd:string"/>
<xsd:complexType name="PurchaseOrderType">
<xsd:sequence>
<xsd:element name="shipTo" type="USAddress"/>
<xsd:element name="billTo" type="USAddress"/>
<xsd:element ref="comment" minOccurs="0"/>
<xsd:element name="items" type="Items"/>
</xsd:sequence>
<xsd:attribute name="orderDate" type="xsd:date"/>
</xsd:complexType>
<xsd:complexType name="USAddress">
<xsd:sequence>
<xsd:element name="name" type="xsd:string"/>
<xsd:element name="street" type="xsd:string"/>
<xsd:element name="city" type="xsd:string"/>
<xsd:element name="state" type="xsd:string"/>
<xsd:element name="zip" type="xsd:decimal"/>
</xsd:sequence>
<xsd:attribute name="country" type="xsd:NMTOKEN"
fixed="US"/>
</xsd:complexType>
<xsd:complexType name="Items">
<xsd:sequence>
<xsd:element name="item" minOccurs="0" maxOccurs="unbounded">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="productName" type="xsd:string"/>
<xsd:element name="quantity">
<xsd:simpleType>
<xsd:restriction base="xsd:positiveInteger">
<xsd:maxExclusive value="100"/>
</xsd:restriction>
</xsd:simpleType>
</xsd:element>
<xsd:element name="USPrice" type="xsd:decimal"/>
<xsd:element ref="comment" minOccurs="0"/>
<xsd:element name="shipDate" type="xsd:date" minOccurs="0"/>
</xsd:sequence>
<xsd:attribute name="partNum" type="SKU" use="required"/>
</xsd:complexType>
</xsd:element>
</xsd:sequence>
</xsd:complexType>
<!-- Stock Keeping Unit, a code for identifying products -->
<xsd:simpleType name="SKU">
<xsd:restriction base="xsd:string">
<xsd:pattern value="\d{3}-[A-Z]{2}"/>
</xsd:restriction>
</xsd:simpleType>
</xsd:schema>
test.xml
<purchaseOrders>
<!-- 1st -->
<purchaseOrder orderDate="1999-10-20">
<shipTo country="US">
<name>Alice Smith</name>
<street>123 Maple Street</street>
<city>Cambridge</city>
<state>MA</state>
<zip>12345</zip>
</shipTo>
<billTo country="US">
<name>Robert Smith</name>
<street>8 Oak Avenue</street>
<city>Cambridge</city>
<state>MA</state>
<zip>12345</zip>
</billTo>
<items/>
</purchaseOrder>
</purchaseOrders>
{}purchaseorders>{}purchaseorder>{}comment>最佳答案
我遇到了这个问题;尝试使用来自 jaxb 引用实现的部分解码示例。
我确定的解决方案是将自定义 com.sun.xml.bind.api.ClassResolver 添加到在上面的 startElement 方法中创建的解码器中。见:
try {
unmarshaller = context.createUnmarshaller();
unmarshaller.setProperty(ClassResolver.class.getName(), myClassResolver);
} catch( JAXBException e ) {
...
这是一个模型解析器...
new ClassResolver()
{
@Override
public Class<?> resolveElementName(String nsUri, String localName) throws Exception
{
if(MY_NAMESPACE.equals(nsUri) && MY_BAR.equals(localName))
return BarType.class;
else
return null;
}
}
关于java - 不使用 @XMLRootElement 的 JAXB 部分解码元素,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6484681/
我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div
我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看rubyzip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t
我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h
我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po