2015-03-19 80 views
0

我想實現類似下面的東西。有兩臺機器(比如他們的ips是A和B)監聽固定端口上的數據(比如P和Q)。我希望他們在他們之間發送消息,而不必關心他是否在聽。我查看套接字,但它遵循客戶端服務器模式,這不符合我的要求。兩臺機器之間的Java網絡通信

例如,我正在尋找的是這樣的。

B則以下

Communicator c = new Communicator(A,P); //does not block 
c.write(byteArray); //does not block 

while(true) 
{ 
    if(c.hasData()) 
    { 
     bytes[] bytes = c.readData(); 
    } 
} 

A不一樣的。 A和B都在一個循環內的不同線程中讀寫,並且不關心另一端是否正在接收數據。

更多解釋:問題是一臺機器可能會超出wifi範圍。如果它在範圍內,我希望它收到消息。

我想你可以說我試圖在沒有服務器的兩臺機器和已知端口之間進行通信。

是否有一個在Java中可以使用的API?或者一些外部圖書館?

+0

如果其中一個沒有在監聽,那麼試圖發送消息的時候沒有人會收到它? – vandale 2015-03-19 06:07:05

+0

我添加了一個解釋。問題在於一臺機器可能會超出wifi範圍。如果它在範圍內,我希望它收到消息。「 – 2015-03-19 06:10:10

回答

0

找到了如何使用DatagramChannelhttp://tutorials.jenkov.com/java-nio/datagram-channel.html的幫助下完成此操作。

以下是驗證碼。

import java.nio.ByteBuffer; 
import java.nio.channels.DatagramChannel; 
import java.nio.charset.StandardCharsets; 

public class ReaderThread extends Thread { 

    private DatagramChannel channel; 

    public ReaderThread(DatagramChannel channel) { 
     this.channel = channel; 
    } 

    @Override 
    public void run() { 
     try { 
      ByteBuffer packet = ByteBuffer.allocate(65536); 
      while (true) { 
       packet.clear(); 
       if (channel.receive(packet) != null) { 
        packet.flip(); 
        int remaining = packet.remaining(); 
        byte[] b = new byte[remaining]; 
        packet.get(b, 0, remaining); 
        System.out.println(new String(b, StandardCharsets.UTF_8)); 
       } 
      } 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 

    } 
} 


import java.net.InetSocketAddress; 
import java.net.SocketAddress; 
import java.nio.ByteBuffer; 
import java.nio.channels.DatagramChannel; 
import java.nio.charset.StandardCharsets; 

public class WriterThread extends Thread { 
    private SocketAddress target; 
    private DatagramChannel channel; 

    public WriterThread(DatagramChannel channel, int othersPort, String othersIp) { 
     super(); 
     this.target = new InetSocketAddress(othersIp, othersPort); 
     this.channel = channel; 
    } 

    @Override 
    public void run() { 
     try { 
      while (true) { 
       ByteBuffer buf = ByteBuffer.allocate(48); 
       buf.clear(); 
       buf.put("Hello world!".getBytes(StandardCharsets.UTF_8)); 
       buf.flip(); 
       channel.send(buf, target); 
       Thread.sleep(1000); 
      } 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 

    } 
} 

import java.io.IOException; 
import java.net.InetSocketAddress; 
import java.nio.channels.DatagramChannel; 

public class Main { 
    public static void main(String[] args) throws IOException { 
     int portA = 1091; 
     String machineA = "localhost";// "10.101.2.40"; 
     int portB = 1092; 
     String machineB = "localhost";// "10.101.2.39"; 

     DatagramChannel channelA = DatagramChannel.open(); 
     channelA.socket().bind(new InetSocketAddress(portA)); 
     channelA.configureBlocking(false); 
     ReaderThread readerA = new ReaderThread(channelA); 
     WriterThread writerA = new WriterThread(channelA, portB, machineB); 
     readerA.start(); 
     writerA.start(); 

     DatagramChannel channelB = DatagramChannel.open(); 
     channelB.socket().bind(new InetSocketAddress(portB)); 
     channelB.configureBlocking(false); 
     ReaderThread readerB = new ReaderThread(channelB); 
     WriterThread writerB = new WriterThread(channelB, portA, machineA); 
     readerB.start(); 
     writerB.start(); 
    } 
} 
相關問題