我有一个带有模型DTO的罐子。
我希望这些类(标有@XmlRootElement
)可用于Jersey 1 HTTP客户端。我的初始化代码是:
ClientConfig cc = new DefaultClientConfig(MyContextResolver.class);
Client client = Client.create(cc);
在MyContextResolver
(实现ContextResolver<JAXBContext>
)中,我尝试过:
jaxbContext = JAXBContext.newInstance(Class1.class, Class2.class);
它有效,但它不是动态的(我必须逐个手动添加类)。
I also used Spring and it worked,但我想要一个免费的Spring解决方案。
有更好的解决方案吗?一个自动扫描jar中所有类并将它们添加到上下文中的类?感谢。
答案 0 :(得分:1)
您是否有理由想要一个同时处理所有类的上下文?如果你有名字冲突,你会冒一个例外。
为什么不简单地使用一个类似于:
的ContextResolverMap<Class,JAXBContect> cache = new ConcurrentHashMap<>();
getContext(Class<?> type) {
JAXBContect context = cache.get(type);
if (context == null) }
context = JAXBContext.newInstance(type);
cache.put(type,context);
}
return context;
}
JAXB将解决必要的类依赖项(唯一的问题是子类,但那些应该在parrent类中标记为@XMLSeeAlso)。
唯一可能的问题是,您将在运行时期间找不到任何JAXB问题,而不是在启动期间。但与此同时,任何错误都只会影响依赖于错误注释类的客户端,其余的将会起作用。
答案 1 :(得分:0)
我最后手动扫描了罐子......
public static List<Class> listClassesInPackage(String jarName, String packageName) throws IOException {
packageName = packageName.replaceAll("\\.", "/");
JarInputStream jarFile = new JarInputStream(new FileInputStream(jarName));
JarEntry jarEntry = null;
List<Class> classes = new ArrayList();
do {
try {
jarEntry = jarFile.getNextJarEntry();
if (jarEntry != null && jarEntry.getName().startsWith(packageName) && jarEntry.getName().endsWith(".class")) {
Class<?> forName = Class.forName(jarEntry.getName().replaceAll("/", "\\.").replaceAll(".class", ""));
XmlRootElement xmlAnnotation = forName.getAnnotation(XmlRootElement.class);
if (xmlAnnotation != null) {
classes.add(forName);
}
}
} catch (ClassNotFoundException | IOException ex) {
// ignore this class
}
} while (jarEntry != null);
return classes;
}
调用:
List<Class> packageClasses = listClassesInPackage(
Meta.class.getProtectionDomain().getCodeSource().getLocation().getPath(), "pt.iol.model2");
jaxbContext = JAXBContext.newInstance(packageClasses.toArray(new Class[0]));