Java网络编程之基础知识点总结

Java网络编程之基础知识点总结

一、 概述

我们知道计算机之间的通信要经过一系列复杂的过程,计算机之间通过传输介质、通信设施和网络通信协议互联,实现资源共享和数据传输。而网络编程就是使用程序使互联网的两个(或多个)计算机之间进行数据传输。当然作为Java语言,为了实现两个计算机之间的数据传输,提供了一系列的接口,使得开发人员可以方便的实现。

二 、网络体系结构

通过网络发送数据是一项复杂的操作,必须仔细地协调网络的物理特性以及所发送数据的逻辑特征。通过网络将数据从一台主机发送到另外的主机,这个过程是通过计算机网络通信来完成。

网络通信的不同方面被分解为多个层,层与层之间用接口连接。通信的双方具有相同的层次,层次实现的功能由协议数据单元(PDU)来描述。不同系统中的同一层构成对等层,对等层之间通过对等层协议进行通信,理解批次定义好的规则和约定。每一层表示为物理硬件(即线缆和电流)与所传输信息之间的不同抽象层次。在理论上,每一层只与紧挨其上和其下的层对话。将网络分层,这样就可以修改甚至替换某一层的软件,只要层与层之间的接口保持不变,就不会影响到其他层。

下面看图:
Java网络编程之基础知识点总结_第1张图片

从上面这张图我们对计算机之间的通信特点进行整理一下:

(1)计算机之间的网络通信被分解为多个层,层与层之间用接口连接。

(2)通信的双方具有相同的层次,层次实现的功能由协议数据单元(PDU)来描述。

(3)不同系统中的同一层构成对等层,对等层之间通过对等层协议进行通信。

(4)在通信的时候数据必须由一层依次传递到下一层,不能跨级传输。

为了促进计算机网络的发展,国际标准化组织ISO在现有网络的基础上,提出了不基于具体机型、操作系统或公司的网络体系结构,称为开放系统互连参考模型,即OSI/RM。但是ISO制定的OSI参考模型过于庞大、复杂招致了许多批评。因此美国国防部提出了TCP/IP协议栈参考模型,简化了OSI参考模型,获得了广泛的应用。

2.1、OSI参考模型

这里首先介绍OSI参考模型,是国际标准化组织ISO提出的,把网络通信的工作分为7层,分别是物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。也是我们在大学计算机网络课程中认识的那样。下面一张图来看看。

Java网络编程之基础知识点总结_第2张图片

(1)物理层处于OSI的最底层,是开放系统的基础,它的功能主要是为计算机提供传送数据的通路以及传输数据。

(2)数据链路层的主要任务是实现计算机网络中相邻节点之间的可靠传输,把原始的、有差错的物理传输数据加上数据链路协议以后,构成逻辑上可靠的数据链路。

(3)网络层主要完成的功能主要包括路由选择、网络寻址、流量控制、拥塞控制、网络互连等。实现两个计算机节点之间的数据传输。

(4)传输层涉及源端节点到目的端节点之间可靠的信息传输。也就是说是采用TCP/IP还是采用UDP协议。他需要三次握手四次挥手。

(5)会话层的主要功能是负责应用程序之间建立、维持和中断会话,提供单工、半双工和全双工3种不同的通信方式,使系统和服务之间有序地进行通信。

(6)表示层关心所传输数据信息的格式定义,其主要功能是把应用层提供的信息变换为能够共同理解的形式,提供字符代码、数据格式、控制信息格式、加密等的统一表示。

(7)应用层是直接为应用进程提供服务的。其作用是多个系统应用进程相互通信的同时,完成一系列业务处理所需的服务。

2 .2、TCP/IP参考模型

TCP/IP参考模型采用4层的层级结构,每一层都呼叫它的下一层所提供的协议来完成自己的需求,这4个层次分别是:网络接口层、网络层(IP层)、传输层(TCP层)、应用层。

Java网络编程之基础知识点总结_第3张图片

(1)网络接口层:对应着OSI参考模型的物理层和数据链路层

(2)网络层(IP层):是整个TCP/IP协议栈的核心。它的功能是把分组数据发往目标网络或主机。可以完成将不同类型的网络(异构网)互连的任务。除此之外,还需要完成拥塞控制的功能。

(3)传输层(TCP层):传输层负责在应用进程之间建立端到端的连接和可靠通信。TCP层涉及两个协议,TCP和UDP。TCP协议提供面向连接的服务,提供按字节流的有序、可靠传输,可以实现连接管理、差错控制、流量控制、拥塞控制等。也就是可靠传输。UDP协议提供无连接的服务,用于不需要或无法实现面向连接的网络应用中。也就是不可靠传输。

(4)应用层:为各种网络应用提供服务。

三、网络协议

如同人与人之间相互交流是需要遵循一定的规则(如语言)一样,计算机之间能够进行相互通信是因为它们都共同遵守一定的规则,即网络协议。

OSI参考模型和TCP/IP模型在不同的层次中有许多不同的网络协议,如图所示:

Java网络编程之基础知识点总结_第4张图片

网络协议之间的关系图如下:

Java网络编程之基础知识点总结_第5张图片

下面我们抽出来一些重要的协议看一下。其他的协议用到的时候再说。

(1)IP协议

IP协议的作用在于把各种数据包准备无误的传递给对方,其中两个重要的条件是IP地址和MAC地址。由于IP地址是稀有资源,不可能每个人都拥有一个IP地址,所以我们通常的IP地址是路由器给我们生成的IP地址,路由器里面会记录我们的MAC地址。而MAC地址是全球唯一的。举例,IP地址就如同是我们居住小区的地址,而MAC地址就是我们住的那栋楼那个房间那个人。IP地址采用的IPv4格式,目前正在向IPv6过渡。

(2)TCP协议

TCP层是位于IP层之上,应用层之下的中间层。有可靠的、像管道一样的TCP连接,还有不可靠的包交换的UDP链接。那TCP协议之间的通信数据格式是什么样子的呢?也就是什么样的数据在传输,下面一张图来看一下:

TCP报文段包括协议首部和数据两部分,协议首部的固定部分是20个字节,首部的固定部分后面是选项部分。

Java网络编程之基础知识点总结_第6张图片

下面是报文段首部各个字段的含义:

  1. 源端口号以及目的端口号:各占2个字节,端口是传输层和应用层的服务接口,用于寻找发送端和接收端的进程,一般来讲,通过端口号和IP地址,可以唯一确定一个TCP连接,在网络编程中,通常被称为一个socket接口。
  2. 序号:Seq序号,占4个字节、32位。用来标识从TCP发送端向TCP接收端发送的数据字节流。发起方发送数据时对此进行标记。
  3. 确认序号:Ack序号,占4个字节、32位。包含发送确认的一端所期望收到的下一个序号。只有ACK标记位为1时,确认序号字段才有效,因此,确认序号应该是上次已经成功收到数据字节序号加1,即Ack=Seq + 1。
  4. 数据偏移:占4个字节,用于指出TCP首部长度,若不存在选项,则这个值为20字节,数据偏移的最大值为60字节。
  5. 保留字段占6位,暂时可忽略,值全为0。
  6. 标志位,6个
    • URG(紧急):为1时表明紧急指针字段有效
    • ACK(确认):为1时表明确认号字段有效
    • PSH(推送):为1时接收方应尽快将这个报文段交给应用层
    • RST(复位):为1时表明TCP连接出现故障必须重建连接
    • SYN(同步):在连接建立时用来同步序号
    • FIN(终止):为1时表明发送端数据发送完毕要求释放连接
  7. **接收窗口:**占2个字节,用于流量控制和拥塞控制,表示当前接收缓冲区的大小。在计算机网络中,通常是用接收方的接收能力的大小来控制发送方的数据发送量。TCP连接的一端根据缓冲区大小确定自己的接收窗口值,告诉对方,使对方可以确定发送数据的字节数。
  8. **校验和:**占2个字节,范围包括首部和数据两部分。
  9. 选项是可选的,默认情况是不选。

TCP是面向连接的协议,因此每个TCP连接都有3个阶段:连接建立、数据传送和连接释放。连接建立经历三个步骤,通常称为“三次握手”。

TCP三次握手过程如下:

Java网络编程之基础知识点总结_第7张图片

  1. 第一次握手(客户端发送请求)

  2. 第二次握手(服务端回传确认)

  3. 第三次握手(客户端回传确认)

注意:握手过程中传送的包里不包含数据,三次握手完毕后,客户端与服务器才正式开始传送数据。

TCP四次挥手过程如下:
Java网络编程之基础知识点总结_第8张图片

由于TCP连接是全双工的,因此每个方向都必须单独进行关闭。这原则是当一方完成它的数据发送任务后就能发送一个FIN来终止这个方向的连接。收到一个FIN只意味着这一方向上没有数据流动,一个TCP连接在收到一个FIN后仍能发送数据。首先进行关闭的一方将执行主动关闭,而另一方执行被动关闭。

注意:为什么连接的时候是三次握手,关闭的时候却是四次挥手?

因为当服务端收到客户端的SYN连接请求报文后,可以直接发送SYN+ACK报文。其中ACK报文是用来应答的,SYN报文是用来同步的。但是关闭连接时,当服务端收到FIN报文时,很可能并不会立即关闭socket,所以只能先回复一个ACK报文,告诉客户端,“你发的FIN报文,我收到了”。只有等到服务端所有的报文都发送完了,我才能发送FIN报文,因此不能一起发送,故需要四步挥手。

(3)UDP协议

UDP,用户数据报协议,它是TCP/IP协议簇中无连接的运输层协议。

  1. UDP是一个非连接的协议,传输数据之前源端和终端不建立连接,当它想传送时就简单地去抓取来自应用程序的数据,并尽可能快地把它扔到网络上。在发送端,UDP传送数据的速度仅仅是受应用程序生成数据的速度、计算机的能力和传输带宽的限制;在接收端,UDP把每个消息段放在队列中,应用程序每次从队列中读一个消息段。

  2. 由于传输数据不建立连接,因此也就不需要维护连接状态,包括收发状态等,因此一台服务器可同时向多个客户端传输相同的消息。

  3. UDP信息包的标题很短,只有8个字节,相对于TCP的20个字节信息包的额外开销很小。

  4. 吞吐量不受拥挤控制算法的调节,只受应用软件生成数据的速率、传输带宽、源端和终端主机性能的限制。

  5. UDP使用尽量最大努力交付,即不保证可靠交付,因此主机不需要维持复杂的链接状态表。

  6. UDP是面向报文的。发送方的UDP对应用程序交下来的报文,在添加首部受就向下交付给IP层。既不拆分,也不合并,而是保留这些报文的边界,因此,应用程序需要选择合适的报文大小。

    UDP协议格式

Java网络编程之基础知识点总结_第9张图片
UDP协议由两部分组成:首部和数据。其中,首部仅有8个字节,包括源端口和目的端口、长度(UDP用于数据报的长度)、校验和。

(4) TCP与UDP的区别

  1. TCP基于连接,UDP是无连接的;
  2. 对系统资源的要求,TCP较多,UDP较少;
  3. UDP程序结构较简单;
  4. TCP是流模式,而UDP是数据报模式;
  5. TCP保证数据正确性,而UDP可能丢包;TCP保证数据顺序,而UDP不保证;

(5)HTTP协议

HTTP,超文本传输协议,它是互联网上应用最为广泛的一种网络协议。HTTP是一种应用层协议,它是基于TCP协议之上的请求/响应式的协议。HTTP协议是Web浏览器和Web服务器之间通信的标准协议。HTTP指定客户端与服务器如何建立连接、客户端如何从服务器请求数据,服务器如何响应请求,以及最后如何关闭连接。HTTP连接使用TCP/IP来传输数据。

对于从客户端到服务器的每一个请求,都有4个步骤:

  1. 默认情况下,客户端在端口80打开与服务器的一个TCP连接,URL中还可以指定其他端口。
  2. 客户端向服务器发送消息,请求指定路径上的资源。这个资源包括一个首部,可选地(取决于请求的性质)还可以有一个空行,后面是这个请求的数据。
  3. 服务器向客户端发送响应。响应以响应码开头,后面是包含数据的首部、一个空行以及所请求的文档或错误消息。
  4. 服务器关闭连接。

现在使用的HTTP协议是HTTP/1.1版本,1997年之前采用的是HTTP1.0版本。HTTP连接在1.0版本中采用非持续连接工作方式,1.1版本采用的是持续连接工作方式,持续连接是指服务器在发送响应后仍然在一段时间内保持这条由TCP运输层协议建立起来的连接,使客户端和服务器可以继续在这条连接上传输HTTP报文。

是否采用持续连接工作方式,1.0中默认是关闭的,需要在HTTP头加入“Connection:Keep-Alive”,才能启用Keep-Alive。HTTP1.1中默认启用Keep-Alive,如果加入“Connection:close”,才关闭。目前大部分浏览器都是用HTTP1.1协议,也就是说默认都会发起Keep-Alive的连接请求了,所以是否能完成一个完整的Keep-Alive连接就看服务器设置情况。

HTTP报文

HTTP协议是基于TCP协议之上的请求/响应式协议,下面主要介绍HTTP报文的格式,HTTP报文主要有请求报文和响应报文两种。

首先看HTTP请求报文的格式

Java网络编程之基础知识点总结_第10张图片

下图是谷歌浏览器内访问服务器查看的HTTP请求例子:

Java网络编程之基础知识点总结_第11张图片
HTTP响应报文格式

Java网络编程之基础知识点总结_第12张图片

HTTP请求方法和响应状态码

在上面的HTTP请求报文例子中,我们可以看到请求方法是GET,这表示请求读取由URL所标志的信息,除了GET,还有其他几种常用的方法。

Java网络编程之基础知识点总结_第13张图片

在HTTP响应报文的例子中,我们可以看到状态码是200,表示响应成功。下表是其他状态码,总共5大类,33种。
Java网络编程之基础知识点总结_第14张图片
HTTP和HTTPS的区别

HTTP协议被用于在Web浏览器和网站服务器之间传递信息。HTTP协议以明文方式发送内容,不提供任何方式的数据加密,如果攻击者截取了Web浏览器和网站服务器之间的传输报文,就可以直接读懂其中的信息,因此HTTP协议不适合传输一些敏感信息,比如信用开号、密码等。

为了解决HTTP协议的这一缺陷,需要使用另一种协议:安全套接字层超文本传输协议HTTPS。为了数据传输的安全,HTTPS在HTTP的基础上加入了SSL协议,SSL依靠证书来验证服务器的身份,并为浏览器和服务器之间的通信加密。

HTTPS和HTTP的区别主要为以下四点:

https协议需要到ca申请证书,一般免费证书很少,需要缴费。http是超文本传输协议,信息是明文传输,https则是具有安全性的ssl加密传输协议。http和https使用的是完全不同的连接方式,用的端口也不一样,前者是80,后者是443。http的连接很简单,是无状态的;https协议是有ssl+http协议构建的可进行加密传输、身份认证的网络协议,比http协议安全。

三、Java Socket网络编程

3.1 Socket概述

Java的网络编程主要涉及到的内容是Socket编程。Socket,套接字,就是两台主机之间逻辑连接的端点。TCP/IP协议是传输层协议,主要解决数据如何在网络中传输,而HTTP是应用层协议,主要解决如何包装数据。Socket是通信的基石,是支持TCP/IP协议的网络通信的基本操作单元。它是网络通信过程中端点的抽象表示,包含进行网络通信必须的五种信息:连接使用的协议、本地主机的IP地址、本地进程的协议端口、远程主机的IP地址、远程进程的协议端口。

