我开始使用JAX WS学习java Web服务。我正在阅读的这本书的第一章展示了如何仅使用java SE构建和部署简单的jax ws Web服务。特别是,Web服务是通过Endpoint类发布的。发布Web服务后,作者指定“开箱即用,端点发布者处理一个客户端请求 时间...如果给定请求的处理应该挂起,那么 所有其他客户端请求都被有效阻止。最后的一个例子 本章的内容显示了Endpoint如何同时处理请求 所以一个挂起的请求不会阻止其他请求。“
为了看到这一点,我尝试向两个线程的Web服务发送两个请求。这是代码:
@WebService(endpointInterface = "ProveVelociJava.WS.MyWsWithJavaSE.SayHello")
public class SayHelloImpl implements SayHello {
public String greetings(String param) {
System.out.println("\nStarting " + param + "...\n");
if(param.equals("miao")) {
try {
Thread.sleep(9000);
}
catch(Exception e) {}
}
System.out.println("Ended " + param + "\n\n");
return "Hi, " + param;
}
}
public class SayHelloPublisher {
public static void main(String[ ] args) {
// 1st argument is the publication URL
// 2nd argument is an SIB instance
Endpoint.publish("http://127.0.0.1:9899/say", new SayHelloImpl());
}
}
class MyClient extends Thread {
private static URL url;
private static QName qname;
private static Service service;
private static SayHello eif;
static {
try {
url = new URL("http://127.0.0.1:9899/say?wsdl");
qname = new QName("http://MyWsWithJavaSE.WS.ProveVelociJava/", "SayHelloImplService");
service = Service.create(MyClient.url, MyClient.qname);
// Extract the endpoint interface, the service "port".
eif = service.getPort(SayHello.class);
}
catch(Exception e) {}
}
private String name;
public MyClient(String n) {
name = n;
}
public void run() {
System.out.println(MyClient.eif.greetings(this.name));
}
public static void main(String args[ ]) throws Exception {
MyClient t1 = new MyClient("miao");
MyClient t2 = new MyClient("bau");
t1.start();
t2.start();
}
}
如果我启动MyClient类,名为“miao”的线程会发送请求然后进入休眠状态。但是,名为“bau”的线程不会等待前一个线程,并且会立即满足其请求。
我错过了什么吗?可以使用java线程来模拟多个请求吗?
非常感谢你的帮助, 尼科。
答案 0 :(得分:2)
我下载了JAX-WS 2.0规范,它反驳了我正在读的书所说的内容:
“端点由充当Web服务的对象组成 实现(这里称为implementsor)加上一些配置 信息......通常会调用端点来提供服务 并发请求,因此应该编写它的实现者 支持多线程。 synchronized关键字可以用作 通常用于控制对关键代码段的访问。更精细 控制用于调度传入请求的线程, 应用程序可以直接设置要使用的执行程序“
(http://jcp.org/aboutJava/communityprocess/final/jsr224/index.html,第5.2.2节“出版”,第67页)
这本书谈到了JAX-WS 2.1,但我没有设法下载该版本的规范。无论如何,JAX-WS 2.2规范(当前版本)确认了Endpoint类的并发性质(它包含与上面相同的句子)。
我不知道这本书的作者究竟是什么意思。
答案 1 :(得分:2)
默认情况下,Endpoint同时处理多个请求,并为每个新的并发请求创建单独的线程。您可以在WebMethod实现中插入以下行,以查看线程ID:
的System.out.println(Thread.currentThread()的toString());
如果您逐个请求,则会打印相同的线程信息,因为已终止的线程将在下一个http请求中重用。但是如果您同时发出多个请求,那么您将看到不同的线程将调用您的web方法。
只要您的webmethod没有同步块,整个webmethod未同步或您正在使用某种信号量,一个请求将不会阻止任何其他并发的请求。
答案 2 :(得分:1)
迟到但可能会有所帮助。
Endpoint.publish(Url,ServiceImplObj)在给定的网址上发布网络服务。 没有。分配给请求处理的线程真正受jvm的控制,因为这是一个轻量级部署,由jvm本身处理。
为了更好地说明,您可以在服务端打印当前线程名称,您可以看到服务线程是由jvm管理的线程池分配的。
[pool-1-thread-1]: Response[57]:
[pool-1-thread-5]: Response[58]:
[pool-1-thread-4]: Response[59]:
[pool-1-thread-3]: Response[60]:
[pool-1-thread-6]: Response[61]:
[pool-1-thread-6]: Response[62]: