- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在我的项目中,我们正在实现一项新功能,其中涉及与 Web 服务对话。我想使用 JAX-WS 来做到这一点,因为它似乎是最简单的,也是所有教程的做法。
现在我们的应用程序中已经有一些 Web 服务调用,但这些调用使用 Axis 1.4,并且因为 Axis 提供了一些类(Service、QName)自己的实现,所以当我尝试与新 Web 通信时,我遇到了一些类加载器问题使用 JAX-WS 提供服务。
我考虑过用 jre 内置的 JAX-WS RI 替换 Axis,但我们使用 Axis 将请求序列化为字节数组,然后以某种方式将其发送到其他系统。
ByteArrayOutputStream outStream = new ByteArrayOutputStream();
// provide an AXIS configuration which overwrites the default HTTP
// handler with our handler
// implementation which writes the SOAP message into the given output
// stream.
SimpleProvider config = new SimpleProvider();
config.deployTransport("http", new SimpleTargetedChain(new ByteArraySender(outStream)));
ArchiveService service = new ArchiveServiceLocator(config);
Archive archive = service.getArchive()
archive.archiveReportWithDefTags(metaData, dataHandler);
return outStream.toString(HTTPConstants.HEADER_DEFAULT_CHAR_ENCODING);
是否可以使用 JAX-WS RI 或 CXF 执行类似的操作(将请求序列化到本地变量/流)?我更喜欢使用这两个而不是 Axis2,因为我在 SO 上读过几次,认为它们是首选。
最佳答案
对于遇到同样问题的人,我找到了 CXF 和 Metro (JAX-WS RI) 的解决方案,但它们最终都没有出现在我的应用程序中,因为它们将我带入类路径 hell 。相反,我选择了旧的 Axis (1.x),因为它可以正常工作。
对于 CXF,我需要在总线中注册一个自定义 ConduitInitiator。 ConduitInitiator 会将请求保存到自定义 Conduit 中的 OutputStream 中。
配置/设置类:
import javax.xml.ws.soap.MTOMFeature;
import org.apache.cxf.Bus;
import org.apache.cxf.BusFactory;
import org.apache.cxf.transport.Conduit;
import org.apache.cxf.transport.ConduitInitiatorManager;
public class ArchiveMessageSerializer {
public String serializeArchiveReportWithDefTags(ArchiveMetadataType metaData) throws Exception {
final SavingConduit savingConduit = new SavingConduit();
registerCustomTransport(savingConduit);
try {
final Service service = new Service();
final Port port = service.getArchive(new MTOMFeature(true));
port.someMethod(metaData);
}
finally {
tearDownBusWithCustomTransport();
}
return savingConduit.getResult().toString("iso-8859-1");
}
private void registerCustomTransport(Conduit conduit) {
Bus bus = BusFactory.getThreadDefaultBus();
CustomConduitInitiator customTransport = new CustomConduitInitiator(conduit);
ConduitInitiatorManager cim = bus.getExtension(ConduitInitiatorManager.class);
cim.registerConduitInitiator("http://schemas.xmlsoap.org/soap/http", customTransport);
cim.registerConduitInitiator("http://schemas.xmlsoap.org/wsdl/soap/http", customTransport);
}
private void tearDownBusWithCustomTransport() {
BusFactory.setThreadDefaultBus(null);
}
}
管道启动器:
import org.apache.cxf.Bus;
import org.apache.cxf.service.model.EndpointInfo;
import org.apache.cxf.transport.Conduit;
import org.apache.cxf.transport.ConduitInitiator;
import org.apache.cxf.ws.addressing.EndpointReferenceType;
class CustomConduitInitiator implements ConduitInitiator {
private final Conduit fMyConduit;
public CustomConduitInitiator(Conduit conduit) {
fMyConduit = conduit;
}
@Override
public Conduit getConduit(EndpointInfo targetInfo, Bus bus) throws IOException {
return fMyConduit;
}
@Override
public Conduit getConduit(EndpointInfo localInfo, EndpointReferenceType target, Bus bus) throws IOException {
return fMyConduit;
}
@Override
public Set<String> getUriPrefixes() {
return null;
}
@Override
public List<String> getTransportIds() {
return null;
}
}
导管:
import org.apache.cxf.message.Exchange;
import org.apache.cxf.message.Message;
import org.apache.cxf.message.MessageImpl;
import org.apache.cxf.transport.Conduit;
import org.apache.cxf.transport.MessageObserver;
import org.apache.cxf.ws.addressing.EndpointReferenceType;
class SavingConduit implements Conduit {
private final ByteArrayOutputStream fOutputStream;
private Exchange fExchange;
private MessageObserver fObserver;
SavingConduit() {
fOutputStream = new ByteArrayOutputStream();
}
ByteArrayOutputStream getResult() {
return fOutputStream;
}
@Override
public void prepare(Message message) throws IOException {
fExchange = message.getExchange();
message.setContent(OutputStream.class, fOutputStream);
}
@Override
public void close(Message message) throws IOException {
final OutputStream outputStream = message.getContent(OutputStream.class);
if (outputStream != null) {
outputStream.close();
message.put(Message.RESPONSE_CODE, HttpURLConnection.HTTP_OK);
final Message inMessage = new MessageImpl();
inMessage.put(Message.HTTP_REQUEST_METHOD, "GET");
inMessage.setContent(InputStream.class, new ByteArrayInputStream(new byte[0]));
inMessage.setExchange(fExchange);
fExchange.setInMessage(inMessage);
fObserver.onMessage(inMessage);
}
final InputStream inputStream = message.getContent(InputStream.class);
if (inputStream != null) {
inputStream.close();
}
}
@Override
public EndpointReferenceType getTarget() {
return null;
}
@Override
public void close() {
}
@Override
public void setMessageObserver(MessageObserver observer) {
fObserver = observer;
}
@Override
public MessageObserver getMessageObserver() {
return fObserver;
}
}
对于 Metro,我必须将 TransportTubeFactory 注册为服务。 Metro 会选择我的类并将其用作创建我的 Tube 的候选工厂,这是 Metro 的客户端传输机制的概念。
文件 META-INF/services/com.sun.xml.ws.api.pipe.TransportTubeFactory:
com.mypackage.RedirectionTransportFactory
设置代码:
import javax.xml.ws.soap.MTOMFeature;
public class ArchiveMessageSerializer {
public String serializeArchiveReportWithDefTags(ArchiveMetadataType metaData) throws UnsupportedEncodingException {
final ByteArrayOutputStream buffer = new ByteArrayOutputStream();
RedirectionTransportFactory.getRegistry().registerRedirectionInThread(buffer);
final Service service = new Service();
final Port port = service.getPort(new MTOMFeature());
port.someMethod(metaData);
return buffer.toString("iso-8859-1");
}
}
运输工厂:
import com.sun.xml.ws.api.pipe.ClientTubeAssemblerContext;
import com.sun.xml.ws.api.pipe.TransportTubeFactory;
import com.sun.xml.ws.api.pipe.Tube;
public class RedirectionTransportFactory extends TransportTubeFactory {
private static final RedirectionTransportRegistry REGISTRY = new RedirectionTransportRegistry();
public static RedirectionTransportRegistry getRegistry() {
return REGISTRY;
}
@Override
public Tube doCreate(ClientTubeAssemblerContext context) {
return getRegistry().createTubeFor(context);
}
}
我的输出流注册表:
import com.sun.xml.ws.api.pipe.ClientTubeAssemblerContext;
import com.sun.xml.ws.api.pipe.Tube;
public class RedirectionTransportRegistry {
private final ThreadLocal<RegisteredTube> fCurrentThreadTube = new ThreadLocal<RegisteredTube>();
Tube createTubeFor(ClientTubeAssemblerContext context) {
final RegisteredTube registeredTube = fCurrentThreadTube.get();
if (registeredTube == null) {
return null;
}
return registeredTube.createTube(context);
}
public void registerRedirectionInThread(OutputStream outputStream) {
final RegisteredTube registeredTube = new RegisteredTube(outputStream);
fCurrentThreadTube.set(registeredTube);
}
private class RegisteredTube {
private final OutputStream fOutputStream;
public RegisteredTube(OutputStream outputStream) {
fOutputStream = outputStream;
}
Tube createTube(ClientTubeAssemblerContext context) {
return new RedirectionTube(context.getCodec(), fOutputStream);
}
}
}
将消息写入给定输出流的管:
import javax.xml.ws.WebServiceException;
import com.sun.xml.ws.api.message.Packet;
import com.sun.xml.ws.api.pipe.Codec;
import com.sun.xml.ws.api.pipe.NextAction;
import com.sun.xml.ws.api.pipe.TubeCloner;
import com.sun.xml.ws.api.pipe.helper.AbstractTubeImpl;
public class RedirectionTube extends AbstractTubeImpl {
private final Codec fCodec;
private final OutputStream fOutputStream;
public RedirectionTube(Codec codec, OutputStream outputStream) {
fCodec = codec;
fOutputStream = outputStream;
}
@Override
public AbstractTubeImpl copy(TubeCloner cloner) {
return this;
}
@Override
public NextAction processRequest(Packet request) {
return doReturnWith(process(request));
}
@Override
public NextAction processResponse(Packet response) {
throw new IllegalStateException("MyAbstractTubeImpl's processException shouldn't be called.");
}
@Override
public NextAction processException(Throwable t) {
throw new IllegalStateException("MyAbstractTubeImpl's processException shouldn't be called.");
}
@Override
public void preDestroy() {
}
@Override
public Packet process(Packet p) {
try {
fCodec.encode(p, fOutputStream);
return p.createClientResponse(null);
}
catch (IOException e) {
throw new WebServiceException(e);
}
}
}
尽管 CXF 有更好的文档,但一旦我了解了 META-INF 文件以及要实现的类(TransportTubeFactory),实际实现 Metro 版本就更容易了。 CXF 有一个更复杂的管道,其中的拦截器都处理相同的消息,而 Metro 使用更干净的网络堆栈式架构及其管道。
我认为尽管他们能够做这种事情,但他们中没有一个人真正适合真正做这件事。由于它们隐藏在 JAX-WS API 后面,因此很难配置。根据您的 JRE 附带的实现,您将陷入类路径 hell 。
关于java - JAX-WS 序列化为流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27171620/
我有一个使用 spring-ws 的 web 服务(服务器端)的实现。我想更改我的代码以使用 spring-integration-ws(ws:入站网关和 channel )。我已经尝试过示例,但仍然
我正在尝试使用 JAX-WS (Metro) 开发一个独立的 Java Web 服务客户端,它使用 WS-Security 和用户名 token 身份验证(密码摘要、随机数和时间戳)和时间戳验证以及
我开始使用 JAX WS 研究 Java Web 服务。我正在阅读的书的第一章展示了如何仅使用 java SE 构建和部署简单的 jax ws web 服务。特别是,Web 服务是通过 Endpoin
我尝试在 Netbean 6.8 中使用 ws-import 生成 Java 类。我想重新生成 jax-ws,因为在 ebay.api.paypalapi 包中发现了一个错误(我认为该错误是由于 Pa
我正在为我的项目编写服务器端,它需要 websockets 提供的功能。我一直在寻找并在 npm 中找到了 3 个库。 ws、websocket 和express-ws。您能否解释一下它们之间的区别并
我有一个用于测试 Web 服务的项目,它使用 spring-ws 库。spring-ws 在不久的将来会支持 WS-Reliable Messaging 吗? 最佳答案 作为 Spring-WS 的领
我在 Tomcat 上部署了一个有状态的 Web 服务。它由工厂服务和主要 API 服务组成,并且运行良好。工厂服务向主 API 实例返回一个 W3CEndpointReference,客户端使用该
我构建了一个最小的 Web 服务并使用 javax.xml.ws.Endpoint 发布它。如果我尝试在http://localhost:1234/AddService?wsdl 工作正常。 尝试在
我正在实现一个必须支持 WS-Trust (1.3) 的网络服务。 OASIS 将 WS-Trust 的命名空间定义为 http://docs.oasis-open.org/ws-sx/ws-trus
有没有办法操纵jaxws中使用的编码器。 我喜欢发送一个在网络服务请求中提交的 cdata,为此我想在这里尝试类似描述的东西:http://odedpeer.blogspot.de/2010/07/j
我正在尝试使用 Spring-WS 构建一个简单的 Web 服务客户端,但遇到了麻烦。我试图调用的 SOAP 服务使用 HTTP 基本身份验证以确保安全。 使用 Spring-WS 教程示例,我已经配
我对网络服务很陌生。我找不到 JAX-WS 和 CXF 之间的区别。 据我了解,JAX-WS是java提供的规范,CXF是实现。 如果我错了,请纠正我。 最佳答案 是的,你是对的。 JAX-WS 是基
我有很多用 @WebService(targetNamespace = "mynamespace") 注释的端点.每@WebResult和 @WebParam与 targetNamespace = "
根据网上的文献,我可以看到有两个与Web Services Eventing相关的规范: WS-BaseNotification - 由 OASIS 于 2004 年提交 WS-Eventing -
对于基于 SOAP 的 Web 服务,为什么应该选择 Spring WS 而不是 JAX-WS。我已经阅读了一些文章,甚至 Spring WS 文档功能,但我仍然不清楚。如果我需要说服某人使用 Spr
我使用 wsimport 创建了一个肥皂客户端,我需要将消息中字符串字段内的 xml 数据发送到网络服务器。我知道我实际上并不需要在 Web 服务调用中使用 cdata,但 Web 服务需要此字段位于
我想问一下如何在 JAX-WS 中指定 SOAP Web 服务,这样如果我希望通过值 A 或值B。任何其他值都应该返回错误。 我想到了Java中的枚举,但仍然不知道如何实现它。 有什么建议吗?预先感谢
我用 pyws 实现了一个简单的 Web 服务服务器和 suds . from pyws.server import SoapServer from pyws.functions.register i
我正在 javascript 中使用 websockets 连接到 API 并读取一些数据。 这是代码: function processHUOBIAPI(client,exchange_name,e
我正在使用 Node.js + Express + Express-ws 创建应用程序,但在连接后尝试发送消息时出现以下错误: const fs = require('fs'); con
我是一名优秀的程序员,十分优秀!