应用层通过传输层进行数据通信时,TCP会遇到同时为多个应用程序进程提供并发服务的问题。多个TCP连接或多个应用程序进程可能需要通过同一个TCP协议端口传输数据。为了区别不同的应用程序进程和连接,许多计算机操作系统为应用程序与TCP/IP协议交互提供了套接字(Socket)接口。应用层可以和传输层通过Socket接口,区分来自不同应用程序进程或网络连接的通信,实现数据传输的并发服务。

Socket,实际上是对TCP/IP协议的封装,Socket本身并不是协议,而是一个调用接口(API),通过Socket,我们才能使用TCP/IP协议。实际上,Socket跟TCP/IP协议没有必然的关系,Socket编程接口在设计的时候,就希望也能适应其他的网络协议。所以说,Socket的出现,只是使得程序员更方便地使用TCP/IP协议栈而已,是对TCP/IP协议的抽象,从而形成了我们知道的一些最基本的函数接口,比如create、listen、accept、send、read和write等等。网络有一段关于socket和TCP/IP协议关系的说法比较容易理解:

“TCP/IP只是一个协议栈,就像操作系统的运行机制一样,必须要具体实现,同时还要提供对外的操作接口。这个就像操作系统会提供标准的编程接口,比如win32编程接口一样,TCP/IP也要提供可供程序员做网络开发所用的接口,这就是Socket编程接口。”

实际上,传输层的TCP是基于网络层的IP协议的,而应用层的HTTP协议又是基于传输层的TCP协议的,而Socket本身不算是协议,就像上面所说,它只是提供了一个针对TCP或者UDP编程的接口。socket是对端口通信开发的工具,它要更底层一些。

3.2 Socket整体流程

Socket编程主要涉及到客户端和服务端两个方面,首先是在服务器端创建一个服务器套接字(ServerSocket),并把它附加到一个端口上,服务器从这个端口监听连接。端口号的范围是0到65536,但是0到1024是为特权服务保留的端口号,我们可以选择任意一个当前没有被其他进程使用的端口。

客户端请求与服务器进行连接的时候,根据服务器的域名或者IP地址,加上端口号,打开一个套接字。当服务器接受连接后,服务器和客户端之间的通信就像输入输出流一样进行操作。

img

3.3 Socket编程示例

实例一

通过 ObjectOutputStream过滤流进通信

服务端:

package ser;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

public class MyServer {
   public static void main(String[] args) throws IOException, ClassNotFoundException {
	   //建立服务端对象,监听6767端口
	   ServerSocket serv = new ServerSocket(6867);
	   System.out.print("服务端创建成功,等待连接");
	   //等待用户接入
	   Socket sc =serv.accept();
	   System.out.println("客户端连接成功,端口为:"+sc.getInetAddress());
	   

```java
    InputStream  in = sc.getInputStream();
    OutputStream out  =sc.getOutputStream();
```

//	    byte[] bs = new byte[1024];
//	    int len = in.read(bs);
//	    String str = new String(bs,0,len);
//	    System.out.println("来自客户端的消息:"+str);
//	    out.write("你好,我是服务器".getBytes());
	    /**

//创建过滤流

    ObjectOutputStream oos = new ObjectOutputStream(out);
    ObjectInputStream ois = new ObjectInputStream(in);
    String s1=(String) ois.readObject();
    System.out.print("接到来自客户端的数据:"+s1);
    sc.close();
    serv.close();

   }
}

客户端

package cli;

import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.net.UnknownHostException;

public class MyClient {
public static void main(String[] args) throws UnknownHostException, IOException {
	// TODO Auto-generated method stub
  /**
   * 编写客户端,不需要关心自己的地址,但一定要知道服务器
   * 地址和端口
   */
	Socket ss = new Socket("localhost", 6867);
	System.out.println("连接服务器成功");
	
	InputStream  in = ss.getInputStream();
    OutputStream out  =ss.getOutputStream();
 //	    out.write("你好,我是客户端".getBytes());
//	    byte[] bs = new byte[1024];
//	    int len = in.read(bs);
//	    String str = new String(bs,0,len);
//	    System.out.println("来自服务器的消息:"+str);
	    /**
	     * 创建过滤流
	     */
	    ObjectInputStream ois = new ObjectInputStream(in);
	    ObjectOutputStream oos = new ObjectOutputStream(out);
	    oos.writeObject("我是客户端,通过过滤流与你交流");
	    ss.close();
	}

}

