4

我想从 jar 中的类文件生成 jaxb xsd 模式。目前,我正在使用 jaxb2-maven-plugin 使用 java 文件生成模式。

<plugin>
        <groupId>org.codehaus.mojo</groupId>
        <artifactId>jaxb2-maven-plugin</artifactId>
        <version>${maven.plugin.jaxb2}</version>
        <executions>
           <execution>
              <id>schemagen</id>
              <goals>
                 <goal>schemagen</goal>
              </goals>
              <phase>process-classes</phase>
              <configuration>
                 <quiet>true</quiet>
                 <includes>
                    <include>com/someProject/domain/*.java</include>
                 </includes>
                 <outputDirectory>${project.build.directory}/schemas</outputDirectory>
                 <clearOutputDir>true</clearOutputDir>
              </configuration>
           </execution>
        </executions>
     </plugin>

但是,我有一个用例,我正在获取一个依赖项 jar 文件,并希望从该 jar 文件中生成一些类。任何人都可以建议如何做到这一点。

4

2 回答 2

2

为简单起见,我重用本文中提供的代码来演示从存储在 Jar 存档中的 Java 类生成 Jaxb Schema 的必要步骤

该代码包含两个类 - Employee 和 Address:

package base.package;

import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlRootElement;

@XmlAccessorType(XmlAccessType.FIELD)
@XmlRootElement(name = "employee")
public class Employee 
{
    @XmlAttribute
    private int id;
    private String name;
    private double salary;
    private String designation;
    private Address address;

    public int getId() { return id; }
    public void setId(int id) { this.id = id; }
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
    public double getSalary() { return salary; }
    public void setSalary(double salary) { this.salary = salary; }
    public String getDesignation() { return designation; }
    public void setDesignation(String designation) { this.designation = designation; }
    public Address getAddress() { return address; }
    public void setAddress(Address address) { this.address = address; }
}

和一个引用的类:

package base.package;

public class Address 
{
    private String line1;
    private String line2;
    private String city;
    private String state;
    private long zipcode;

    public String getLine1() { return line1; }
    public void setLine1(String line1) { this.line1 = line1; }
    public String getLine2() { return line2; }
    public void setLine2(String line2) { this.line2 = line2; }
    public String getCity() { return city; }
    public void setCity(String city) { this.city = city; }
    public String getState() { return state; }
    public void setState(String state) { this.state = state; }
    public long getZipcode() { return zipcode; }
    public void setZipcode(long zipcode) { this.zipcode = zipcode; }
}

现在使用以下命令编译此代码:javac -d bin src/base/package/*.java. 这会将位于 src 中的所有源文件编译到 bin 目录:

base-dir
|- src
|  \- base
|     \- package
|        |- Employee.java
|        \- Address.java
\- bin
   \- base
      \- package
         |- Employee.class
         \- Address.class

要为已编译的类获取正确的 Jar-archive,请使用:jar -cf test.jar -C bin/ .这将生成test.jar包含以下内容的存档:

test.jar
|- base
|  \- package
|     |- Employee.class
|     \- Address.class
\- META-INF
   \- MANIFEST.MF

您现在可以删除 bin 目录及其所有内容,因为我们需要的所有文件都存储在存档中,并证明架构实际上是从 Jar 存档中的文件生成的。

随着所有准备工作最终完成,可以解决实际问题 - 如何从位于该 test.jar 存档中的 .class-files 生成模式:

只需运行此命令: schemagen -cp test.jar base.package.Employee它应该会生成类似于以下代码段的模式定义:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<xs:schema version="1.0" xmlns:xs="http://www.w3.org/2001/XMLSchema">

  <xs:element name="employee" type="employee"/>

  <xs:complexType name="employee">
    <xs:sequence>
      <xs:element name="address" type="address" minOccurs="0"/>
      <xs:element name="designation" type="xs:string" minOccurs="0"/>
      <xs:element name="salary" type="xs:double"/>
      <xs:element name="name" type="xs:string" minOccurs="0"/>
    </xs:sequence>
    <xs:attribute name="id" type="xs:int" use="required"/>
  </xs:complexType>

  <xs:complexType name="address">
    <xs:sequence>
      <xs:element name="city" type="xs:string" minOccurs="0"/>
      <xs:element name="line1" type="xs:string" minOccurs="0"/>
      <xs:element name="line2" type="xs:string" minOccurs="0"/>
      <xs:element name="state" type="xs:string" minOccurs="0"/>
      <xs:element name="zipcode" type="xs:long"/>
    </xs:sequence>
  </xs:complexType>
</xs:schema>

希望这很简单


编辑:似乎 jaxb2-maven-plugin 和 ant-task 根本无法使用类文件,因此最简单和最简单的解决方案可能是提供一个脚本文件(Windows 上的 .bat; . sh on *nix/Mac),您只需手动调用命令:

由于我目前在 Windows 7 上运行,因此自动将架构生成到项目的架构子目录中的脚本如下所示:

schemagen -cp path/to/jar/*.jar -d ./schemas/ package.ClassName

scripts然后,您可以使用绑定到 generate-sources 阶段的 maven exec-plugin简单地调用该脚本(我已将其放置在项目的子目录中):

<plugin>
    <artifactId>exec-maven-plugin</artifactId>
    <groupId>org.codehaus.mojo</groupId>
    <executions>
        <execution>
            <id>Generate schemas from class files contained in a jar</id>
            <phase>generate-sources</phase>
            <goals>
                <goal>exec</goal>
            </goals>
            <configuration>
                <executable>${basedir}/scripts/generate-sources.bat</executable>
            </configuration>
        </execution>
    </executions>
</plugin>

然后应该在执行mvn generate-sources或任何后期阶段 maven 提供时自动生成模式。


编辑:我已经稍微修改了脚本,因为它能够处理通配符,尽管您必须指定*.jar而不是仅指定*- 但我想这应该足够好,至少这可以避免您手动输入 Jar- 的名称包含 JAXB 类的文件

于 2014-01-19T19:35:25.310 回答
1

您可以编写程序JAXBContext在您的域模型上创建一个,然后调用该generateSchema方法JAXBContext来生成 XML Schema。

例子

import java.io.IOException;
import javax.xml.bind.*;
import javax.xml.transform.Result;
import javax.xml.transform.stream.StreamResult;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(PageableResponse.class);   
        jc.generateSchema(new SchemaOutputResolver() {

            @Override
            public Result createOutput(String namespaceURI, String suggestedFileName)
                throws IOException {
                return new StreamResult(suggestedFileName);
            }

        });

    }

}
于 2014-01-19T21:54:57.317 回答