我想向地图中的所有用户发送一条消息。

    for (User u : _userMap.values()) {
        u.getMessages().add(data);

        u.getKey().interestOps(SelectionKey.OP_WRITE);
    }


但是当我运行此功能时,我看到


  线程“主”中的异常java.lang.IllegalArgumentException


这行出错

u.getKey().interestOps(SelectionKey.OP_WRITE);


getKey()返回SelectionKey,getMessages返回ArrayList,数据是一个byte []数组,其中包含我使用channel.read(buffer)读取的消息。

更多信息:

在构造函数中,我使选择器

_selector = Selector.open();


我运行服务器

public void startServer() throws IOException {
    while (true) {
        _selector.select();

        Iterator<SelectionKey> keys = _selector.selectedKeys().iterator();

        while (keys.hasNext()) {
            SelectionKey key = keys.next();
            keys.remove();

            if (!key.isValid())
                continue;
            if (key.isAcceptable())
                accept(key);
            else if (key.isReadable())
                read(key);
            else if (key.isWritable())
                write(key);
        }
    }
}


我接受连接

private void accept(SelectionKey key) throws IOException {
    ServerSocketChannel serverChannel = (ServerSocketChannel) key.channel();
    SocketChannel channel = serverChannel.accept();
    channel.configureBlocking(false);

    User u = new User(key);
    _userMap.put(channel, u);

    channel.register(_selector, SelectionKey.OP_READ);
}


在读取功能中,当我读取消息时,每个循环都有此消息。但是,当您是一个用户,并且我在循环后面移动了有兴趣操作的行时,它就可以工作。

        //u.getKey().interestOps(SelectionKey.OP_WRITE);
    }
    key.interestOps(SelectionKey.OP_WRITE);


完整的读写功能:

private void read(SelectionKey key) throws IOException {
    SocketChannel channel = (SocketChannel) key.channel();

    ByteBuffer buffer = ByteBuffer.allocate(2048);
    int read = -1;

    try {
        read = channel.read(buffer);
    } catch (Exception e) {
        e.printStackTrace();
    }

    if (read == -1) {
        _userMap.remove(channel);

        channel.close();
        key.cancel();

        return;
    }

    byte[] data = new byte[read];

    System.arraycopy(buffer.array(), 0, data, 0, read);

    /// WYSyŁA DO WSZYSTKICH. usunąć

    for (User u : _userMap.values()) {
        u.getMessages().add(data);

        u.getKey().interestOps(SelectionKey.OP_WRITE);
    }
    //key.interestOps(SelectionKey.OP_WRITE);

    ///////
}

private void write(SelectionKey key) throws IOException {
    SocketChannel channel = (SocketChannel) key.channel();

    ArrayList<byte[]> msg = _userMap.get(channel).getMessages();
    Iterator<byte[]> i = msg.iterator();

    while (i.hasNext()) {
        byte[] item = i.next();
        i.remove();

        channel.write(ByteBuffer.wrap(item));
    }

    key.interestOps(SelectionKey.OP_READ);
}


解:

我现在无法回答自己的问题,因此请放在这里:

接受方法中的SelectionKey有一点障碍。我试图用read方法中的新密钥替换它,并且它可以工作。因此,在User类中,我不再保留SelectionKey var,现在保留SocketChannel。 SocketChannel有keyFor方法,所以当我有选择器时,我可以获得键

        u.getChannel().keyFor(_selector).interestOps(SelectionKey.OP_WRITE);

最佳答案

听起来可能与记录的完全一样:


  投掷
  IllegalArgumentException-如果集合中的某个位与该键的通道支持的操作不对应,即set & ~(channel().validOps()) != 0


在不了解有关频道的情况下,很难知道为什么会这样...

09-10 05:57