一、概述
- 计算机网络:
计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接起来,在网络操作系统,网络管理软件及网络通信协议的管理和协调下,实现资源共享和信息传递的计算机系统。 - 网络编程的目的:
传播交流信、数据交换、通信(无线电台) - 想要达到这个效果需要什么:
- 如何准确的定位网络上的一台主机 ip地址:端口,定位到这个计算机上的某个资源
- 找到这个主机,(如何传输数据呢?)
javaweb开发:网页编程 B/S架构
网络编程: TCP/IP C/S架构
二、网络通信的要素
- 网络编程中有两个主要的问题:
- 如何准确的定位到网络上的一台或者多台主机
- 找到主机之后如何进行通信
- 网络编程中的要素:
- IP和端口号
- 网络通信协议
- 面向对象:万物皆对象
TCP/IP参考模型:
通过域名获得目标ip地址:
三、IP
- ip地址:
- 唯一定位一台网络上计算机
- 127.0.0.1:本机localhost
- ip地址的分类:
- IP地址分类:IPv4、IPv6
- IPV4 127.0.0.1 4个字节组成,0-255。(共42亿个,30亿都在北美,亚洲4亿,2011年就用完了)
- IPV6 2001:0bbb:3451:0000:… 8个无符号整数组成(0-9,a-e),128位。
- 公网-私网
- IP地址分类:IPv4、IPv6
测试ip-代码实现:
package kuangshen;
import java.net.InetAddress;
import java.net.UnknownHostException;
//测试ip
public class TestInetAddress {
public static void main(String[] args) {
// new InetAddress();//由于这个方法没有构造器,所以不可以直接实例化,应调用它的静态方法返回它本身
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);
///127.0.0.1
//localhost/127.0.0.1
//查询网站ip地址
InetAddress inetAddress2 = InetAddress.getByName("wwww.baidu.com");//捕获异常
System.out.println(inetAddress2);
//wwww.baidu.com/220.181.38.148
} catch (UnknownHostException e) {
e.printStackTrace();
}
}
}
使用套接字获取ip地址-代码实现:
package kuangshen;
import java.net.InetSocketAddress;
//使用套接字获取ip地址
public class TsetInetSocketAddress {
public static void main(String[] args) {
InetSocketAddress socketAddress=new InetSocketAddress("127.0.0.1",8080);
InetSocketAddress socketAddress2=new InetSocketAddress("localhost",8080);
System.out.println(socketAddress);
System.out.println(socketAddress2);
System.out.println(socketAddress.getAddress());
System.out.println(socketAddress.getHostName());//地址
System.out.println(socketAddress.getPort());//端口
///127.0.0.1:8080
//localhost/127.0.0.1:8080
///127.0.0.1
//127.0.0.1
//8080
}
}
四、端口(PID)
-
端口表示计算机上的一个程序的进程;
- 不同的进程有不同的端口号。用来区分软件。
- 被规定范围:0-65535
- TCP,UDP:共65535*2个端口。单个协议下,端口号不能冲突(但是可以TCP:80,UDP:80)。
- 端口分类:
- 公有端口:0-1023(百度为例:HTTP:80,HTTPS:443)
- 程序注册端口:1024-49151,分配用户或者程序(Tomcat:8080,Mysql:3306,Oracle:1521)
- 动态、私有:49152-65535(举netstat为例)
- netstat -ano #查看所有的端口
- netstat -ano|findstr “5900” # | :管道符,起过滤作用。#查看指定端口:5900
- tasklist|findstr “8696” #查看指定端口的进程
端口对应-实现通信(左QQ->右QQ发信息)
五、通信协议
- TCP/IP协议簇:实际上是一组协议
- 重要协议:TCP(用户传输协议),UDP(用户数据报协议),IP(网络互连协议)
- TCP和UDP对比
- TCP(打电话):
- 需要连接、稳定
- 三次握手,四次挥手
- 客户端、服务端
- 传输完成,释放连接,效率低
- UDP(发短信):
- 不需要连接、不稳定
- 客户端、服务端:没有明确的界限
- 不管对方有没有准备好,都可以发送(导弹、DDOS:洪水攻击(饱和攻击))
- TCP(打电话):
六、TCP
知识补充:
b – 数据
off – 在数据偏移的开始
len – 写入的字节数
客户端
- 连接服务器Socket
- 发送消息
客户端-代码实现:
package kuangshen;
import java.io.IOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.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("你好,我是客户".getBytes());
} catch (Exception e) {
e.printStackTrace();
}finally {
//关闭数据流
if (socket!=null){
try {
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (os!=null){
try {
os.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
服务端
- 建立服务的端口ServerSocket
- 等待用户的连接accept
- 接收用户的消息
服务器-代码实现:
package kuangshen;
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);
//2.等待客户端连接过来
accept = serverSocket.accept();//accept和客户端中的socket相同
//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(流!=null)不是指数据还在传输,意思是通过这个判断流有传输数据,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();
}
}
}
}
}
案例:文件上传
服务端
代码实现:
package kuangshen;
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 TcpServerDemo02 {
public static void main(String[] args) throws Exception{
//1.创建服务
ServerSocket serverSocket = new ServerSocket(9000);
//2.监听客户端的连接
Socket accept = serverSocket.accept();//阻塞式监听,会一直等待客户端连接类似于Scanner方法
//3.获取输入流
InputStream is = accept.getInputStream();
//4.文件输出
FileOutputStream fis = new FileOutputStream(new File("recevice.png"));
byte[] buffer = new byte[1024];
int len;
while ((len=is.read(buffer))!=-1){
fis.write(buffer,0,len);//从下标0开始,写入buffer数组,共写入len个字节
}
//通知客户端接收完毕
OutputStream os = accept.getOutputStream();
os.write("我接受完毕了".getBytes());//返回byte类型的数据
//5.关闭数据流
fis.close();
is.close();
accept.close();
serverSocket.close();
}
}
客户端
代码实现:
package kuangshen;
import java.io.*;
import java.net.InetAddress;
import java.net.Socket;
public class TcpClientDemo02 {
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("C:\\...\\02.png"));
//4.写出文件
byte[] buffer=new byte[1024];
int len;
while ((len=fis.read(buffer))!=-1){
os.write(buffer,0,len);
}
//通知服务器,已经结束了
//避免的问题:服务器的read方法一直在等下一组数据,而客户端早就结束了传输,但是服务器不知道,而客户端传输完之后就关闭数据流了
socket.shutdownOutput();//我已经传输完了
//确定服务器接收完毕,才能断开连接
InputStream inputStream = socket.getInputStream();
//getByte方法,返回的是byte类型的数据
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buffer2=new byte[1024];
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();
}
}
Tomcat
客户端
- 自定义:C
- 浏览器:B
服务端
- 自定义:S
- Tomcat服务器:S,用于java后台开发
UDP
案例:发送消息
发送端
代码实现:
package kuangshen;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
//不需要连接接收端一样可以发送,UDP中没有服务器这个概念,两遍都是接收端,也都是发送端
public class UdpClientDemo01 {
public static void main(String[] args) throws Exception{
//1.创建一个Socket
DatagramSocket socket = new DatagramSocket();
//2.创建个包
String msg="你好,服务器";
//2.1发送给谁
InetAddress localhost = InetAddress.getByName("localhost");
int port=9090;
//DatagramPacket(数据,数据长度的开始,数据长度的结尾,接受者的IP,端口)
DatagramPacket packet = new DatagramPacket(msg.getBytes(), 0, msg.getBytes().length, localhost, port);
//3.发送包
socket.send(packet);
//4.关闭流
socket.close();
}
}
接收端
代码实现:
package kuangshen;
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[] 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();
//127.0.0.1
//你好,服务器
}
案例:咨询
知识补充:
在接收端收到消息进行equals字符串比较时,由于创建byte数组时,设置长度为1024,之后读取的字符串长度也是1024,与设置的bye字符串的长度不一致导致结果总是false。由此看出,用equals方法比较时,首先比较的是读取的字符串长度,如果其中由于包含空格、复制错误,将不能正确比较,所以此时需要使用trim方法。代码为:receiveData.trim().equals(“bye”),此时当读取到bye时,将停止接收。
- 问题二:使用过BufferedReader之后,为什么不关闭:
- 引用文章:点此进入
- 直接调用最外层的close():一个链条,关闭了最外面的->最外面的关闭里面的->里面的关闭更里面的。
- 看BufferedReader的close源码,它关闭的是它说装饰的Reader,而Reader的关闭方法这是直接使用close方法。
- 以上两点为可以直接关闭最外层的,如果都要关闭,则按照:最先创建的最后关,最后创建的最先关的顺序在finally中依次关闭。
发送端
代码实现:
package kuangshen;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetSocketAddress;
public class UdpSenderDemo01 {
public static void main(String[] args) throws Exception{
//创建一个Socket
DatagramSocket socket = new DatagramSocket(8888);
//准备数据:控制台读取System.in
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
while (true){
//读取数据
String data = reader.readLine();
byte[] datas = data.getBytes();
DatagramPacket packet = new DatagramPacket(datas, 0, datas.length, new InetSocketAddress("localhost",6666));
//发送数据
socket.send(packet);
if (data.equals("over")){
break;
}
}
socket.close();
}
}
接收端
代码实现:
package kuangshen;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
public class UdpReceiveDemo01 {
public static void main(String[] args) throws Exception{
DatagramSocket socket = new DatagramSocket(6666);
while (true){
//准备接收包裹
byte[] container = new byte[1024];
DatagramPacket packet = new DatagramPacket(container, 0, container.length);
socket.receive(packet);//阻塞式接收包裹
//断开连接 读取到 bye时
byte[] data=packet.getData();
String receiveData = new String(data, 0, data.length);
System.out.println(receiveData);
if (receiveData.trim().equals("bye")){
break;
}
}
socket.close();
}
}
在线咨询:学生与教师之间相互通信
- 两人都可以是发送方,也都可以是接收方。
TalkSend-代码实现:
package kuangshen;
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 TalkSend implements Runnable {
DatagramSocket socket =null;
BufferedReader reader =null;
private int fromIP;
private String toIP;
private int toPort;
public TalkSend(int fromIP, String toIP, int toPort) {//负责从那里发、发到哪
this.fromIP = fromIP;
this.toIP = toIP;
this.toPort = toPort;
try {
socket = new DatagramSocket(8888);
//准备数据:控制台读取System.in
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[] datas = data.getBytes();
DatagramPacket packet = new DatagramPacket(datas, 0, datas.length, new InetSocketAddress(this.toIP,this.toPort));
//发送数据
socket.send(packet);
if (data.equals("over")){
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
TalkReceive-代码实现:
package kuangshen;
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
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[] container = new byte[1024];
DatagramPacket packet = new DatagramPacket(container, 0, container.length);
socket.receive(packet);
//断开连接 读取到 bye时
byte[] data=packet.getData();
String receiveData = new String(data, 0, data.length);
System.out.println(msgFrom+":"+receiveData);
if (receiveData.trim().equals("bye")){
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
socket.close();
}
}
学生端-代码实现:
package kuangshen;
public class TalkStudent {
public static void main(String[] args) {
//开启两个线程
new Thread(new TalkSend(7777,"localhost",9999)).start();
new Thread(new TalkReceive(8888,"老师")).start();
}
}
教师端-代码实现:
package kuangshen;
public class TalkTeacher {
public static void main(String[] args) {
//从端口5555发送给localhost的8888端口
new Thread(new TalkSend(5555,"localhost",8888)).start();
//接收端口9999,给“学生”说
new Thread(new TalkReceive(9999,"学生")).start();
}
}
图解:
补充:
- 本代码最终运行,会提示端口被占用的错误:java.net.BindException: Address already in use: Cannot bind。解决方法:点此进入
URL
- 统一资源定位符:用来定位资源的,定位互联网上的某一个资源
- DNS域名解析:www.baidu.com-----》xxx.xxx.xxx.xxx…(IP地址)
基础方法-代码实现:
package kuangshen.wlbc;
import java.net.MalformedURLException;
import java.net.URL;
public class URLDemo01 {
public static void main(String[] args) throws MalformedURLException {
//捕获异常(因为这个URL不存在),Tomcat端口号:8080
URL url = new URL("http://localhost:8080/helloworld/inde.jsp?uesrname=kuangshen&password=123");
System.out.println(url.getProtocol());//获得协议名
System.out.println(url.getHost());//获得主机ip
System.out.println(url.getPort());//获得端口号
System.out.println(url.getPath());//获得文件
System.out.println(url.getFile());//获得全路径
System.out.println(url.getQuery());//获得url查询的名字、参数
//http
//localhost
//8080
///helloworld/inde.jsp
//uesrname=kuangshen&password=123
}
}
根据URL下载网络资源-代码实现:
package kuangshen.wlbc;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
public class UrlDown {
public static void main(String[] args) throws Exception{
//1.下载地址
URL url = new URL("Http://nie.res.netease.com/r/pic/20210608/61039f93-a986-42bf-8da6-6b7f875c1cdf.jpg");
//2.连接到这个资源 HTTP
HttpURLConnection urlConnection=(HttpURLConnection) url.openConnection();
InputStream inputStream=urlConnection.getInputStream();
FileOutputStream fis = new FileOutputStream("yjwj.jpg");
byte[] buffer=new byte[1024];
int len;
while ((len=inputStream.read(buffer))!=-1){
fis.write(buffer,0,len);//写出这个数据
}
fis.close();
inputStream.close();
urlConnection.disconnect();//断开连接
}
}
步骤:F12—》网络