网络编程
1.1概述
计算机网络:
将不同位置的电脑通过某种协议连接起来,实现通信资源共享
网络编程的目的:
无线电台--传播交流信息,数据交流、通信
想要达到这个点需要什么:
- 如何准确的定位网络上的一台主机? IP地址+port口,定位到计算机上的某个资源(如QQ)
- 找到这个主机,如何传输数据呢?(协议)
javaweb:网页编程 B/S
网络编程:TCP/IP C/S
TCP&UDP 类比
打电话 -- 连接--接了--通话 TCP
发短信--发送就完事--接收 UDP (不稳定,会丢包)
1.2 网络通信的要素
人工智能:智能汽车:工厂、人少。目前不能量产涉及伦理
如何实现网络的通信?
通信双地址:
- ip
- 端口号 (www.baidu.com域名和IP是对应的)
规则: 网络通信的协议
![]()
1.3 IP
ip地址:类InetAddress 不能new,api里构造函数灰色
唯一定位一台网络上计算机
127.0.0.1 本机localhost
ip地址的分类
- Ip地址分类Ipv4/ipv6
- ipv4:127.0.0.1 4个字节组成,共 0-255, 42亿~,30亿在北美,亚洲4亿,2011年用尽
- ipv6:fe80::915d:470e:d522:4339%16,共128位,8个无符号整数,四个数字一组=16位,每个整数用16进制表示,当有连续多个0可以用:省略代替,但是只可以用一次[https://zhidao.baidu.com/question/748939481379152052.html]()
- 公网(互联网)-私网(局域网)
- ABCD类地址,字段分开给不同组织用
- 192.168.xx.xx 专门给组织内部使用的
*域名:记忆IP问题! - IP:www.vip.com 越短越贵
1.4 端口
端口表示计算机傻瓜一个程序的进程(类比一栋楼是一个IP,每一户是一个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 -anao|findstr "5900" # 查看指定的端口 tasklist|findstr "8696" # 查看指定端口对应的应用进程
- 公有端口 0-1023
- InetSocketAddress
InetSocaketAddress soketAddress2 = new InetSocketAddress("localhost",8080);
socketAddress.getHostName();//对应window系统C:\Windows\System32\drivers\etc\host里面可以设置IP和域名的对应关系
1.5 通信协议
- 协议:约定通信规则
- 网络通信协议:速率,传输马路,代码结构,传输控制...
- 问题:非常复杂
大事化小:分层
TCP/IP协议(实际上是一组协议)
TCP:用户传输协议
UDP:用户传输协议
IP: 网络互连协议 - TCP:打电话
- 连接,稳定
- 三次握手,四次挥手
握手最少需要三次,保证稳定连接
A:你瞅啥?
B:瞅你咋地?
A:干一场!
分开四次
A:我要走了!
B:你要走了?
B:你真的要走了?
A:我真的要走了!
- 区分客户端,服务端
- 传输完成,释放连接,效率低
- UDP:发短信
- 不连接,不稳定
- 客户端服务端,没有明确的界限
- 不管有没有准备号,都可以发送给你
- 导弹,DDOS 洪水攻击,造成端口堵塞(饱和攻击)
1.6 TCP
客户端
- 连接服务器socket
- 发送消息
服务器 - 建立服务的端口 ServerSocket
- 通过accept等待客户端连接
- 接收消息
例1 发送接收消息
TcpServerDemo01.java
package demo1;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class TcpServerDemo01 {
public TcpServerDemo01() {
// TODO Auto-generated constructor stub
}
public static void main(String[] args) {
ServerSocket serverSocket = null;
Socket accept = null;
InputStream is = null;
ByteArrayOutputStream baos = null;
try {
// 1. should have a address, test on same computer local host
serverSocket = new ServerSocket(9999);
while (true) {
// 2. wait clent to connect
accept = serverSocket.accept();
// 3.read client message
is = accept.getInputStream();
/*
* if has chinese word,broke 1024 may broke word into garbled
* word
* byte[] buffer = new byte[1024];
* int len;
* while ((len = is.read(buffer))!=-1)
* { String msg = new String(buffer,0,len);
* System.out.print(msg); }
*/
// Pipestream ---ByteArrayOutputStream
baos = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len;
while ((len = is.read(buffer)) != -1) {
baos.write(buffer, 0, len);
}
// all the bytes are written into pipestream, and output at one
// time.
System.out.print(baos.toString());
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// close resource
try {
if (baos != null) {
baos.close();
}
if (is != null) {
is.close();
}
if (accept != null) {
is.close();
}
if (serverSocket != null) {
serverSocket.close();
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
TcpClientDemo01.java
package demo1;
import java.io.IOException;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.Socket;
public class TcpClientDemo01 {
public TcpClientDemo01() {
// TODO Auto-generated constructor stub
}
public static void main(String[] args){
Socket socket = null;
OutputStream os = null;
try {
//1.should know server address,port
InetAddress serverIP = InetAddress.getByName("127.0.0.1");
int port = 9999;
//2. create a socket connnection
socket = new Socket(serverIP,port);
//3. send message
os = socket.getOutputStream();
os.write("hello, welcome to learn java".getBytes());
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
try {
if(os != null){
os.close();
}
if(socket!=null){
socket.close();
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
扩展:
server端读数据打印到屏幕上,第二种方法不同于注释掉的第一种方法,使用了管道。
将所有数据读入管道流,然后再一次性输出,不会出现中文占两位被截断,输出乱码的问题。

例2 发送接收文件
TcpClientDemo02.java
package demo1;
import java.io.ByteArrayOutputStream;
import java.io.File;
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{
//1. create socket connection
Socket socket = new Socket(InetAddress.getByName("127.0.0.1"),9000);
OutputStream os = socket.getOutputStream();
//2. prepare fileinputStream
FileInputStream fis = new FileInputStream(new File("food.png"));
//3. read and write file
byte[] buffer = new byte[1024];
int len;
while((len = fis.read(buffer))!= -1){
os.write(buffer, 0, len);
}
//notify server output is over
socket.shutdownOutput();
InputStream is = socket.getInputStream();
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buffer2 = new byte[1024];
int len2;
// ensure server finish reception
while ((len2 = is.read(buffer2))!=-1){
baos.write(buffer2, 0, len2);
}
System.out.println(baos.toString());
baos.close();
fis.close();
os.close();
socket.close();
}
}
TcpServerDemo02.java
package demo1;
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. create service, IP is local PC, only need port
ServerSocket serverSocket = new ServerSocket(9000);
Socket accept = serverSocket.accept();
// 3. get inputstream
InputStream is = accept.getInputStream();
//4. get outputStream.
//refer to project, picture and src directory is in same level.
FileOutputStream fos = new FileOutputStream(new File("copy.png"));
byte[] buffer = new byte[1024];
int len;
while((len = is.read(buffer))!= -1){
fos.write(buffer, 0, len);
}
// nofify client has finished
OutputStream os = accept.getOutputStream();
os.write("Have received all".getBytes());
fos.close();
is.close();
serverSocket.close();
}
}
扩展:
- socket.shutdownOutput/shutdownInput作用
在C/S基本结构中,为了防止同一个流中,会出现输入/输出流发生交叉影响,会先设置好输出输入的先后顺序,然后在写入客户端或者服务端的时候,要将优先使用过的输入输出流进行暂时性关闭,保证输入输出流的数据流畅和准确性(解释refer 吾爱破解)
此处client端先socket.shutdownOutput,通知server端写完了。(关闭流没特别懂,后续懂了补详细解释) - Tomcat
服务端
- 自定义 S
- Tomcat服务器 S: JAVA后台开发用别人的服务器
客户端 - 自定义 C
- 浏览器 B
1.7 UDP
发短信,不用连接,但需要知道对方的地址
例1. UDP发送消息
接收端
package demo1;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
public class UdpServerDemo01 {
public static void main(String[] args) throws Exception {
//1. open the port
DatagramSocket socket = new DatagramSocket(9000);
//2. receive packet
byte[] buffer = new byte[1024];
DatagramPacket packet =new DatagramPacket(buffer,0,buffer.length);
socket.receive(packet);
System.out.println(new String(packet.getData(),0,packet.getLength()));
socket.close();
}
}
发送端
package demo1;
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. create socket
DatagramSocket socket = new DatagramSocket();
//2. create packet
String msg = "hello server";
InetAddress IP = InetAddress.getByName("localhost");
int port = 9000;
DatagramPacket packet = new DatagramPacket(msg.getBytes(),0,msg.getBytes().length,IP,port);
//2. send packet
socket.send(packet);
socket.close();
}
}
例2 UDP聊天(应用网页上咨询客服对话框)
例2.1咨询
UdpSenderDemo01.java
package demo1;
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 {
DatagramSocket socket = new DatagramSocket(8888);
while(true){
// prepare data from console
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
String data = reader.readLine();
byte[] datas = data.getBytes();
DatagramPacket packet = new DatagramPacket(datas, 0, datas.length, new InetSocketAddress("localhost",6666));
socket.send(packet);
if("bye bye".equals(data)){
socket.close();
break;
}
}
}
}
UdpReceiverDemo01.java
package demo1;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
public class TcpReceiverDemo01 {
public static void main(String[] args) throws Exception {
DatagramSocket socket = new DatagramSocket(6666);
while (true) {
byte[] datas = new byte[1024];
DatagramPacket packet = new DatagramPacket(datas, datas.length);
socket.receive(packet);// block until receive packet
String receiveData = new String(packet.getData());
/* 错了这样写,byte[]数组长度为1024转化位字符串长度也是1024,前面是有效字符后面追加的都是空个'\0'
应调用trim把空格剪除再进行比较
if ("bye bye".equals(receiveData)){
socket.close();
break;
}
*/
System.out.println(receiveData);
if ("bye bye".equals(receiveData.trim())){
socket.close();
break;
}
}
}
}
例2.2在线咨询
两个人都可以是发送方,也是接收方,因为有while(true)在循环接收发送,所以要启动两个线程来实现
建立连接一定在run方法外面,然后初始化赋值的都在构造器里,run方法里真正主体是循环等待发送接收
抽象两个工具类实现runnable接口,一个线程专门用来发消息,一个专门用来接收消息
一个线程占用一个端口,发送的端口号随意,但是接收的端口要保证一致
TalkReceive.java
package demo1;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
public class TalkReceive implements Runnable {
DatagramSocket socket = null;
int fromPort;
private String msgFrom;
public TalkReceive(int fromPort, String msgFrom) {
super();
this.fromPort = fromPort;
this.msgFrom = msgFrom;
try {
// Constructor can't throw exception out, except deal inside
socket = new DatagramSocket(fromPort);
} catch (SocketException e) {
e.printStackTrace();
}
}
public void run() {
while (true) {
try {
byte[] datas = new byte[1024];
DatagramPacket packet = new DatagramPacket(datas, datas.length);
socket.receive(packet);// block until receive packet
String receiveData = new String(packet.getData());
System.out.println(msgFrom+": "+receiveData);
if ("bye bye".equals(receiveData.trim())) {
socket.close();
break;
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
TalkSend.java
package demo1;
import java.io.BufferedReader;
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 fromPort;
private String toIP;
private int toPort;
public TalkSend(int fromPort, String toIP, int toPort) {
super();
this.fromPort = fromPort;
this.toIP = toIP;
this.toPort = toPort;
try {
socket = new DatagramSocket(fromPort);
reader = new BufferedReader(new InputStreamReader(
System.in));
} catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void run() {
while (true) {
try {
// prepare data from console
String data = reader.readLine();
byte[] datas = data.getBytes();
DatagramPacket packet;
packet = new DatagramPacket(datas, 0, datas.length,
new InetSocketAddress(this.toIP, this.toPort));
socket.send(packet);
if ("bye bye".equals(data)) {
socket.close();
break;
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
调用线程聊天的二者,老师,学生
TalkTeacher.java
package demo1;
public class TalkTeacher {
public static void main(String[] args) {
new Thread(new TalkSend(6666,"localhost",9999)).start();
new Thread(new TalkReceive(8888,"Student")).start();
}
}
TalkStudent.java
package demo1;
public class TalkStudent {
public static void main(String[] args) {
// open two thread
new Thread(new TalkSend(7777,"localhost",8888)).start();
//one thread occupy one port, so fromIp is different
new Thread(new TalkReceive(9999,"Teacher")).start();
}
}
测试结果(用学的知识做个东西,感觉妙啊,视频里同学都刷秒)

1.8 URL
https://www.baidu.com/
统一资源定位符:定位资源的,定位互联网上的某一个资源
DNS域名解析 www.baidu.com xxx.xxx.xxx.xx IP地址
一个服务器里可以跑几个网站(几个项目)
URL组成部分如下,可以少,但是不可以多
协议://IP地址:端口/项目名/资源
例1 URL类
public class URLDemo01 {
public static void main(String[] args) throws MalformedURLException {
URL url = new URL("http://localhost:8080/helloworld/index.jsp?"
+ "username=xiaofang&password=123");
System.out.println(url.getProtocol()); //协议
System.out.println(url.getHost()); //IP
System.out.println(url.getPort()); //Port
System.out.println(url.getPath()); //文件
System.out.println(url.getFile()); //文件全路径
System.out.println(url.getQuery());//参数
}
}
输出结果
public class URLDemo01 {
public static void main(String[] args) throws MalformedURLException {
URL url = new URL("http://localhost:8080/helloworld/index.jsp?"
+ "username=xiaofang&password=123");
System.out.println(url.getProtocol());
System.out.println(url.getHost());
System.out.println(url.getPort());
System.out.println(url.getPath());
System.out.println(url.getFile());
//the URL parameter
System.out.println(url.getQuery());
}
}
例2测试URL下载文件
网络上所有的东西都是以流的形式存在的,可以爬取
package demo1;
import java.io.File;
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.download address
URL url = new URL("https://ss3.bdstatic.com/70cFv8Sh_Q1YnxGkpoWK1HF6hhy/it/u=2380135284,3640663381&fm=26&gp=0.jpg");
//2. connect to this address
HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
//All the things in website are Stream
InputStream inputStream = urlConnection.getInputStream();
FileOutputStream fos = new FileOutputStream (new File("1.jpg"));
byte[] buffer = new byte[1024];
int len;
while((len=inputStream.read(buffer))!=-1){
fos.write(buffer, 0, len);
}
fos.close();
inputStream.close();
urlConnection.disconnect();
}
}


浙公网安备 33010602011771号