2

我在让一个极其简单的独立 Tyrus websocket 服务器工作时遇到问题。我已经让它在对我没有意义的非常具体的情况下工作。

正常工作的情况是我定义了一个顶级(在它自己的文件中)服务器端点类并使用@ServerEndpoint 对其进行注释。此类包括使用@OnOpen、@OnMessage 和@OnClose 注释的方法,这些都是典型的东西。我将此类传递给 Server 构造函数。我的简单客户端可以连接到该服务器,并成功发送服务器接收到的消息。

当我将顶级服务器端点类更改为初始化服务器的类的内部类时,就会出现问题(这是我所做的唯一更改)。在这种情况下,我的客户端可以连接并调用客户端的 @OnOpen 方法。但是服务器不会实例化服务器端点,因此它的 @OnOpen 方法永远不会被调用。显然,服务器消息接收不会发生。

Tyrus 是否要求带注释的服务器端点类不能是内部类?如果没有,服务器端点类是否有特定权限(所有内容都已公开,试图让它正常工作)?我在 Mac OSX 1.9.5 上使用 Tyrus 1.9 和 JDK 1.7。

简单服务器(包括并配置为使用失败的内部服务器端点):

package tyrus.example;

import java.util.concurrent.TimeUnit;

import javax.websocket.CloseReason;
import javax.websocket.OnClose;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;

import org.glassfish.tyrus.server.Server;

public class SimpleServer
{
    private static final String HOST_ADDR = "localhost";
    private static final int HOST_PORT = 8025;

    public static void main(String[] args) {
        Server websocketServer = new Server(HOST_ADDR, HOST_PORT, "/ws", null, InnerSimpleServerEndpoint.class);

        try {
            websocketServer.start();
            Thread.sleep(TimeUnit.MINUTES.toMillis(5));
        }
        catch (Exception e) {
            e.printStackTrace();
        }

        websocketServer.stop();

        System.out.println("Done.");
    }

    @ServerEndpoint("/myapp")
    public class InnerSimpleServerEndpoint {
        @OnOpen
        public void onOpen(Session session) {
            System.out.println("Connection received for "+session.getRequestURI());
        }

        @OnMessage
        public void onMessage(String message, Session session) {
            System.out.println("Message received: "+message);
        }

        @OnClose
        public void onClose(Session session, CloseReason closeReason) {
            System.out.println("Session closed, reason: "+closeReason);
        }
    }
}

简单的客户端:

package tyrus.example;

import java.io.IOException;
import java.net.URI;

import javax.websocket.ClientEndpointConfig;
import javax.websocket.Endpoint;
import javax.websocket.EndpointConfig;
import javax.websocket.MessageHandler;
import javax.websocket.Session;

import org.glassfish.tyrus.client.ClientManager;

public class SimpleClient
{
    private static final String DEF_WS_URL = "ws://localhost:8025/ws/myapp";

    public static void main(String[] args) {
        ClientEndpointConfig cec = ClientEndpointConfig.Builder.create().build();
        ClientManager client = ClientManager.createClient();

        try {
            client.connectToServer(new ClientEndpoint(), cec, new URI(DEF_WS_URL));
        } catch (Exception e) {
            e.printStackTrace();
        }

        System.out.println("Done.");
    }

    private static class ClientEndpoint extends Endpoint {
        @Override
        public void onOpen(Session session, EndpointConfig config) {
            System.out.println("ClientEndpoint: server session opened: "+session);

            session.addMessageHandler(new MessageHandler.Whole<String>() {
                @Override
                public void onMessage(String message) {
                    System.out.println("ClientEndpoint: received message: "+message);
                }
            });

            try {
                session.getBasicRemote().sendText("Hello server!");
            }
            catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
}

最后是在服务器使用它时工作的非内部类服务器端点:

package tyrus.example;

import javax.websocket.CloseReason;
import javax.websocket.OnClose;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;

@ServerEndpoint("/myapp")
public class SimpleServerEndpoint {
    @OnOpen
    public void onOpen(Session session) {
        System.out.println("Connection received for "+session.getRequestURI());
    }

    @OnMessage
    public void onMessage(String message, Session session) {
        System.out.println("Message received: "+message);
    }

    @OnClose
    public void onClose(Session session, CloseReason closeReason) {
        System.out.println("Session closed, reason: "+closeReason);
    }
}
4

3 回答 3

1

在服务器上,端点可以是内部类,但也必须是静态的。即使代码运行并提供服务,出于某种原因,我也只能在服务器端点也是静态的情况下使其工作。即:更改为在服务器代码中添加静态:

@ServerEndpoint("/myapp") public static class InnerSimpleServerEndpoint { }

您可能还想添加

@OnError public void onError(Session session, Throwable throwable) { //... }

于 2016-04-09T00:07:27.590 回答
0

Endpoint 可以是内部类,但它必须是可实例化的 -ClientEndpoint您失败的示例中的类是private,因此 Tyrus 无法创建它的实例。

将其更改为公开,它应该可以按预期工作。

于 2015-03-09T07:10:38.040 回答
0

内部类应该是静态的,否则如果没有外部类的实例就无法实例化。看看这个: https : //www.javatpoint.com/why-we-use-static-class-in-java 所以如果用@ServerEndpoint 注解的类不是静态的,它就不能被实例化。

于 2021-07-22T00:06:34.650 回答