gpt4 book ai didi

java - XMLGregorianCalendar 日期序列化为空字符串

转载 作者:塔克拉玛干 更新时间:2023-11-02 08:24:03 27 4
gpt4 key购买 nike

使用 Java 1.6 wsimport 我从 WSDL 为 Web 服务生成了源代码。请求结构中的一个字段在 WSDL 包含的 XML 架构中具有类型 xs:dateTime,在生成的代码中具有类型 javax.xml.datatype.XMLGregorianCalendar

通过使用 soapUI 进行手动测试,我确定 Web 服务接受以下序列化值:2011-12-082011-12-08Z。以下内容不被接受,在这种情况下的响应是空回复(不是显式错误):2011-12-08T20:00:002011-12-08T20:00: 00-05:00。如果重要的话,服务本身是 .NET 驱动的。

我的想法是服务器应该接受完整的日期/时间并只拒绝日期,但相反的情况是正在发生的事情。但我不认为服务器的维护者会愿意改变。所以我试图说服客户只发送一个日期。

我无法说服我的客户端代码将 XMLGregorianCalendar 对象序列化为仅日期。好吧,实际上我可以,除了生成的代码执行此操作时。当生成的客户端代码(由 wsimport 生成)执行时,序列化值为空字符串,服务器正确返回错误。我使用数据包嗅探器验证了这一点。

以下是我在请求中创建和填充日期字段的方式:

import java.util.Calendar;
import java.util.GregorianCalendar;
import javax.xml.datatype.DatatypeConfigurationException;
import javax.xml.datatype.DatatypeConstants;
import javax.xml.datatype.DatatypeFactory;
import javax.xml.datatype.XMLGregorianCalendar;
import java.util.TimeZone;
// also import GeneratedRequest from generated packages

private makeRequest() {
GeneratedRequest request;
// ...
request.setDateField(xmlDayNow(TimeZone.getTimeZone("America/New_York"),
6)); // broadcast day starts at 6 am EST
// ...
}

@XmlSchemaType(name="date")
private static XMLGregorianCalendar xmlDayNow(TimeZone tz, int localHourStart)
throws MyException {
GregorianCalendar cal = gregorianBroadcastDayNow(tz, localHourStart);
XMLGregorianCalendar result;
try {
result = DatatypeFactory.newInstance().newXMLGregorianCalendarDate(
cal.get(Calendar.YEAR), cal.get(Calendar.MONTH) + 1,
cal.get(Calendar.DAY_OF_MONTH), DatatypeConstants.FIELD_UNDEFINED)
.normalize();
} catch (DatatypeConfigurationException e) {
throw new MyException("XMLGregorianCalendar issue", e);
}
return result;
}

protected static GregorianCalendar gregorianBroadcastDayNow(TimeZone tz,
int localHourStart) {
GregorianCalendar now = new GregorianCalendar(tz);
if (now.get(GregorianCalendar.HOUR_OF_DAY) < localHourStart) {
now.add(GregorianCalendar.DAY_OF_MONTH, -1);
}
return now;
}

在我的例子中,XMLGregorianCalendar 的实现类是 com.sun.org.apache.xerces.internal.jaxp.datatype.XMLGregorianCalendarImpl。在调试器中,或者如果我添加日志输出,调用日期对象的 toXMLFormat() 方法仅返回一个日期,例如 2011-12-09。使用调试器检查日期对象本身,我看到它的 yeardaymonth 字段已填充,所有其他字段都已填充null-2147483648DatatypeConstants.FIELD_UNDEFINED 的值。根据我找到的所有文档和互联网搜索结果,我的日期对象的格式是正确的。

我疯了吗?服务器真的出错了吗?生成的客户端代码拒绝发送日期只对吗?这是一个合理的“未定义行为”案例吗?是否使用了错误的实现类(无论如何这可能很重要)? wsimport 是否存在影响我的已知问题?

最佳答案

我发现 JAXB 的日期转换不应该留给他们自己的设备。不太熟悉 wsimport,但它是否允许您指定绑定(bind)文件?我使用 Joda Date/Time,但我敢肯定这个想法是一样的。将此添加到我的 binding.xjb:

<globalBindings>
<serializable />
<javaType name="org.joda.time.DateTime" xmlType="xsd:dateTime"
printMethod="someStaticDateConverterClass.printDateIso"
parseMethod="someStaticDateConverterClass.parseDateIso" />
<javaType name="org.joda.time.DateTime" xmlType="xs:date"
printMethod="someStaticDateConverterClass.printDateYYYYDashMMDashDD"
parseMethod="someStaticDateConverterClass.parseDateYYYYDashMMDashDD" />
</globalBindings>

这样的类:

import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;
import org.joda.time.format.ISODateTimeFormat;

public class someStaticDateConverterClass{

private static DateTimeFormatter isoFormat = ISODateTimeFormat.dateTime();
private static DateTimeFormatter YYYYDashMMDashDDFormat = DateTimeFormat
.forPattern("yyyy-MM-dd");
private static DateTimeFormatter YYYYMMDDFormat = DateTimeFormat
.forPattern("yyyyMMdd");

public static String printDateIso(DateTime value) {
return printDate(value, isoFormat);
}

public static DateTime parseDateIso(String value) {
return parseDate(value, isoFormat);
}

public static String printDateYYYYDashMMDashDD(DateTime value) {
return printDate(value, YYYYDashMMDashDDFormat);
}

public static DateTime parseDateYYYYDashMMDashDD(String value) {
return parseDate(value, YYYYDashMMDashDDFormat);
}

public static String printDateYYYYMMDD(DateTime value) {
return printDate(value, YYYYMMDDFormat);
}

public static DateTime parseDateYYYYMMDD(String value) {
return parseDate(value, YYYYMMDDFormat);
}

private static String printDate(DateTime value, DateTimeFormatter format) {

String dateAsStr;

if (value != null) {
dateAsStr = value.toString(format);
} else {
dateAsStr = null;
}

return dateAsStr;
}

private static DateTime parseDate(String value, DateTimeFormatter format) {

DateTime strAsDate;

if (value != null) {
strAsDate = format.parseDateTime(value);
} else {
strAsDate = null;
}
return strAsDate;
}
}

关于java - XMLGregorianCalendar 日期序列化为空字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8448266/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com