网络编程
概述
地球村概念
计算机网络
将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接起来,在网络操作系统,网络管理软件及网络通信协议的管理和协调下,实现资源共享和信息传递的计算机系统
网络编程的目的
传播交流信息(数据交换,通信)
达到这个效果需要什么
- 如何准确定位网络上的一台主机 192.168.xx.xxx:端口,定位到这个计算机上的某个资源
- 找到了主机后,如何传输数据
网络编程:TCP/IP
网络通信的要素
人工智能:智能汽车(仅仅运用在工厂,因为人少) => 伦理问题
如何实现网络的通信?
(1)通信双方的地址:ip,端口号
(2)规则:网络通信的协议 eg:http,TCP
重点:传输层 TCP,UDP
小结
- 网络编程中有两个主要问题:(1)如何如何准确定位网络上的一台或多台主机 (2)找到主机后如何进行通信
- 网络编程中的要素(1)IP和端口号 (2)网络通信协议 tcp,udp
- java万物皆对象
IP
InetAddress
ip作用:
- 唯一定位一台网络上的计算机
- 127.0.0.1: 本机 localhost
- ip地址的分类:
- ipv4/ipv6
- IPV4:127.0.0.1,4个字节组成,0~255,大约42亿;30亿都在北美,亚洲四亿,在2011年就用尽了。
- IPV6:128位,8个无符号整数
- 公网(互联网)和私网(局域网)
- 192.168.xx.xx 专门给组织内部使用的
- ABCD类地址
- ipv4/ipv6
- 域名:记忆IP问题
端口
端口表示计算机上的一个程序的进程
-
不同的进程有不同的端口号!
-
被规定 0~65535
-
TCP,UDP:65535*2
-
端口分类
-
共有端口 0~1023
- HTTP:80
- HTTPS:443
- FTP:21
- Telnet:23
-
程序注册端口:1024~49151,分配给用户或者重续
- Tomcat:8080
- MySQL:3306
- Oracle:1521
-
动态,私有端口:49152~65535
netstat -ano #查看所有端口 netstat -ano|findstr "" #查看指定的端口 tasklist|findstr "" #查看指定端口的进程
-
package com.nocilantro.demo01; import java.net.InetAddress; import java.net.UnknownHostException; /** * 测试IP */ public class TestInetAddress { public static void main(String[] args) { try { // 查询本机地址 InetAddress inetAddress1 = InetAddress.getByName("127.0.0.1"); System.out.println(inetAddress1); InetAddress inetAddress3 = InetAddress.getByName("localhost"); System.out.println(inetAddress3); InetAddress inetAddress4 = InetAddress.getLocalHost(); System.out.println(inetAddress4); // 查询网站ip地址 InetAddress inetAddress2 = InetAddress.getByName("www.baidu.com"); System.out.println(inetAddress2); // 常用方法 System.out.println(inetAddress2.getAddress()); System.out.println(inetAddress2.getCanonicalHostName()); System.out.println(inetAddress2.getHostAddress()); System.out.println(inetAddress2.getHostName()); } catch (UnknownHostException e) { e.printStackTrace(); } } }
-
package com.nocilantro.demo01; import java.net.InetSocketAddress; public class TestInetSocketAddress { public static void main(String[] args) { InetSocketAddress inetSocketAddress = new InetSocketAddress("127.0.0.1", 8080); InetSocketAddress inetSocketAddress2 = new InetSocketAddress("localhost", 8080); System.out.println(inetSocketAddress); System.out.println(inetSocketAddress2); System.out.println(inetSocketAddress.getAddress()); System.out.println(inetSocketAddress.getHostName()); // 地址 System.out.println(inetSocketAddress.getPort()); // 端口 } }
-
通信协议
协议:约定,比如说的语言(普通话)
网络通信协议:速率,传输码率,代码结构,传输控制… (非常复杂)
TCP/IP协议簇
- TCP:用户传输协议
- UDP:用户数据报协议
- IP:网络互连协议
TCP UDP对比
TCP:打电话
-
连接稳定
-
三次握手,四次挥手
-
三次握手:最少需要三次才能保证稳定连接 A:你瞅啥? B:瞅你咋地? A:干一场 四次挥手:为了双方都释放连接的资源 A:我要走了 B:我知道你要走了吗 B:你真的真的要走了吗 A:我真的要走了
-
-
客户端,服务端
-
传输完成,释放连接,效率低
UDP:发短信
- 不连接,不稳定
- 客户端、服务端:没有明确的界限
- 不管有没有准备好,都可以发给你
- DDOS:洪水攻击
TCP
客户端
- 连接服务器Socket
- 发送消息
package com.nocilantro.demo02;
import java.io.IOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
/**
* 客户端
*/
public class TCPClientDemo01 {
public static void main(String[] args) {
Socket socket = null;
OutputStream os = null;
try {
// 要知道服务器的地址,端口号
InetAddress serverIP = InetAddress.getByName("127.0.0.1");
int port = 9999;
// 创建一个socket连接
socket = new Socket(serverIP, port);
// 发送消息
os = socket.getOutputStream();
os.write("Welcome".getBytes());
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
if (os != null) {
try {
os.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (os != null) {
try {
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
服务器
- 建立服务的端口 ServerSocket
- 等到用户的连接 accept
- 接收用户的消息
package com.nocilantro.demo02;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
/**
* 服务器
*/
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);
// 等待客户端连接
accept = serverSocket.accept();
is = accept.getInputStream();
// 管道流
baos = new ByteArrayOutputStream();
byte[] buf = new byte[1024];
int len = 0;
while ((len = is.read(buf)) != -1) {
baos.write(buf, 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();
}
}
}
}
}
文件上传案例
package com.nocilantro.demo02;
import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
/**
* 客户端读取文件
*/
public class TCPClientDemo02 {
public static void main(String[] args) throws Exception{
// 创建一个socket连接
Socket socket = new Socket(InetAddress.getByName("127.0.0.1"), 9000);
// 创建一个输出流
OutputStream os = socket.getOutputStream();
// 文件流 读取文件
FileInputStream fis = new FileInputStream("test.png");
// 写出文件
byte[] buf = new byte[1024];
int len = 0;
while ((len = fis.read(buf)) != -1) {
os.write(buf, 0, len);
}
// 通知服务器传输完了
socket.shutdownOutput();
// 确定服务器接收完毕,才能断开连接
InputStream inputStream = socket.getInputStream();
// String => byte[]
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buf2 = new byte[1024];
int len2 = 0;
while ((len2 = inputStream.read(buf2)) != -1) {
baos.write(buf2, 0, len2);
}
System.out.println(baos.toString());
// 关闭资源
baos.close();
inputStream.close();
fis.close();
os.close();
socket.close();
}
}
package com.nocilantro.demo02;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class TCPServerDemo02 {
public static void main(String[] args) throws Exception{
// 创建一个ServerSocket
ServerSocket serverSocket = new ServerSocket(9000);
// 监听客户端的连接
Socket socket = serverSocket.accept(); // 阻塞式监听,会一直等待客户端连接
// 获取输入流
InputStream is = socket.getInputStream();
// 文件输出
FileOutputStream fos = new FileOutputStream("receive.png");
byte[] buf = new byte[1024];
int len = 0;
while ((len = is.read(buf)) != -1) {
fos.write(buf, 0, len);
}
// 通知客户端接收完毕
OutputStream outputStream = socket.getOutputStream();
outputStream.write("我接受完毕了".getBytes());
// 关闭资源
fos.close();
is.close();
socket.close();
serverSocket.close();
}
}
Tomcat
服务端
- 自定义 S
- Tomcat服务器 S
客户端
- 自定义 C
- 浏览器 B
UDP
发短信:不用连接,需要知道对方的地址
相关类:
-
DatagramPacket
-
DatagramSocket
发送消息
package com.nocilantro.demo03;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
public class UDPClientDemo01 {
public static void main(String[] args) throws Exception{
// 建立一个Socket
DatagramSocket datagramSocket = new DatagramSocket();
// 建一个包
String msg = "Hello";
DatagramPacket packet =
new DatagramPacket(msg.getBytes(), 0, msg.getBytes().length, InetAddress.getByName("localhost"), 9090);
// 发送包
datagramSocket.send(packet);
// 关闭资源
datagramSocket.close();
}
}
接收消息
package com.nocilantro.demo03;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
// 还是要等到客户端的连接
public class UDPServerDemo01 {
public static void main(String[] args) throws Exception{
// 开放端口
DatagramSocket socket = new DatagramSocket(9090);
// 接收数据包
byte[] buf = new byte[1024];
DatagramPacket packet = new DatagramPacket(buf, 0, buf.length);// 接收
socket.receive(packet); // 阻塞接收
System.out.println(new String(packet.getData(), 0, packet.getLength()));
// 关闭连接
socket.close();
}
}
在线咨询:两人都可以是发送方,也都可以是接收方
package com.nocilantro.chat;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
import java.net.SocketException;
public class UDPSenderDemo01 implements Runnable{
DatagramSocket socket;
BufferedReader reader;
private int fromPort;
private String toIP;
private int toPort;
public UDPSenderDemo01(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) {
try {
String data = reader.readLine();
byte[] dataByte = data.getBytes();
DatagramPacket packet = new DatagramPacket(dataByte, 0, dataByte.length, new InetSocketAddress(toIP, toPort));
socket.send(packet);
if (data.equals("bye")) {
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
socket.close();
}
}
package com.nocilantro.chat;import java.net.DatagramPacket;import java.net.DatagramSocket;import java.net.SocketException;public class UDPReceiverDemo01 implements Runnable{ DatagramSocket socket; private int port; public UDPReceiverDemo01(int port) { this.port = port; try { socket = new DatagramSocket(port); } catch (SocketException e) { e.printStackTrace(); } } @Override public void run() { while (true) { try { // 准备接收包裹 byte[] container = new byte[1024]; DatagramPacket packet = new DatagramPacket(container, 0, container.length); socket.receive(packet); // 断开连接bye byte[] data = packet.getData(); String receive = new String(data, 0, data.length); System.out.println("From " + port + ": " + receive); if (receive.equals("bye")) { break; } } catch (Exception e) { e.printStackTrace(); } } socket.close(); }}
package com.nocilantro.chat;import java.net.SocketException;public class Talker1 { public static void main(String[] args) throws SocketException { new Thread(new UDPSenderDemo01(6666, "localhost", 7777)).start(); new Thread(new UDPReceiverDemo01(8888)).start(); }}
package com.nocilantro.chat;import java.net.SocketException;public class Talker2 { public static void main(String[] args) throws SocketException { new Thread(new UDPSenderDemo01(5555, "localhost", 8888)).start(); new Thread(new UDPReceiverDemo01(7777)).start(); }}
URL
统一资源定位符:定位互联网上的某一个资源
DNS域名解析
协议: //ip地址:端口/项目名/资源