实例二: 通过socket进行文件传输

服务端:

package ser;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;

public class FileServer {
public static void main(String[] args) throws IOException {
	   //建立服务端对象,监听6767端口
	   ServerSocket serv = new ServerSocket(8887);
	   System.out.print("服务端创建成功,等待连接");
	   //等待用户接入
	   Socket sc =serv.accept();
	   System.out.println("客户端连接成功,端口为:"+sc.getInetAddress());
	   
	    InputStream  in = sc.getInputStream();
	    OutputStream out  =sc.getOutputStream();
	    
	    ObjectOutputStream oos = new ObjectOutputStream(out);
	    ObjectInputStream ois = new ObjectInputStream(in);
	    //具体文件路径自己选择
	    File src =new File("C:\\Users\\ASUS\\Desktop\\1.jpg");
	    
	    InputStream fileStream = new FileInputStream(src);
	    
	    byte[] bs = new byte[1024];
	    
	    oos.writeObject(src.getName());
	    
	    for(int len =0;(len=fileStream.read(bs))!=-1;) {
	    	out.write(bs,0,len);
	    }
	    
	    System.out.print("文件发送完成!!!");
	    
	    fileSt客户端eam.close();
	    
	    sc.close();
 }
}

客户端

package cli;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.net.UnknownHostException;

public class FileClient {

	public static void main(String[] args) throws UnknownHostException, IOException, ClassNotFoundException {
		// TODO Auto-generated method stub
		Socket ss = new Socket("192.168.1.4", 8887);
		System.out.println("连接服务器成功");
		
		
		InputStream  in = ss.getInputStream();
	    OutputStream out  =ss.getOutputStream();
		/**
	     * 创建过滤流
	     */
	    ObjectInputStream ois = new ObjectInputStream(in);
	    ObjectOutputStream oos = new ObjectOutputStream(out);
	    
	    String fileName = (String) ois.readObject();
	    
	    File xin = new File("C:\\Users\\Public",fileName);
	    OutputStream fileOut = new FileOutputStream(xin);
	    System.out.println("开始接收文件:"+xin.getAbsolutePath());
	    byte[] bs = new byte[1024];
	    for(int len=0;(len=in.read(bs))!=-1;) {
	    	fileOut.write(bs,0,len);
	    }
	    
	    System.out.println("文件接收完成");
	    fileOut.close();
	    ss.close();
	}

}

实例三 计算正方形面积

服务端:

package ser;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

public class SocketServer {
    public static void main(String[] args) throws IOException {

        // 端口号
        int port = 7000;
        // 在端口上创建一个服务器套接字
        ServerSocket serverSocket = new ServerSocket(port);
        // 监听来自客户端的连接
        Socket socket = serverSocket.accept();

        DataInputStream dis = new DataInputStream(
                new BufferedInputStream(socket.getInputStream()));

        DataOutputStream dos = new DataOutputStream(
                new BufferedOutputStream(socket.getOutputStream()));

        do {
            double length = dis.readDouble();
            System.out.println("服务器端收到的边长数据为:" + length);
            double result = length * length;
            dos.writeDouble(result);
            dos.flush();
        } while (dis.readInt() != 0);

        socket.close();
        serverSocket.close();
    }
}


客户端

package cli;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Scanner;


