java - 当我运行 Atmosphere 样本的 Atmosphere 聊天多房间时,@Message注释不起作用

标签 java servlets intellij-idea websocket atmosphere

我已经搜索了这个问题很长一段时间,但没有找到适合我的解决方案。

这里是一些代码(我刚刚添加了一些 System.out.println 短语)。

web.xml

<display-name>Atmosphere Chat</display-name>
<servlet>
    <description>AtmosphereServlet</description>
    <servlet-name>AtmosphereServlet</servlet-name>
    <servlet-class>org.atmosphere.cpr.AtmosphereServlet</servlet-class>
    <load-on-startup>0</load-on-startup>
    <async-supported>true</async-supported>
</servlet>
<servlet-mapping>
    <servlet-name>AtmosphereServlet</servlet-name>
    <url-pattern>/chat/*</url-pattern>
</servlet-mapping>

ChatRoom.java

@ManagedService(path = "/chat/{room}")
public class ChatRoom {

    private final ConcurrentHashMap<String, String> users = new ConcurrentHashMap<String, String>();

    private final static String CHAT = "/chat/";

    @PathParam("room")
    private String room;

    @Inject
    private BroadcasterFactory factory;

    @Inject
    private AtmosphereResourceFactory resourceFactory;

    @Inject
    private MetaBroadcaster metaBroadcaster;

    private static Collection<String> getRooms(Collection<Broadcaster> broadcasters) {
        Collection<String> result = new ArrayList<String>();
        for (Broadcaster broadcaster : broadcasters) {
            if (!("/*".equals(broadcaster.getID()))) {
                // if no room is specified, use ''
                String[] p = broadcaster.getID().split("/");
                result.add(p.length > 2 ? p[2] : "");
            }
        };
        return result;
    }
    @Ready(encoders = {JacksonEncoder.class})
    @DeliverTo(DeliverTo.DELIVER_TO.ALL)
    public ChatProtocol onReady(final AtmosphereResource resource) {
        System.out.println("Client " + resource.uuid() + " connected.");
        return new ChatProtocol(users.keySet(), getRooms(factory.lookupAll()));
    }

    @Disconnect
    public void onDisconnect(AtmosphereResourceEvent event) {
        if (event.isCancelled()) {
            System.out.println("Client " + event.getResource().uuid() + " unexpectedly disconnected.");
        } else if (event.isClosedByClient()) {
            System.out.println("Client " + event.getResource().uuid() + " closed the connection");
        }
    }

    @Message(decoders = {UserDecoder.class})
    public void onPrivateMessage(UserMessage user) {
        System.out.println("Client " + user.getUser() + " send a message.");
        String userUUID = users.get(user.getUser());
        if (userUUID != null) {
            // Retrieve the original AtmosphereResource
            AtmosphereResource r = resourceFactory.find(userUUID);

            if (r != null) {
                ChatProtocol m = new ChatProtocol(user.getUser(), " sent you a private message: " + user.getMessage().split(":")[1], users.keySet(), getRooms(factory.lookupAll()));
                if (!user.getUser().equalsIgnoreCase("all")) {
                    factory.lookup(CHAT + room).broadcast(m, r);
                }
            }
        } else {
            ChatProtocol m = new ChatProtocol(user.getUser(), " sent a message to all chatroom: " + user.getMessage().split(":")[1], users.keySet(), getRooms(factory.lookupAll()));
            metaBroadcaster.broadcastTo("/*", m);
        }
    }

    @Message(encoders = {JacksonEncoder.class}, decoders = {ProtocolDecoder.class})
    public ChatProtocol  onMessage(ChatProtocol message) throws IOException {
        if (message.getMessage().contains("disconnecting")) {
            users.remove(message.getAuthor());
            return new ChatProtocol(message.getAuthor(), " disconnected from room " + room, users.keySet(), getRooms(factory.lookupAll()));
        }

        if (!users.containsKey(message.getAuthor())) {
            users.put(message.getAuthor(), message.getUuid());
            return new ChatProtocol(message.getAuthor(), " entered room " + room, users.keySet(), getRooms(factory.lookupAll()));
        }

        message.setUsers(users.keySet());
        return new ChatProtocol(message.getAuthor(), message.getMessage(), users.keySet(), getRooms(factory.lookupAll()));
    }
}

当我在 IntelliJ IDEA 中运行此示例时,Websocket 可以建立。除了 @Message 注释之外,所有功能都工作正常,我可以选择一个聊天室意味着 @Ready 注释可以工作,导致控制台日志打印如下。

Client 6c981440-0ae6-43ed-be40-95dfad23d586 connected.

当我关闭浏览器时,控制台日志为

Client 6c981440-0ae6-43ed-be40-95dfad23d586 closed the connection

这意味着@Disconnect有效。

但我在 input 中输入一个名称,然后单击 Enter。什么都没发生。 Choose Name

任何人都可以提供帮助,我们将不胜感激。提前致谢。

PS:运行环境为JDK1.7,tomcat7,atmosphere-runtime版本为2.4.3。我昨天对此进行了调试,在输入名称后,IllegalArgumentException 在下面的代码中被捕获。

    public static Object invokeMethod(Method method, Object objectToInvoke, Object... parameters) {
    Object objectToEncode = null;
    boolean hasMatch = false;

    try {
        objectToEncode = method.invoke(objectToInvoke, method.getParameterTypes().length == 0?new Object[0]:parameters);
        hasMatch = true;
    } catch (IllegalAccessException var6) {
        logger.trace("", var6);
    } catch (InvocationTargetException var7) {
        logger.error("", var7);
    } catch (IllegalArgumentException var8) {
        logger.trace("", var8);
    } catch (Throwable var9) {
        logger.error("", var9);
    }

    if(!hasMatch) {
        logger.trace("No Method\'s Arguments {} matching {}", method.getName(), objectToInvoke);
    }

    return objectToEncode;
}

哪些代码位于org\atmosphere\atmosphere-runtime\2.4.3\atmosphere-runtime-2.4.3.jar!\org\atmosphere\config\driven\Invoker.class

最佳答案

我遇到过这个问题,可能是编码器解码器导致了这个问题,请尝试将您的方法参数更改为String

关于java - 当我运行 Atmosphere 样本的 Atmosphere 聊天多房间时,@Message注释不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36783978/

相关文章:

java - 使用 intellij 在 GUI 设计器中打开 .form 文件

java - 类路径有问题

java - 将文件保存到 Jersey API 中的服务器应用程序目录

java - 无法修复 java.lang.ArrayIndexOutOfBoundsException

java - 如何告诉 hibernate validator 仅验证一个注释?

java - 如何从 servlet 获取数据到 JSP 页面?

java - jetty AsyncProxyServlet/AsyncMiddleManServlet

java - Intellij 中出现错误 "Package org.eclipse.... does not exist"

android-studio - IntelliJ : Missing messages window option in "tool window" menu

Java 显示带有格式化字符串的 Jlabel.setText