Q:什么是网络?
A:网络本质就是两台设备之间进行数据交换,在计算机网络中设备主要指计算机。
Q:什么是IP地址和域名?
A:由于互联网是由许多能够进行数据交换的计算机互相连接产生的集合,所以如果要向某台计算机发送数据,则必须有个地址以能够找到这台计算机,这个地址就是IP。
由于IP地址是由4个0-255的数字组成的,不方便记忆,所以有了域名(Domain Name)。实际上域名就是IP地址的名字,如果IP地址是身份证号码,那么域名就是姓名。
实际上网络中是用IP地址来传输,所以在传输之前,需要把域名解析成IP地址,DNS就是专门用来解析域名的服务器。
Q:什么是端口?
A:为了能够在一台设备上运行多个程序,人们设计了端口(Port),一个设备有65536个端口,一个网络程序可对应一个端口,传输进来的信息流经过端口的分发,不同程序需要的信息传进对应的程序。
Q:TCP和UDP的区别?
A:TCP和UDP都是网络间的传输协议,他们的特点是
udp: a、是面向无连接, 将数据及源的封装成数据包中,不需要建立建立连接
b、每个数据报的大小在限制64k内
c、因无连接,是不可靠协议
d、不需要建立连接,速度快
tcp:a、建议连接,形成传输数据的通道.
b、在连接中进行大数据量传输,以字节流方式
c、通过三次握手完成连接,是可靠协议
d、必须建立连接,效率会稍低
1.UDP传输协议
UDP协议传输,是将数据直接按指定地址发送,但是并不验证对方是否接收,所以UDP传输是不可靠传输
1.1UDP服务端
UDPServer.java
public class UDPServer {
public static void main(String[] args) throws Exception {
System.out.println("UDP服务端启动。。。");
// 在8080端口监听接收
DatagramSocket ds = new DatagramSocket(8080);
// 定义1024长度的byte数组
byte[] buf = new byte[1024];
// DatagramPacket(byte[] buf, int length)
// 构造 DatagramPacket,用来接收长度为 length 的数据包。
DatagramPacket dp = new DatagramPacket(buf, buf.length);
// 监听8080端口等待接收包,此为阻塞方法
ds.receive(dp);
// 输出哪台设备发过来的和端口号
System.out.println("来源:" + dp.getAddress().getHostAddress() + ",端口:" + dp.getPort());
// 将接收到的包new读成String
String str = new String(dp.getData(), 0, dp.getLength());
// 输出接收到的包
System.out.println("接收到的数据:" + str);
// 记得关闭连接
ds.close();
}
}
1.2UDP客户端
UDPClient.java
public class UDPClient {
// 直接执行Client端也不会报错,因为UDP协议不会关心对方是否收到或存在,只要发出去就行了
public static void main(String[] args) throws Exception {
System.out.println("UDP客户端发送消息");
// 创建UDP连接
DatagramSocket ds = new DatagramSocket();
// 要发送的字符串转成byte数组
byte[] strByte = "UDP客户端发送的数据!".getBytes();
// DatagramPacket(byte[] buf, int length, InetAddress address, int port)
// 构造数据报包,用来将长度为 length 的包发送到指定主机上的指定端口号。
DatagramPacket dp = new DatagramPacket(strByte, strByte.length, InetAddress.getByName("192.168.1.100"), 8080);
// 执行发送方法
ds.send(dp);
// 记得关闭连接
ds.close();
}
}
2.TCP传输协议
2.1三次握手
在TCP/IP协议中,TCP协议采用三次握手建立一个连接。
第一次握手:建立连接时,客户端发送SYN包(SYN=J)到服务器,并进入SYN_SEND状态,等待服务器确认;
第二次握手:服务器收到SYN包,必须确认客户的SYN(ACK=J+1),同时自己也发送一个SYN包(SYN=K),即SYN+ACK包,此时服务器V状态;
第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ACK=K+1),此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手。
完成三次握手,客户端与服务器开始传送数据。
2.2四次挥手
由于TCP连接是全双工的,因此每个方向都必须单独进行关闭。这个原则是当一方完成它的数据发送任务后就能发送一个FIN来终止这个方向的连接。收到一个FIN只意味着这一方向上没有数据流动,一个TCP连接在收到一个FIN后仍能发送数据。首先进行关闭的一方将执行主动关闭,而另一方执行被动关闭。
第一次挥手:客户端A发送一个FIN,用来关闭客户A到服务器B的数据传送。
第二次挥手:服务器B收到这个FIN,它发回一个ACK,确认序号为收到的序号加1。和SYN一样,一个FIN将占用一个序号。
第三次挥手:服务器B关闭与客户端A的连接,发送一个FIN给客户端A。
第四次挥手:客户端A发回ACK报文确认,并将确认序号设置为收到序号加1。
2.3TCP服务端
TCPServer.java
public class TCPServer {
public static void main(String[] args) throws Exception {
System.out.println("TCP服务端启动。。。");
// ServerSocket(int port)
// 创建绑定到特定端口的服务器套接字。
ServerSocket serverSocket = new ServerSocket(8080);
// 开始监听端口准备连接,此为阻塞方法
Socket socket = serverSocket.accept();
// 拿到IO流
InputStream inputStream = socket.getInputStream();
// 定义1024长度的byte数组
byte[] buf = new byte[1024];
// 将读取到的信息缓冲在数组buf中,并返回数组长度
int len = inputStream.read(buf);
// 将数组buf中的数据读取成字符串
String str = new String(buf, 0, len);
// 输出str
System.out.println("接收到的数据:" + str);
// 记得关闭连接
serverSocket.close();
}
}
2.4TCP客户端
TCPClient.java
public class TCPClient {
// 直接执行Client端会报错,因为无法与对方建立连接
public static void main(String[] args) throws Exception{
System.out.println("TCP客户端发送消息");
// Socket(InetAddress address, int port)
// 创建一个流套接字并将其连接到指定 IP 地址的指定端口号。
Socket socket = new Socket("192.168.1.100", 8080);
// 拿到IO流
OutputStream outputStream = socket.getOutputStream();
// 要发送的字符串转成byte数组
byte[] strByte = "TCP客户端发送的数据".getBytes();
// 使用IO流将byte数组写入发送
outputStream.write(strByte);
// 记得关闭连接
socket.close();
}
}