public class SocketClient {
    public static void main(String[] args) throws UnknownHostException, IOException {

        int port = 7000;
        String host = "localhost";

        // 创建一个套接字并将其连接到指定端口号
        Socket socket = new Socket(host, port);

        DataInputStream dis = new DataInputStream(
                new BufferedInputStream(socket.getInputStream()));

        DataOutputStream dos = new DataOutputStream(
                new BufferedOutputStream(socket.getOutputStream()));

        Scanner sc = new Scanner(System.in);

        boolean flag = false;

        while (!flag) {

            System.out.println("请输入正方形的边长:");
            double length = sc.nextDouble();

            dos.writeDouble(length);
            dos.flush();

            double area = dis.readDouble();

            System.out.println("服务器返回的计算面积为:" + area);

            while (true) {

                System.out.println("继续计算?(Y/N)");

                String str = sc.next();

                if (str.equalsIgnoreCase("N")) {
                    dos.writeInt(0);
                    dos.flush();
                    flag = true;
                    break;
                } else if (str.equalsIgnoreCase("Y")) {
                    dos.writeInt(1);
                    dos.flush();
                    break;
                }
            }
        }
        socket.close();
    }
}

可以看到上面的服务器端程序和客户端程序是一对一的关系,为了能让一个服务器端程序能同时为多个客户提供服务,可以使用多线程机制,每个客户端的请求都由一个独立的线程进行处理。

public class SocketServerM {
    public static void main(String[] args) throws IOException {

        int port = 7000;
        int clientNo = 1;

        ServerSocket serverSocket = new ServerSocket(port);

        // 创建线程池
        ExecutorService exec = Executors.newCachedThreadPool();

        try {

            while (true) {
                Socket socket = serverSocket.accept();
                exec.execute(new SingleServer(socket, clientNo));
                clientNo++;
            }

        } finally {
            serverSocket.close();
        }

    }
}

class SingleServer implements Runnable {

    private Socket socket;
    private int clientNo;

    public SingleServer(Socket socket, int clientNo) {
        this.socket = socket;
        this.clientNo = clientNo;
    }

    @Override
    public void run() {
\\这里写逻辑
}

下面我还是举一个基于UDP广播的实例供大家参考,可以发现UDP与TCP的区别

服务端:

package receive;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;

public class DataReceive {

	public static void main(String[] args) throws IOException {
		// TODO Auto-generated method stub
		//在本机6789建立监听
		 DatagramSocket socket = new DatagramSocket(8888);
		 
		 byte[] buffer = new byte[128];
		 
		 DatagramPacket pack = new DatagramPacket(buffer,buffer.length);
		 
		 socket.receive(pack);
		 
		 
		 String str = new String(buffer,0,pack.getLength());
		 
		 System.out.println("接受到数据:"+str);
		 

	}

}

客户端:

package send;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
import java.net.SocketException;

public class DataSender {

	public static void main(String[] args) throws IOException {
		// TODO Auto-generated method stub
		//用于发送广播的socket
     DatagramSocket socket = new DatagramSocket();
     
     String str ="Hello World!";
     byte[] bs =str.getBytes();
     
     //目标电脑
     InetSocketAddress target = new InetSocketAddress("localhost",8888);
     
     DatagramPacket packet = new DatagramPacket(bs,0,bs.length,target);
     socket.send(packet);
     System.out.println("发送成功");
	}

}

参考资料:
https://www.jianshu.com/p/066d99da7cbd

https://www.cnblogs.com/swordfall/p/10781281.html

https://baijiahao.baidu.com/s?id=1639651267164686059&wfr=spider&for=pc

本博客利用了部分网络资源,版权归原作者及网站所有,虽力求保存原有的版权信息并尽可能注明来源,但由于很多资料经过多次转摘,已经无法确定真实来源;部分因为操作上的原因 可能已将原有信息丢失,所以敬请原作者原谅。如果您对本站所载文章及作品版权的归属存有异议,请立即通知我,我将在第一时间予以删除,同时向你表示歉意!

你可能感兴趣的:(java网络)