java - 卡在受信号量保护的交互中

标签 java loops concurrency udp semaphore

出于学习目的,我正在使用 TCP 机制实现 UDP(以保证安全传输)。

我使用的Semaphore是二进制的,所以它的sem = new Semaphore(1);。 我使用这个信号量来控制我的 sendBuf 的入口,它是一个包含所有已发送但尚未确认的包的 List 。由于我有时会在收到 ACK 时从中删除包,因此我需要确保在另一个线程从中删除某些内容时不会使用一个线程进行迭代。

真正困扰我的是:

public void timeoutTask(long seqNum) {
    System.out.println("Timeout for package with SeqNum " + seqNum
            + " happened.");
    timeoutValue *= 2;
    try {
        System.out.println("Acquire? in timeouttask");
        sem.acquire();
        System.out.println("Acquired! in timeouttask");
    } catch (InterruptedException e1) {
        System.out.println("semaphore not acquired");
        e1.printStackTrace();
    }for (FCpacket packet : sendBuf) {

        System.out.println("Iterating!");
        if (packet.getSeqNum() == seqNum) {
            System.out.println("Package for seqNum " + seqNum + " found!");
            reSendData = packet.getSeqNumBytesAndData();
            DatagramPacket reSendPacket = new DatagramPacket(reSendData,
                    reSendData.length, hostaddress, SERVER_PORT);

            try {
                clientSocket.send(reSendPacket);
                System.out.println("Packet with seq " + seqNum
                        + " send again");
                packet.setTimestamp(0);
                startTimer(packet);
                new ReceiveHandler(reSendData, reSendData.length,
                        clientSocket, rcvData, UDP_PACKET_SIZE, this).run();
            } catch (IOException e) {
                System.out.println("Couldn't send package");
                e.printStackTrace();
            }
        }
    }
    sem.release();
    System.out.println("released! in timeouttask");

控制台输出给我以下内容:

Acquire? in timeouttask

Acquired! in timeouttask

Iterating!

Paket for seqNum 1 found!

Packet with seq 1 send again

因此它获取信号量,开始迭代,甚至发送包,所以现在它应该:再次迭代(“迭代!”)或释放信号量。上述情况都没有发生,只是卡住了。我不知道为什么 - 有什么想法吗?

最佳答案

如果ReceiveHandler是一个Thread,它应该被调用为

new ReceiveHandler(reSendData, reSendData.length, clientSocket, rcvData, UDP_PACKET_SIZE, this).start();

但如果它是一个Runnable,它应该被调用为

new Thread(new ReceiveHandler(reSendData, reSendData.length, clientSocket, rcvData, UDP_PACKET_SIZE, this)).start();

run() 不会在单独的线程中执行任务。

参见:What's the difference between Thread start() and Runnable run()

关于java - 卡在受信号量保护的交互中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40720252/

相关文章:

Javascript 实时循环数组

java - 多个Java进程可以同时读取同一个文件吗?

java - 使用 FutureTask 实现并发

java - 运行 H2(在 DB2 模式下)时的 DB2 语法(限制和脏读)

java - 用于包装请求正文的 Sling Filter

java - Hibernate JDBC 连接在一分钟后断开

javascript - 我怎样才能通过 if operator only 获得一个警报? (刚开始JavaScript学习)

java - 在 Java 中返回一对值的最佳方法是什么?

Flash 电影上传后不循环播放

java - 在不同的独立设备上生成一天有效的不同随机数?