0

我正在尝试在我的石英调度程序之上创建一个休息 api 服务器。我希望能够将 org.quartz.Trigger 和 org.quartz.JobDetail 对象作为 JSON 返回。问题是我无法将@XmlRootElement 添加到这些类而无需重新编译jar,这会导致未来升级等问题。我已经测试并且能够在添加@XmlRootElement 时序列化类列表但是当我尝试返回一个列表我收到错误消息“找不到 Java 类 java.util.ArrayList 和 Java 类型 java.util.List 和 MIME 媒体类型 application/json 的消息正文编写器”。我尝试添加自定义 MessageBodyWriter ,但这似乎也不能解决问题。有没有办法将石英类编组为 JSON,而无需修改石英代码以添加 @XmlRootElement。

@Path("/jobs")
public class JobsResource {

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public List<Trigger> listScheduledJobs() throws SchedulerException {
        return TaskEngine.getInstance().listScheduledJobs();
    }

}

网络服务器配置

public class TaskEngineWebServer {

    private static final Logger logger = Logger.getLogger(TaskEngineWebServer.class.getName());

    private Server server;

    public TaskEngineWebServer() {
        this(8585);
    }

    public TaskEngineWebServer(Integer port) {
        server = new Server(port);

        logger.info("Configuring rest service to start at url /r");
        ServletContextHandler handler = new ServletContextHandler(ServletContextHandler.NO_SECURITY);
        //handler.getInitParams().put("com.sun.jersey.api.json.POJOMappingFeature", "true");
        PackagesResourceConfig packagesResourceConfig = new PackagesResourceConfig("com.hp.vf.scheduler.server.rest");

        ServletContainer servletContainer = new ServletContainer(packagesResourceConfig);
        handler.addServlet(new ServletHolder(servletContainer), "/r/*");

        server.setHandler(handler);
        logger.info("Done configuring rest service");
    }

    public void start() throws Exception {
        server.start();
    }

    public void stop() throws Exception {
        server.stop();
    }

    public boolean isStarted() {
        return server.isStarted();
    }

    public boolean isStopped() {
        return server.isStopped();
    }
}
4

2 回答 2

1

我认为您不能直接将 List 作为 JSON 返回。您需要有一个包含此列表的包装类。例如尝试这样的事情

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class TriggerWrapper{

  private List<Triggers> triggers;

  public List<Triggers> getTriggers(){
   if(triggers==null){
     triggers = new ArrayList<Triggers>();
   }
   return triggers;
  }
}

然后在你的休息服务课上:

@Path("/jobs")
public class JobsResource {

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public TriggerWrapper listScheduledJobs() throws SchedulerException {
        TriggerWrapper response = new TriggerWrapper();
        List<Triggers> triggers =  TaskEngine.getInstance().listScheduledJobs();
        response.getTriggers.addAll(triggers);
        return response;
    }

}

你的 json 会是这样的:

{
    "triggerwrapper": {
        "triggers": [
            {
                "triggerid": 1
            },
            {
                "triggerid": 2
            }
        ]
    }
}

当然,如果你愿意,你可以从你的 json 中删除根元素标签,它可以在 jersey 中配置。

于 2012-12-05T06:09:09.033 回答
1

我终于找到了一个干净的解决方案,它涉及创建我自己的 MediaBodyWriter 类并将其添加为提供程序。您必须确保您没有使用 jersey-bundle jar,因为默认的 jaxb 到 json 提供程序将覆盖您的。

需要罐子

jersey-core jersey-servlet jersey-server

jackson-annotations jackson-databind jackson-core

我在网上某处找到了这个 MediaWriter 示例。抱歉没有网址,但感谢写它的人。

@Provider
@Produces({ MediaType.APPLICATION_JSON })
public class JacksonWriter implements MessageBodyWriter<Object> {
    private static final ObjectMapper MAPPER = new ObjectMapper();

    @Override
    public boolean isWriteable(Class<?> aClass, Type type, Annotation[] annotations, MediaType mediaType) {
        return true;
    }

    @Override
    public long getSize(Object value, Class<?> type, Type genericType, Annotation[] annotations, MediaType mediaType) {
        return -1;
    }

    @Override
    public void writeTo(Object value, Class<?> type, Type genericType, Annotation[] annotations, MediaType mediaType, MultivaluedMap<String, Object> httpHeaders,
            OutputStream entityStream) {
        try {
            MAPPER.writeValue(entityStream, value);
        } catch (IOException ex) {
            throw new RuntimeException(ex);
        }
    }
}

当它加载时,您将看到一条日志消息,表明您的提供程序已加载。

这给了我期望的 json 输出,因为它不依赖于 JAXB 注释,而只是使用对象映射器/反射。可能效率较低,但对我来说没关系。

于 2012-12-05T16:23:17.287 回答