zoukankan      html  css  js  c++  java
  • java-NIO-DatagramChannel(UDP)

    Java NIO中的DatagramChannel是一个能收发UDP包的通道。因为UDP是无连接的网络协议,所以不能像其它通道那样读取和写入。它发送和接收的是数据包。

    打开 DatagramChannel

    下面是 DatagramChannel 的打开方式:

    DatagramChannel channel = DatagramChannel.open();
    channel.socket().bind(new InetSocketAddress(9999));

    这个例子打开的 DatagramChannel可以在UDP端口9999上接收数据包。

    接收数据

    通过receive()方法从DatagramChannel接收数据,如:

    ByteBuffer buf = ByteBuffer.allocate(48);
    buf.clear();
    channel.receive(buf);

    receive()方法会将接收到的数据包内容复制到指定的Buffer. 如果Buffer容不下收到的数据,多出的数据将被丢弃。

    发送数据

    通过send()方法从DatagramChannel发送数据,如:

    String newData = "New String to write to file..." + System.currentTimeMillis();
    ByteBuffer buf = ByteBuffer.allocate(48);
    buf.clear();
    buf.put(newData.getBytes());
    buf.flip();
    int bytesSent = channel.send(buf, new InetSocketAddress("jenkov.com", 80));

    这个例子发送一串字符到”jenkov.com”服务器的UDP端口80。 因为服务端并没有监控这个端口,所以什么也不会发生。也不会通知你发出的数据包是否已收到,因为UDP在数据传送方面没有任何保证。

    连接到特定的地址

    可以将DatagramChannel“连接”到网络中的特定地址的。由于UDP是无连接的,连接到特定地址并不会像TCP通道那样创建一个真正的连接。而是锁住DatagramChannel ,让其只能从特定地址收发数据。

    这里有个例子:

    channel.connect(new InetSocketAddress("jenkov.com", 80));

    当连接后,也可以使用read()和write()方法,就像在用传统的通道一样。只是在数据传送方面没有任何保证。这里有几个例子:

    int bytesRead = channel.read(buf);
    int bytesWritten = channel.write(but);

    完整实例

    服务端: 接收客户端发送消息,收取到消息后,给发送方一个回应

    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.net.SocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.DatagramChannel;
    
    /**
     * 服务端: 接收客户端发送消息,收取到消息后,给发送方一个回应
     */
    public class TestDataDramdChannelReceive {
        public static void main(String[] args) throws IOException {
            // 获取通道
            DatagramChannel datagramChannel = DatagramChannel.open();
            // 绑定端口
            datagramChannel.bind(new InetSocketAddress(8989));
            // 分配Buffer
            ByteBuffer buffer = ByteBuffer.allocate(1024);
            byte b[];
            while (true){
                // 清空Buffer
                buffer.clear();
                // 接受客户端发送数据
                SocketAddress socketAddress = datagramChannel.receive(buffer);
                if (socketAddress != null) {
                    int position = buffer.position();
                    b = new byte[position];
                    buffer.flip();
                    for(int i=0; i<position; ++i) {
                        b[i] = buffer.get();
                    }
                    System.out.println("receive remote " +  socketAddress.toString() + ":"  + new String(b, "UTF-8"));
                    //接收到消息后给发送方回应
                    sendReback(socketAddress,datagramChannel);
                }
            }
        }
        public static void sendReback(SocketAddress socketAddress, DatagramChannel datagramChannel) throws IOException {
            String message = "I has receive your message";
            ByteBuffer buffer = ByteBuffer.allocate(1024);
            buffer.put(message.getBytes("UTF-8"));
            buffer.flip();
            datagramChannel.send(buffer, socketAddress);
        }
    }

    客户端: 发送控制台输入的内容并接收服务端回应

    import java.io.IOException;
    import java.io.UnsupportedEncodingException;
    import java.net.InetSocketAddress;
    import java.net.SocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.DatagramChannel;
    import java.util.Scanner;
    
    /**
     * 客户端: 发送控制台输入的内容并接收服务端回应
     */
    public class TestDataGramdChannelSend {
        public static void main(String[] args) throws IOException {
            final DatagramChannel channel = DatagramChannel.open();
            //接收消息线程
            new Thread(new Runnable() {
                @Override
                public void run() {
                    ByteBuffer buffer = ByteBuffer.allocate(1024);
                    byte b[];
                    while(true) {
                        buffer.clear();
                        SocketAddress socketAddress = null;
                        try {
                            socketAddress = channel.receive(buffer);
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                        if (socketAddress != null) {
                            int position = buffer.position();
                            b = new byte[position];
                            buffer.flip();
                            for(int i=0; i<position; ++i) {
                                b[i] = buffer.get();
                            }
                            try {
                                System.out.println("receive remote " +  socketAddress.toString() + ":"  + new String(b, "UTF-8"));
                            } catch (UnsupportedEncodingException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }).start();
    
            //发送控制台输入消息
            while (true) {
                Scanner sc = new Scanner(System.in);
                String next = sc.next();
                try {
                    sendMessage(channel, next);
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    
        public static void sendMessage(DatagramChannel channel, String mes) throws IOException {
            if (mes == null || mes.isEmpty()) {
                return;
            }
            ByteBuffer buffer = ByteBuffer.allocate(1024);
            buffer.clear();
            buffer.put(mes.getBytes("UTF-8"));
            buffer.flip();
            System.out.println("send msg:" + mes);
            int send = channel.send(buffer, new InetSocketAddress("localhost",8989));
        }
    }

    运行服务端程序,然后再运行客户端程序,在客户端控制台输入数据,按回车键发送,在服务端控制台可以收到发送的数据。

    源码地址:https://github.com/qjm201000/io_nio_datagramchannel.git

    转自:http://ifeve.com/datagram-channel/

  • 相关阅读:
    关于数据库的alter table操作和索引概念
    mysql_fetch_array()和 mysql_fetch_array()的区别
    left 截取
    学会设置五大类MySQL参数
    MySQL性能优化的最佳20+条经验
    varchar to int error
    2003服务器重启
    缺少注释的结尾标记 '*/'。 '*' 附近有语法错误。
    2003的服务器终端服务器超出最大连接数的解决办法转载
    电脑报警音解读转载
  • 原文地址:https://www.cnblogs.com/qjm201000/p/10296012.html
Copyright © 2011-2022 走看看