概述
地球村:在西安,你有一个美国朋友
计算机网络:
计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路和通信设备连接起来,在网络操作系统,网络管理软件及网络通信协议的管理和协调下,实现资源共享和信息传递的计算机系统。
网络编程的目的:
无线电台:传播交流信息,数据交换、通信
想要这个效果需要什么:
- 如何准确定位网络上的一台主机 192.168.16.124:端口,定位到这个计算机上的某个资源
- 找到了这个主机,如何传输数据呢?
- Javaweb:网页编程 B/S
- 网络编程:TCP/IP C/S
网络通信的要素
人工智能:智能汽车:工厂,人少!?
如何实现网络的通信?
通信双方的地址:
- ip
- 端口号
- 192.168.16.124:5900
规则:网络通信的协议
TCP
小结:
1.网络编程中有两个主要的问题
- 如何准确的定位到网络上的一台或多台主机
- 找到主机之后如何通信
2.网络编程中的要素
- IP和端口号 IP类
- 网络通信协议 udp,tcp
3.万物皆对象
IP
IP地址:InetAddress
唯一定位一台网络上的计算机
127.0.0.1:本机localhost
ip地址的分类
- IP地址分类
- ipv4/ipv6
- IPV4 127.0.0.1,四个字节组成,0-255,42亿;30亿在北美,亚洲四亿
- IPV6 fe80::bae7:8b5f:8e2c:3551%19,128位,8个无符号整数
-
2001:0bb2:aaaa:0015:0000:0000:1aaa:1312
- ipv4/ipv6
- 公网-私网
-
公网:给互联网使用
-
私网:局域网
-
- 域名:记忆 IP 问题
端口
端口表示计算机上的一个程序的进程,不同的进程有不同的端口号,用来区分软件
TCP / UDP:都是0~65535
单个协议下,端口号不能冲突
端口分类:
- 公有端口:0~1023(不建议占用)
- HTTP:80
- HTTPS:443
- FTP:21
- SSH:22
- Telent:23
- 程序注册端口:1024~49151(分配给用户使用)
- Tomacat:8080
- MySQL:3306
- Oracle:1521
- 动态(私有)端口:49152~65535(同样不建议使用)
netstat -ano #查看全部端口
netstat -ano|findstr "port" #查询指定端口(port)
tasklist|findstr "port" #查询指定端口(port)的进程
Ctrl + Shift + ESC #打开任务管理器
通信协议
速率、传输码率、代码结构、传输控制...
TCP/IP协议簇:
- TCP:用户传输协议
- UDP:用户数据报协议
TCP-UDP协议对比
TCP(类似打电话) | UDP(类似发短信) | |
稳定性 | 连接稳定 (三次握手、四次挥手) | 不稳定 |
客户端、服务端 | 区分 | 没有明确界限 |
传输效率 | 传输完成后释放连接,效率低 | 不管是否准备好都发送 |
三次握手四次挥手:
连接:三次握手
断开:四次挥手
TCP实现聊天
客户端
- 获取服务器IP、端口
- 建立Socket连接
- 发送信息IO流
package demoNetContact;
import java.io.IOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
/**
* @Description:
* @Author: Nanoic
* @Date: 2024-02-02
* @FileName: tcpClient
**/
public class tcpClient {
public static void main(String[] args) {
//要知道服务器的地址
Socket socket = null;
OutputStream os = null;
try {
//1. 获取服务器地址
InetAddress serverIP = InetAddress.getByName("127.0.0.1");
//2. 端口号
int port = 14514;
//3. 创建Scoket连接
socket = new Socket(serverIP, port);
//4. 发消息 IO流
os = socket.getOutputStream();
os.write("Hello".getBytes());
} catch (IOException e) {
throw new RuntimeException(e);
} finally {
try {
socket.close();
os.close();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
}
}
服务器
- 创建端口号
- 等待客户端Socket连接
- 读取客户端信息
- 创建管道流
- 输出信息
package demoNetContact;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
/**
* @Description:
* @Author: Nanoic
* @Date: 2024-02-02
* @FileName: tcpServer
**/
public class tcpServer {
public static void main(String[] args) {
ByteArrayOutputStream baos = null;
InputStream is = null;
Socket socket = null;
ServerSocket serverSocket = null;
try {
//1. 创建服务器地址
serverSocket = new ServerSocket(14514);
while(true){
//2. 等待客户端连接
socket = serverSocket.accept();
//3. 接收消息
is = socket.getInputStream();
//4. 创建管道流接收
baos = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len;
while ((len = is.read(buffer)) != -1) {
baos.write(buffer, 0, len);
}
System.out.println(baos.toString());
}
} catch (IOException e) {
e.printStackTrace();
} finally {
//关闭资源
try {
baos.close();
is.close();
socket.close();
serverSocket.close();
} catch (IOException e) {
throw new RuntimeException(e);
}
}
}
}
TCP文件上传
服务器:
package demoNetContact;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class TcpClientDemo2 {
public static void main(String[] args) throws Exception{
//1.创建服务
ServerSocket serverSocket=new ServerSocket(9000);
//2.监听客户端连接
Socket socket= serverSocket.accept();//阻塞式监听,会一直等待客户端连接
//3.获取输入流
InputStream is= socket.getInputStream();
//4.文件输出
FileOutputStream fos=new FileOutputStream(new File("receive.jpg"));
byte[] buffer=new byte[1024];
int len;
while((len=is.read(buffer))!=-1){
fos.write(buffer,0,len);
}
//通知客户端接收完毕
OutputStream os=socket.getOutputStream();//发送
os.write("我接收完毕".getBytes());
//5.关闭资源
fos.close();
is.close();
socket.close();
serverSocket.close();
}
}
客户端:
package demoNetContact;
import java.io.*;
import java.net.InetAddress;
import java.net.Socket;
public class TcpClientDemo {
public static void main(String[] args)throws Exception {
//1.创建一个Socket连接
Socket socket = new Socket(InetAddress.getByName("127.0.0.1"),9000);
//2.创建一个输出流
OutputStream os=socket.getOutputStream();
//3.读取流
FileInputStream fis = new FileInputStream(new File("1.jpg"));
//4.写出文件
byte[] buffer=new byte[1024];
int len;
while ((len = fis.read(buffer) )!= -1){
os.write(buffer,0,len);
}
//通知服务器,我已经结束了
socket.shutdownOutput();//我已经传输完了
//确认服务器接收完毕,才能断开连接
InputStream inputStream = socket.getInputStream();//接收
//String byte[]
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buffer2 = new byte[2014];
int len2;
while((len2=inputStream.read(buffer2))!=-1){
baos.write(buffer2,0,len2);
}
System.out.println(baos.toString());
//5.关闭资源
fis.close();
os.close();
socket.close();
baos.close();
inputStream.close();
}
}
TOMCAT
- 服务端:
- 自定义 S
- Tomcat服务器 S : Java后台开发
- 客户端:
- 自定义 C
- 浏览器 B
UDP 消息发送
发送端和接收端不需要连接
/* 或许是客户端 */
package demoNetContact;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
//不需要服务器
public class UdpClientDemo01 {
public static void main(String[] args) throws Exception {
//1.建立一个socket
DatagramSocket socket= new DatagramSocket();
//2.建个包
String msg="你好,服务器";
//发送给谁
InetAddress localhost =InetAddress.getByName("localhost");
int port=9090;
//数据,数据起始:从0开始,数据末,要发送给谁,
DatagramPacket packet= new DatagramPacket(msg.getBytes(),0,msg.getBytes().length,localhost,port);
//3.发送包
socket.send(packet);
//4.关闭流
socket.close();
}
}
/* 或许是服务端 */
package demoNetContact;
import com.sun.org.apache.xpath.internal.operations.String;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
public class UdpServerDemo02 {
public static void main(String[] args) throws Exception{
//开发端口
DatagramSocket socket=new DatagramSocket(9090);
//接收数据包
byte[] buffer= new byte[1024];
DatagramPacket packet= new DatagramPacket(buffer,0,buffer.length);//接收
socket.receive(packet);//阻塞接收
//输出
System.out.println(packet.getAddress().getHostAddress());
System.out.println(new String(packet.getData(),0,packet.getLength()));
//关闭连接
socket.close();
}
}
UDP消息发送
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
public class UdpMessageSender {
public static void main(String[] args) {
try {
DatagramSocket socket = new DatagramSocket();
String message = "Hello, UDP!";
byte[] data = message.getBytes();
InetAddress address = InetAddress.getByName("127.0.0.1");
int port = 12345;
DatagramPacket packet = new DatagramPacket(data, data.length, address, port);
socket.send(packet);
System.out.println("Message sent successfully.");
socket.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
在上面的示例中,我们创建了一个DatagramSocket实例,然后创建了一个包含要发送数据的DatagramPacket,并通过socket.send()方法发送数据。在实际应用中,我们可以将这段代码封装成一个可重用的UDP消息发送工具类。
UDP聊天实现
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
public class UdpChatUser {
public static void main(String[] args) {
try {
DatagramSocket socket = new DatagramSocket();
Thread sender = new Thread(() -> {
try {
while (true) {
// 从控制台读取用户输入
byte[] sendData = new byte[1024];
System.in.read(sendData);
InetAddress address = InetAddress.getByName("127.0.0.1");
int port = 12345;
DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length, address, port);
socket.send(sendPacket);
}
} catch (Exception e) {
e.printStackTrace();
}
});
Thread receiver = new Thread(() -> {
try {
while (true) {
// 接收消息
byte[] receiveData = new byte[1024];
DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);
socket.receive(receivePacket);
String message = new String(receivePacket.getData(), 0, receivePacket.getLength());
System.out.println("Received: " + message);
}
} catch (Exception e) {
e.printStackTrace();
}
});
sender.start();
receiver.start();
} catch (Exception e) {
e.printStackTrace();
}
}
}
在上面的示例中,我们创建了一个DatagramSocket实例,然后使用两个线程分别负责发送和接收消息。发送消息的线程从控制台读取用户输入,并将消息发送到指定的地址和端口;接收消息的线程不断接收来自指定地址和端口的消息,并将消息内容打印到控制台。