1

我使用 jersey-client 来使用 REST 服务。我需要请求的实体和 Last-Modified 标头。

所以我做了以下事情:

ClientResponse response = webResource.get(ClientResponse.class);
Person person = response.getEntity(Person.class);

这样可行。我得到响应,我可以将实体(即 XML)编组到我的 POJO 中。当我调试并查看响应的标头时,我看到有一个 Last-Modified 标头集。

但是当我尝试通过检索日期时

response.getLastModified();

我在 URLConnectionClientHandler 的某个地方得到了一个 NPE。

有谁知道我做错了什么?

编辑:根据要求跟踪

java.lang.NullPointerException: null
at com.sun.jersey.api.client.ClientResponse.getLastModified(ClientResponse.java:647) ~[jersey-client-1.12.jar:1.12]
at a.o.u.user.dao.impl.uds.PersonenUdsClient.getPerson(PersonenUdsClient.java:103) ~[um-user-2.5.0-Beta1-SNAPSHOT.jar:na]
at a.o.u.user.dao.impl.UserDaoUdsImpl.mergeWithUdsUser(UserDaoUdsImpl.java:282) ~[um-user-2.5.0-Beta1-SNAPSHOT.jar:na]
at a.o.u.user.dao.impl.UserDaoUdsImpl.getUserWithEmail(UserDaoUdsImpl.java:124) ~[um-user-2.5.0-Beta1-SNAPSHOT.jar:na]
at ...

编辑:正如 npe 建议的那样,我深入研究了代码。我想我找到了问题所在。除了 jersey-client 之外,我在类路径中也有 cxf。jersey 和 cxf 都提供了一个名为RuntimeDelegateImpl的类。但是 CXFs 版本没有 DateHeaderDelegate。我认为 RuntimeDelegateImpl 的错误版本(CXF)采用。

到目前为止,我还没有找到如何明确设置要使用的RuntimeDelegateImpl

4

2 回答 2

2

使用来源,卢克

1.12 版的实现ClientResponse#getLastModified()如下所示:

/*639*/  /**
/*640*/   * Get the last modified date.
/*641*/   *
/*642*/   * @return the last modified date, otherwise <code>null</code> if not present.
/*643*/   */
/*644*/  public Date getLastModified() {
/*645*/      String d = getHeaders().getFirst("Last-Modified");
/*646*/  
/*647*/      return (d != null) ? dateDelegate.fromString(d) : null;
/*648*/  }

NullPointerException在第 647 行得到一个,所以看起来,dateDelegate就是null. 现在,dateDelegate对象在第 321 行初始化,如下所示:

/*321*/  protected static final HeaderDelegate<Date> dateDelegate =
/*322*/          RuntimeDelegate.getInstance().createHeaderDelegate(Date.class);

现在,该字段是final,因此在此初始化之后无法更改它 - 这意味着从一开始就 - 这意味着dateDelegate有某种配置问题并且未创建委托。null

此外,委托是在AbstractRuntimeDelegate类中创建的(此处为 1.12 的源代码),如下所示:

/* 88*/  map.put(Date.class, _createHeaderDelegate(Date.class));

这个兔子洞越来越深,所以我要在这里停下来,但你知道路。

最后但并非最不重要的一点 - 调试器是你的朋友,我的朋友 ;-)

于 2012-06-21T09:26:16.867 回答
1

To make it a wrap.

The problem is that I have both jersey-client and cxf in the classpath. Both RuntimeDelegateImpl. But CXFs version does not feature a DateHeaderDelegate. The wrong version (CXFs) of RuntimeDelegateImpl is taken.

I solved the issue by retrieving the Last-Modified Header 'manually':

private Date getLastModified(ClientResponse response){
    String lastModifiedString = response.getHeaders().getFirst(
            "Last-Modified");
    if (StringUtils.isEmpty(lastModifiedString)) {
        LOG.warn("Expect to get Last-Modified header when retrieving a Person by pnr "
                + "but there is none.");
        return null;
    } else {
        try {
            // format is Thu, 21 Jun 2012 08:00:42 GMT
            return new SimpleDateFormat(
                    "EEE, dd MMM yyyy HH:mm:ss zzz", Locale.US)
                    .parse(lastModifiedString);
        } catch (ParseException e) {
            LOG.error("Could not parse Last-Modified date "
                    + e.getMessage());
            return null;
        }
    }
}

Thanx to npe for the hints.

于 2012-06-22T12:26:48.967 回答