狂神视频地址:https://www.bilibili.com/video/BV1LJ411z7vY
1. 概述
计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接(有线性、无线)起来,在网络操作系统、网络管理软件及网络通信协议的管理和协调下,实现资源共享和信息传递的计算机系统。
网络编程的目的
- 传播交流信息
- 数据交换、通信
想要达到这个效果,需要什么
- 如何准确的定位网络上的一台主机?
192.168.1.100: 端口,定位到这个计算机上的某个资源。 - 找到了这个主机,如何传输数据呢?
JavaWeb : 网页编程 B/S架构
网络编程: TCP/IP C/S架构
2. 网络通信的要素
如何实现网络的通信?
- 通信双方的地址:IP+端口号(192.168.1.100:8080)
- 规则:网络通信的协议
TCP/IP参考模型
小结:
- 网络编程中两个主要问题
如何准确定位到网络上的一台或多台主机
找到主机之后如何进行通信 - 网络编程中的要素
IP 和 端口号
网络通信协议 - Java 万物皆对象
3. IP
ip地址:InetAddress
- 唯一定位一台网络上计算机
- 127.0.0.1: 本机localhost
- ip地址的分类
- IPV4/IPv6
- IPV4 :127.0.0.1,4个字节组成,0-255 42亿个 30亿都在北美,亚洲4亿。2011年就用尽
- IPV6 :128位。8个无符号整数!
- 公网(互联网)-私网(局域网)
- 192.168.xx.xx:专门给组织内部使用的
- IPV4/IPv6
- 域名:记忆ip问题
//测试ip
public class TestInetAddress {
public static void main(String[] args) {
try {
//查询本机地址
InetAddress address1 = InetAddress.getByName("127.0.0.1");
System.out.println(address1); ///127.0.0.1
InetAddress address3 = InetAddress.getByName("localhost");
System.out.println(address3); //localhost/127.0.0.1
InetAddress address4 = InetAddress.getLocalHost();
System.out.println(address4); //Sunny/192.168.100.1
//查询网址ip地址
InetAddress address2 = InetAddress.getByName("www.baidu.com");
System.out.println(address2); //www.baidu.com/36.152.44.95
} catch (UnknownHostException e) {
e.printStackTrace();
}
}
}
4. 端口
端口表示计算机上的一个程序的进程。
- 不同的进程有不同的端口号,用来区分软件
- 被规定:0-65535
- TCP/UDP:65535*2, 单个协议下,端口号不能冲突
- 端口分类:
- 公有端口:0~1023
- HTTP : 80
- HTTPS :443
- FTP : 21
- Telet : 23
- 程序注册端口:1024-49151,分配给用户或者程序
- Tomcat:8080
- Mysql:3306
- Oracle:1521
- 动态、私有:49152-65535
- 公有端口:0~1023
netstat -ano #查看所有端口
netstat -ano | findstr "5900" #查看指定的端口
tasklist | findstr "8696" #查看指定端口的进程
Ctrl + Shift + ESC # 打开任务管理器
public class TestInetSocketAddress {
public static void main(String[] args) {
InetSocketAddress socketAddress = new InetSocketAddress("127.0.0.1", 8080);
InetSocketAddress socketAddress1 = new InetSocketAddress("localhost", 8080);
System.out.println(socketAddress); ///127.0.0.1:8080
System.out.println(socketAddress1); //localhost/127.0.0.1:8080
System.out.println(socketAddress.getAddress()); //ip地址 /127.0.0.1
System.out.println(socketAddress.getHostName()); //主机名称 127.0.0.1
System.out.println(socketAddress.getHostString()); //127.0.0.1
System.out.println(socketAddress.getPort());//端口 8080
}
}
5. 通信协议
协议:约定,就好比我们现在说的是普通话。
网络通信协议:速率、传输码率、代码结构、传输控制…
TCP/IP协议簇: 实际上是一组协议
重要:
- TCP:用户传输协议
- UDP:用户数据报协议
出名的协议:
- TCP
- IP
TCP和UDP 对比:
TCP:打电话
- 连接: 稳定
- 三次握手
最少需要三次,保证稳定连接
A:你愁啥?
B:瞅你咋地?
A:干一场!
- 四次挥手
A:我要断开了 (我要走了)
B:我知道你要断开了(你真的要走了吗?)
B:你真的断开了吗?(你真的真的要走了吗?)
A:我真的断开了 (我真的要走了)
- 客户端,服务端
- 传输完成,释放连接、效率低
UDP:发短信
- 不连接,不稳定
- 客户端、服务端:没有明确的界限
- 不管有没有准备好,都可以发给你…
- 导弹
- DDOS:洪水攻击!(饱和攻击)
6. TCP
客户端
- 连接服务器 Socket
- 发送消息
//客户端
public class TcpClientDemo01 {
public static void main(String[] args) {
Socket socket = null;
OutputStream os = null;
try {
// 1. 要知道服务器地址
InetAddress serverIp = InetAddress.getByName("127.0.0.1");
//端口号
int port = 9999;
// 2. 创建一个Socket连接
socket = new Socket(serverIp,port);
// 3. 发送消息 IO流
os = socket.getOutputStream();
os.write("你好,欢迎学习java".getBytes());
} catch (Exception e) {
e.printStackTrace();
}finally {
if (os != null) {
try {
os.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (socket != null) {
try {
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
服务器
- 建立服务的端口 ServerSocket
- 等待用户的连接 accept
- 接受用户的消息
//服务器
public class TcpServerDemo01 {
public static void main(String[] args) {
ServerSocket serverSocket = null;
Socket accept = null;
InputStream is = null;
ByteArrayOutputStream baos = null;
try {
// 1. 我得有一个地址
serverSocket = new ServerSocket(9999);
// 2. 等待客户端连接
accept = serverSocket.accept();
// 3. 读取客户端的消息
is = accept.getInputStream();
//管道流
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 {
//关闭资源
if (baos != null){
try {
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (is != null) {
try {
is.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (accept != null){
try {
accept.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (serverSocket != null){
try {
serverSocket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
TCP实现文件上传
//服务器
public class TcpServerDemo02 {
public static void main(String[] args) throws Exception {
//1.创建服务
ServerSocket serverSocket = new ServerSocket(9000);
//2.监听客户端连接
Socket accept = serverSocket.accept(); //阻塞式监听,会一直等待客户端连接
//3.获取输入流
InputStream is = accept.getInputStream();
//4.文件输出
FileOutputStream fos = new FileOutputStream(new File("receive.png"));//接收文件就要用文件的管道流
byte[] buff = new byte[1024];
int len;
while ((len = is.read(buff)) != -1){
fos.write(buff,0,len);
}
//通知客户端我接收完毕了
OutputStream os = accept.getOutputStream();
os.write("我接收完毕了,你可以断开了".getBytes());
fos.close();
is.close();
accept.close();
serverSocket.close();
}
}
//客户端
public class TcpClientDemo02 {
public static void main(String[] args) throws Exception{
//1.建立连接
Socket socket = new Socket(InetAddress.getByName("127.0.0.1"), 9000);
//2.创建一个输出流
OutputStream os = socket.getOutputStream();
//3.读取文件
FileInputStream is = new FileInputStream(new File("1.png"));
byte[] buff = new byte[1024];
int len;
//4.写出文件
while ((len = is.read(buff)) != -1) {
os.write(buff, 0, len);
}
//通知服务器,我已经结束了
socket.shutdownOutput();//我已经传输完了的意思
//确定服务器接收完毕,才能够断开连接
InputStream inputStream = socket.getInputStream();//接收字符、就用字节的管道流
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buff2 = new byte[1024];
int len2;
while ((len2 = inputStream.read(buff2)) != -1) {
baos.write(buff2, 0, len2);
}
System.out.println(baos.toString());
//5.释放资源
baos.close();
inputStream.close();
is.close();
os.close();
socket.close();
}
}
7. UDP
发短信,不用连接,需要知道对方IP地址。
- 发送消息
发送端
//不需要连接服务器
public class UdpClientDemo1 {
public static void main(String[] args) throws Exception{
//1.建立一个Socket
DatagramSocket socket = new DatagramSocket();
//2.建个包
String msg = "你好啊,服务器";
//3.发送给谁
InetAddress address = InetAddress.getByName("localhost");
int port = 9090;
//数据,数据的起始长度,发送给谁
DatagramPacket packet = new DatagramPacket(msg.getBytes(), 0, msg.getBytes().length, address, port);
//4.发送包
socket.send(packet);
//5.关闭流
socket.close();
}
}
接收端
//还是要等待客户端连接
public class UdpServerDemo1 {
public static void main(String[] args) throws Exception{
//开放端口
DatagramSocket socket = new DatagramSocket(9090);
//接收数据包
byte[] buff =new byte[1024];
DatagramPacket packet = new DatagramPacket(buff, 0, buff.length);
socket.receive(packet);//阻塞接收
System.out.println(packet.getAddress());
System.out.println(new String(packet.getData(),0,packet.getData().length));
//关闭连接
socket.close();
}
}
- UDP 咨询
循环发送消息
发送:
public class UdpSenderDemo01 {
public static void main(String[] args) throws Exception {
DatagramSocket socket = new DatagramSocket(8888);
//准备数据,控制台读取,system.in
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
while (true) { //循环输入
String data = reader.readLine();
byte[] bytes = data.getBytes();
DatagramPacket dgp = new DatagramPacket(bytes, 0, bytes.length, new InetSocketAddress("localhost", 6666));
socket.send(dgp);
if (data.equals("bye")){
break;
}
}
socket.close();
}
}
接收
public class UdpReceiveDemo01 {
public static void main(String[] args) throws Exception {
DatagramSocket socket = new DatagramSocket(6666);
while (true) { //循环接收
//准备接收包裹
byte[] bytes = new byte[1024];
DatagramPacket packet = new DatagramPacket(bytes,0,bytes.length);
socket.receive(packet); //阻塞式接收包裹
//断开连接
byte[] data = packet.getData();
String receiceData = new String(data, 0, data.length);
System.out.println(receiceData);
if (receiceData.equals("bye")){
break;
}
}
socket.close();
}
}
- 在线咨询
两个人都可以是发送方,都可以是接收方
发送:
public class TalkSend implements Runnable {
DatagramSocket socket = null;
BufferedReader reader = null;
private int fromPort;
private String toIp;
private int toPort;
public TalkSend(int fromPort, String toIp, int toPort) {
this.fromPort = fromPort;
this.toIp = toIp;
this.toPort = toPort;
try {
socket = new DatagramSocket(fromPort);
reader = new BufferedReader(new InputStreamReader(System.in));
} catch (SocketException e) {
e.printStackTrace();
}
}
@Override
public void run() {
while (true) { //循环输入
String data = null;
try {
data = reader.readLine();
byte[] bytes = data.getBytes();
DatagramPacket dgp = new DatagramPacket(bytes, 0, bytes.length, new InetSocketAddress(this.toIp, this.toPort));
socket.send(dgp);
if (data.equals("bye")){
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
socket.close();
}
}
接收:
public class TalkReceive implements Runnable {
DatagramSocket socket = null;
private int port;
private String msgFrom;
public TalkReceive(int port, String msgFrom) {
this.port = port;
this.msgFrom = msgFrom;
try {
//开启连接
socket = new DatagramSocket(port);
} catch (SocketException e) {
e.printStackTrace();
}
}
@Override
public void run() {
while (true) { //循环接收
try {
//准备接收包裹
byte[] bytes = new byte[1024];
DatagramPacket packet = new DatagramPacket(bytes,0,bytes.length);
socket.receive(packet); //阻塞式接收包裹
//断开连接
byte[] data = packet.getData();
String receiceData = new String(data, 0, data.length);
System.out.println(msgFrom + ":" + receiceData);
if (receiceData.equals("bye")){
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
socket.close();
}
}
学生:
public class TalkStudent {
public static void main(String[] args) {
//开启两个线程
new Thread(new TalkSend(7777,"localhost",9999)).start();
new Thread(new TalkReceive(8888,"老师")).start();
}
}
老师:
public class TalkTeacher {
public static void main(String[] args) {
//开启两个线程
new Thread(new TalkSend(5555,"localhost",8888)).start();
new Thread(new TalkReceive(9999,"学生")).start();
}
}
8. URL
统一资源定位符:定位资源的,定位互联网上的某个资源。
https://www.baidu.com/
协议://ip地址:端口/项目名/资源
public class TestUrlDemo {
public static void main(String[] args) throws MalformedURLException {
URL url = new URL("http://localhost:8080/hello.java/index.jsp?username=zs&password=123");
System.out.println(url.getProtocol()); //协议 http
System.out.println(url.getHost()); //主机ip localhost
System.out.println(url.getPort()); //端口 8080
System.out.println(url.getPath()); //全路径 /hello.java/index.jsp
System.out.println(url.getFile()); //文件 /hello.java/index.jsp?username=zs&password=123
System.out.println(url.getQuery()); //参数 username=zs&password=123
}
}
下载网络资源
public class TestUrlDown {
public static void main(String[] args) throws Exception {
//下载地址
URL url = new URL("http://localhost:8080/urlTest/test.txt");
//连接到这个资源 http
HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
InputStream is = urlConnection.getInputStream();
FileOutputStream fos = new FileOutputStream("test1.text");
byte[] bytes = new byte[1024];
int len;
while ((len=is.read(bytes)) != -1){
fos.write(bytes,0,len); //写出这个数据
}
fos.close();
is.close();
urlConnection.disconnect(); //断开连接
}
}