Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/335.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java kryonet侦听器未接收到消息?_Java_Junit_Kryonet - Fatal编程技术网

Java kryonet侦听器未接收到消息?

Java kryonet侦听器未接收到消息?,java,junit,kryonet,Java,Junit,Kryonet,因此,我在测试我的网络时遇到了一个问题,下面有很多代码,但这是我可以举个例子的最小部分。我遇到的问题是,当我发送消息时,我在服务器和客户端注册的侦听器似乎从未被调用 lock.await(5000,TimeUnit.ms)应等待消息收到或超时5秒。但是,在设置了防止测试失败的响应变量后,侦听器应该告诉锁继续lock.countDown() 正如您所猜测的,这不会发生,消息会被发送,锁会由于超时而继续,测试会失败,因为response为null import com.esotericsoftwar

因此,我在测试我的网络时遇到了一个问题,下面有很多代码,但这是我可以举个例子的最小部分。我遇到的问题是,当我发送消息时,我在服务器和客户端注册的侦听器似乎从未被调用

lock.await(5000,TimeUnit.ms)
应等待消息收到或超时5秒。但是,在设置了防止测试失败的
响应
变量后,侦听器应该告诉锁继续
lock.countDown()

正如您所猜测的,这不会发生,消息会被发送,锁会由于超时而继续,测试会失败,因为
response
为null

import com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryonet.Client;
import com.esotericsoftware.kryonet.Connection;
import com.esotericsoftware.kryonet.Listener;
import com.esotericsoftware.kryonet.Server;
import com.esotericsoftware.minlog.Log;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;

import static junit.framework.TestCase.fail;

public class TestMessage {
    private Server server;
    private Client client;
    private String response;
    private CountDownLatch lock = new CountDownLatch(1);

    @Before
    public void setUp() {
        Log.set(Log.LEVEL_DEBUG);

        server = new Server();
        try {
            Log.debug("Binding to port: " + 30454 + "... ");
            server.bind(30454);
            Log.debug("Bound to port" + 30454);
        } catch (IOException e) {
            Log.debug("failed");
            Log.error("Unable to bind to port: " + e.getMessage());
            server.stop();
            fail("Unable to start server");
        }
        Kryo kryo = server.getKryo();
        kryo.register(Message.class);
        Log.debug("Adding server listener");
        server.addListener(new TestListener());
        Log.debug("Starting server... ");
        server.start();
        Log.debug("Server started successfully");
    }

    @Test
    public void testPacketSending() throws IOException, InterruptedException {
        client = new Client();
        client.addListener(new TestListener());
        Kryo kryo = client.getKryo();
        kryo.register(Message.class);
        client.start();
        client.connect(5000, "127.0.0.1", 30454);

        client.sendTCP(new Message("RECEIVED"));

        lock.await(5000, TimeUnit.MILLISECONDS);

        Assert.assertNotNull(response);
    }

    private class TestListener extends Listener {
        @Override
        public void received(Connection connection, Object o) {
            Message m = (Message) o;
            response = m.message;
            Log.debug(m.message);
            lock.countDown();
        }
    }

    private class Message {
        String message;

        Message(String message) {
            this.message = message;
        }
    }
}

这里的问题是:

00:00 ERROR: [kryonet] Error reading TCP from connection: Connection 1
com.esotericsoftware.kryonet.KryoNetException: Error during deserialization.
    at com.esotericsoftware.kryonet.TcpConnection.readObject(TcpConnection.java:141)
    at com.esotericsoftware.kryonet.Server.update(Server.java:205)
    at com.esotericsoftware.kryonet.Server.run(Server.java:372)
    at java.lang.Thread.run(Thread.java:745)
Caused by: com.esotericsoftware.kryo.KryoException: Class cannot be created (non-static member class): org.glytching.sandbox.kryo.TestMessage$MessageA
    at com.esotericsoftware.kryo.Kryo$DefaultInstantiatorStrategy.newInstantiatorOf(Kryo.java:1308)
    at com.esotericsoftware.kryo.Kryo.newInstantiator(Kryo.java:1127)
    at com.esotericsoftware.kryo.Kryo.newInstance(Kryo.java:1136)
    at com.esotericsoftware.kryo.serializers.FieldSerializer.create(FieldSerializer.java:562)
    at com.esotericsoftware.kryo.serializers.FieldSerializer.read(FieldSerializer.java:538)
    at com.esotericsoftware.kryo.Kryo.readClassAndObject(Kryo.java:816)
    at com.esotericsoftware.kryonet.KryoSerialization.read(KryoSerialization.java:55)
    at com.esotericsoftware.kryonet.TcpConnection.readObject(TcpConnection.java:139)
    ... 3 more
Kryonet无法反序列化为
消息
,因为(a)它是一个非静态的内部类,并且(b)它没有公共的零参数构造函数。FWIW,有一些关于为什么Kryonet不支持非静态内部类的序列化

如果您(a)将
消息
重构为它自己的类或使其成为静态的,并且(b)给它一个零参数构造函数,那么您的测试将通过

使您的测试通过所需的最小更改是替换此

private class Message {
    String message;

    Message(String message) {
        this.message = message;
    }
}
。。。为此:

private static class Message {
    String message;

    Message() {

    }

    Message(String message) {
        this.message = message;
    }
}
关于这一点:

下面有很多代码,但这是我所能举的一个例子


您问题中的复制案例已经足够了,谢谢。

啊,所以所有的Kryonet消息类都需要是静态的?只有当它们是内部类时。嗯,谢谢。因为只有这个示例使用内部类,所以我似乎无法重现我的问题。可能是因为我的类没有0参数构造函数。您是如何获得此错误输出的?当我设置
Log.set(Log.LEVEL\u ERROR)
时,我没有看到控制台的输出。我只是使用您提供的代码运行了测试。我没有改变你提供的东西。然后,当问题出现时,我添加了static modified和zero and构造函数,然后测试通过了。我修复了实际问题,因为我的类都没有0个arg构造函数。