JAX WS多个请求管理

时间:2012-07-20 22:15:02

标签: jax-ws

我开始使用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线程来模拟多个请求吗?

非常感谢你的帮助, 尼科。

3 个答案:

答案 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]: