Java网络编程基础(BIO)

[TOC]

TCP/IP协议分层模型

  • TCP/IP分层模型
自顶向下 协议 功能 PDU(协议数据单元)
应用层 HTTP、SSH、FTP、TELNET等等 通过应用进程间的交互来完成特定网络应用 报文
运输层 TCP、UDP 在应用程序端点之间传输应用层报文 报文段
网络层 IP 将数据包从一台主机传输到另一台主机 数据报
链路层 以太网、wifi、电缆接入网的DOCSIS等 数据链路层将网络层交下来的IP数据报组装成帧,在两个相邻节点间的链路上传送帧
物理层 与传输媒介相关 实现相邻计算机节点之间比特流的透明传送,尽可能屏蔽掉具体传输介质和物理设备的差异 比特

下层提供接口供上层使用,为上层服务。

Socket简述

Socket是运输层提供给应用层的接口,使得应用层可以基于socket接口实现基于TCP/UDP协议的通信。它处于应用层和运输层之间。Socket是面向应用程序端点的,封装了IP地址和端口号,二者标识了网络主机上的某一个应用程序,IP地址标识网络主机,端口号标识主机上的应用程序,这样唯一指定了通信地址,便可以发起应用程序端之间的网络通信。

在操作系统层面,socket是抽象成文件的,它和文件相似,都是数据来源,只不过socket底层还有网络通信链路和各种底层协议的支持,可以简单的看成一个远端的文件,使用socket打开文件,对这个特殊的文件进行读写操作和交互,从而达到通信(分享)的目的。

基于BIO的TCP通信

当客户端想要打开一个连接到服务器的TCP/IP连接时,就要使用到Java Socket类。socket类只需要被告知连接的IP地址和TCP端口,其余的都有Java实现。

假如我们想要打开一个监听服务,来监听客户端连接某些指定TCP端口的连接,那就需要使用Java ServerSocket类。当客户端通过Socket连接服务器端的ServerSocket监听时,服务器端会指定这个连接的一个Socket,此时客户端与服务器端间的通信就变成Socket与Socket之间的通信。

ServerSocket类

ServerSocket,由其名称即可知,是作为服务端的socket对象,在网络编程中,它又称为欢迎套接字,专门监听主机上的某个端口,如果有连接到此端口,则产生一个与该连接相连的Socket类对象。

Socket类是对网络套接字的封装,也即为主机应用端通信接口的抽象(可能是远程也可能是本地,但是是通过网络来通信的进程)。

当socket对象之间连接建立成功,进程持有的socket即可以抽象为远端文件,通过对这个文件进行读写进行数据交换也就是所谓的通信。

ServerSocket的使用目的是监听主机端口,接收到连接后返回一个Socket对象建立通信,这是基于TCP的可靠连接通信。UDP通信中无需双方建立连接。

  • ServerSocket(缩写成SS)的核心属性:
    1. bindAddr, 宿主主机的IP地址
    2. port,监听的端口号
    3. backlog,请求传入连接队列的最大长度
  • 核心API
  1. 实例化对象
// 创建一个未绑定的SS.此方法创建的SS需要调用bind方法绑定端口
public ServerSocket() throws IOException;

// 创建一个绑定到指定端口号的SS
// 端口号指定为0,则自动选择
// backlog=50
public ServerSocket(int port) throws IOException {
        this(port, 50, null);
    }

public ServerSocket(int port, int backlog) throws IOException;

// 核心方法
// @param bindAddr the local InetAddress the server will bind to
// InetAddress 是对IP地址的封装
public ServerSocket(int port, int backlog, InetAddress bindAddr) 
  throws IOException ;
  1. bind
    手动绑定到端
public void bind(SocketAddress endpoint) throws IOException
public void bind(SocketAddress endpoint, int backlog) throws IOException ;

endpoint是SocketAddress类型,使用其实现类InetSocketAddress来构造地址(IP+Port)。指定端地址的创建SS方法不需要再手动绑定。

  1. 监听连接
public Socket accept() throws IOException ;

阻塞方法,在连接到达之前一直阻塞,有连接时返回一个对该连接抽象的Socket对象

  1. 其他
    获取地址信息: getInetAddress(),getLocalPort(),getLocalSocketAddress()[IP+Port]
    检查状态:isBound(),isClosed()
    关闭:close() (ServerSocket实现了Closeable接口)
    accept超时时间:setSoTimeout(int timeout),getSoTimeout()
  • 示例
public class TestServerSocket {

    private static final int PORT = 12345;
    private static final int BACKLOG = 100;

    public static void main(String[] args) throws IOException {
        ServerSocket ss = getServerSocket();
        System.out.println("server start...");
        while (true) {
            Socket socket = ss.accept();
            System.out.println("connect incoming: " 
                                        + socket.getRemoteSocketAddress());

        }
    }

    private static ServerSocket getServerSocket() throws IOException {
        ServerSocket ss = new ServerSocket(PORT);
        // or
        ss = new ServerSocket(PORT, BACKLOG);
        // or
        InetAddress bindAddr = InetAddress.getLocalHost();
        ss = new ServerSocket(PORT, BACKLOG, bindAddr);
        // or
        ss = new ServerSocket();
        SocketAddress endpoint = new InetSocketAddress("localhost", PORT);
        ss.bind(endpoint, BACKLOG);

        return ss;
    }
}

Socket类

A socket is an endpoint for communication between two machines.

  • Socket使用:
  1. 创建连接到远端端点的Socket对象(实例化和连接)
  2. 读写socket进行面向流的通信
  • 创建socket(连接)
// 创建一个未连接的socket对象,可显式调用connect连接
public Socket();
// 指定代理创建一个未连接的socket对象,可显式调用connect连接
public Socket(Proxy proxy);
// 创建一个连接到指定地址的socket对象
public Socket(String host, int port)
        throws UnknownHostException, IOException;
public Socket(InetAddress address, int port) throws IOException;
// 同时为socket绑定本地地址,也可以显式调用bind方法来绑定本地地址;默认是自动分配(端口号)
public Socket(String host, int port, InetAddress localAddr,
                  int localPort) throws IOException 

/**
连接、bind
*/
public void connect(SocketAddress endpoint) throws IOException ;
/*
* Connects this socket to the server with a specified timeout value.
* A timeout of zero is interpreted as an infinite timeout. The connection
* will then block until established or an error occurs.
*/
public void connect(SocketAddress endpoint, int timeout) throws IOException ;
// Binds the socket to a local address.
public void bind(SocketAddress bindpoint) throws IOException
  • 读写socket进行通信
socket.getInputStream();
socket.getOutputStream();
  • 示例
    private static final int PORT = 12345;
    private static final String HOST = "localhost";
    private static final int LOCAL_PORT = 23456;

    public static void main(String[] args) throws IOException {
        Socket socket = getSocket();
        // communicate
        OutputStream os = socket.getOutputStream();
        os.write("hellosocket".getBytes());
        os.flush();
        socket.close();

    }

    private static Socket getSocket() throws IOException {
        Socket sock = new Socket();
        SocketAddress endpoint = new InetSocketAddress(HOST, PORT);
        sock.connect(endpoint);
        // or sock = new Socket(HOST, PORT);
        // or sock = new Socket(InetAddress.getLocalHost(), PORT);
        // or sock = new Socket(HOST, PORT, InetAddress.getLocalHost(),
        // LOCAL_PORT);
        return sock;
    }

基于BIO的UDP通信

UDP的工作方式与TCP相比略有不同。使用UDP通信时,在客户端与服务器之间并没有建立连接的概念,客户端发送到服务器的数据,服务器可能(也可能并没有)收到这些数据,而且客户端也并不知道这些数据是否被服务器成功接收。当服务器向客户端发送数据时也是如此。

正因为是不可靠的数据传输,UDP相比与TCP来说少了很多的协议开销。

使用UDP通信不需要建立连接,没有TCP三次握手的连接动作,只需要在通信时指定消息的接收者地址即可,类似于短信,交互需要发送-接收-发送,每一步都是独立的。

DatagramSocket

DatagramSocket类实现“UDP通信”,包括客户端和服务器端。虽然UDP方式的网络通讯不需要建立专用的网络连接,但是毕竟还是需要发送和接收数据,DatagramSocket实现的就是发送数据时的发射器,以及接收数据时的监听器的角色。

  • 创建实例
// 创建一个自动分配本地端口的DatagramSocket
public DatagramSocket() throws SocketException ;
// 指定绑定的本地地址
public DatagramSocket(SocketAddress bindaddr) throws SocketException ;
// 指定本地绑定端口
public DatagramSocket(int port) throws SocketException ;
//  指定本地绑定地址和端口
public DatagramSocket(int port, InetAddress laddr) throws SocketException;

  • bind
    public synchronized void bind(SocketAddress addr) throws SocketException

  • send
    发送信息

/**
     * Sends a datagram packet from this socket. The
     * {@code DatagramPacket} includes information indicating the
     * data to be sent, its length, the IP address of the remote host,
     * and the port number on the remote host.
     *
*/
public void send(DatagramPacket p) throws IOException

发送的目的地址在报文对象内配置。

  • receive
    /**
     * Receives a datagram packet from this socket. When this method
     * returns, the {@code DatagramPacket}'s buffer is filled with
     * the data received. The datagram packet also contains the sender's
     * IP address, and the port number on the sender's machine.
     * <p>
     * This method blocks until a datagram is received. The
     * {@code length} field of the datagram packet object contains
     * the length of the received message. If the message is longer than
     * the packet's length, the message is truncated.
     * <p>
    */
    public synchronized void receive(DatagramPacket p) throws IOException 

DatagramPacket

DatagramPacket类实现对于网络中传输的数据封装,也就是说,该类的对象代表网络中交换的数据。在UDP方式的网络编程中,无论是需要发送的数据还是需要接收的数据,都必须被处理成DatagramPacket类型的对象,该对象中包含发送到的地址、发送到的端口号以及发送的内容等。其实DatagramPacket类的作用类似于现实中的信件,在信件中包含信件发送到的地址以及接收人,还有发送的内容等,邮局只需要按照地址传递即可。在接收数据时,接收到的数据也必须被处理成DatagramPacket类型的对象,在该对象中包含发送方的地址、端口号等信息,也包含数据的内容。和TCP方式的网络传输相比,IO编程在UDP方式的网络编程中变得不是必须的内容,结构也要比TCP方式的网络编程简单一些。

    /**
     * Constructs a {@code DatagramPacket} for receiving packets of
     * length {@code length}, specifying an offset into the buffer.
     * <p>
     * The {@code length} argument must be less than or equal to
     * {@code buf.length}.
     *
     * @param   buf      buffer for holding the incoming datagram.
     * @param   offset   the offset for the buffer
     * @param   length   the number of bytes to read.
     *
     * @since 1.2
     */
    public DatagramPacket(byte buf[], int offset, int length) {
        setData(buf, offset, length);
        this.address = null;
        this.port = -1;
    }

 
    public DatagramPacket(byte buf[], int length) {
        this (buf, 0, length);
    }

    /**
     * Constructs a datagram packet for sending packets of length
     * {@code length} with offset {@code ioffset}to the
     * specified port number on the specified host. The
     * {@code length} argument must be less than or equal to
     * {@code buf.length}.
     *
     * @param   buf      the packet data.
     * @param   offset   the packet data offset.
     * @param   length   the packet data length.
     * @param   address  the destination address.
     * @param   port     the destination port number.
     * @see java.net.InetAddress
     *
     * @since 1.2
     */
    public DatagramPacket(byte buf[], int offset, int length,
                          InetAddress address, int port) {
        setData(buf, offset, length);
        setAddress(address);
        setPort(port);
    }

    /**
     * Constructs a datagram packet for sending packets of length
     * {@code length} with offset {@code ioffset}to the
     * specified port number on the specified host. The
     * {@code length} argument must be less than or equal to
     * {@code buf.length}.
     *
     * @param   buf      the packet data.
     * @param   offset   the packet data offset.
     * @param   length   the packet data length.
     * @param   address  the destination socket address.
     * @throws  IllegalArgumentException if address type is not supported
     * @see java.net.InetAddress
     *
     * @since 1.4
     */
    public DatagramPacket(byte buf[], int offset, int length, 
                                      SocketAddress address) {
        setData(buf, offset, length);
        setSocketAddress(address);
    }

    public DatagramPacket(byte buf[], int length,
                          InetAddress address, int port) {
        this(buf, 0, length, address, port);
    }

    public DatagramPacket(byte buf[], int SocketAddress address) {
        this(buf, 0, length, address);
    }

指定address的用作发送的数据封装,没有address的用来封装接收到的数据。

示例

  • UDPserver
/*
* UDP 服务端
*/

public class UDPServer {
    public static void main(String[] args) throws Exception {
        // 绑定端口用于在端口监听数据到来
        DatagramSocket ds = new DatagramSocket(12345);
        // 接收数据的buffer
        byte[] buf = new byte[1024];
        DatagramPacket dp = new DatagramPacket(buf, 1024);
        // 阻塞方法,阻塞直到有数据到达
        ds.receive(dp);
        // 解析数据
        String data = new String(dp.getData(), 0, dp.getLength());
        String ip = dp.getAddress().getHostAddress();
        int port = dp.getPort();

        System.out.println("ip地址:" + ip + " 端口号:" + port + " 消息:" + data);
        // 关闭socket
        ds.close();
    }
}

  • UDPclient
public class UDPClient {
    public static void main(String[] args) throws Exception {
        // 创建socket
        DatagramSocket ds = new DatagramSocket();
        String message = "Hello Java World!";
        // 封装报文对象
        DatagramPacket dp = new DatagramPacket(message.getBytes(), 
                           message.length(), InetAddress.getByName("127.0.0.1"),
                12345);
        // 发送报文
        ds.send(dp);
        ds.close();
    }
}

参考资料
[1] Java网络教程
[2] Java网络编程:UDP通信

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 计算机网络概述 网络编程的实质就是两个(或多个)设备(例如计算机)之间的数据传输。 按照计算机网络的定义,通过一定...
    蛋炒饭_By阅读 1,217评论 0 10
  • 网络编程 网络编程对于很多的初学者来说,都是很向往的一种编程技能,但是很多的初学者却因为很长一段时间无法进入网络编...
    程序员欧阳阅读 2,012评论 1 37
  • 1 网络编程----TCPNo24 【 public class Server { public static...
    征程_Journey阅读 1,254评论 0 4
  • 我学会了包饺子。奶奶包饺子有一手,我是跟她学的。 我们先去菜场买了点面粉,这是做饺子的主要材料。然后,我们要...
    施千惠阅读 299评论 0 0
  • 不要像个落难者,告诉所有人你的不幸。总有一天你会明白,你的委屈要自己消化,你的故事不用逢人就讲起,真正理解的你没有...
    齐鲁大哥阅读 243评论 0 0