网络编程
网络编程概述
网络模型
OSI参考模型
TCP/IP参考模型网络通讯要素
IP地址
端口号
传输协议
网络参考模型
网络通讯要素
IP地址:InetAddress
网络中设备的标识
不易记忆,可用主机名
本地回环地址:127.0.0.1 主机名:localhost1
2
3InetAddress ia = InetAddress.getByName("www.baidu.com");
System.out.println("address : " + ia.getHostAddress());
System.out.println("name : " + ia.getHostName());端口号
用于标识进程的逻辑地址,不同进程的标识
有效端口:0~65535,其中0~1024系统使用或保留端口传输协议
通讯的规则
常见协议:TCP,UDP
TCP和UDP
UDP
将数据及源和目的封装成数据包中,不需要建立连接
每个数据报的大小在限制在64k内
因无连接,是不可靠协议
不需要建立连接,速度快TCP
建立连接,形成传输数据的通道
在连接中进行大数据量传输
通过三次握手完成连接,是可靠协议
必须建立连接,效率会稍低
Socket
- Socket就是为网络服务提供的一种机制
- 通信的两端都有Socket
- 网络通信其实就是Socket间的通信
- 数据在两个Socket间通过IO传输
UDP传输
发送端与接收端是两个独立的运行程序
- DatagramSocket与DatagramPacket
- 建立发送端,接收端
- 建立数据包
- 调用Socket的发送接收方法
- 关闭Socket
发送端
在发送端,要在数据包对象中明确目的地 IP 及端口1
2
3
4
5DatagramSocket ds = new DatagramSocket();
byte[] by = "hello,udp".getBytes();
DatagramPacket dp = new DatagramPacket(by, 0, by.length, InetAddress.getByName("127.0.0.1"), 10000);
ds.send(dp);
ds.close();
接收端
在接收端,要指定监听的端口1
2
3
4
5
6
7DatagramSocket ds = new DatagramSocket(10000);
byte[] by = new byte[1024];
DatagramPacket dp = new DatagramPacket(by, by.length);
ds.receive(dp);
String str = new String(dp.getData(), 0, dp.getLength());
System.out.println(str + "---" + dp.getAddress());
ds.close();
基于UDP的聊天程序
1 | import java.io.*; |
TCP传输
- Socket和ServerSocket
- 建立客户端和服务器端
- 建立连接后,通过Socket中的IO流进行数 据的传输
- 关闭socket同样,客户端与服务器端是两个独立的应用程序
基本思路(客户端)
- 客户端需要明确服务器的ip地址以及端口,这样才可以去试着建立连接,如果连接失败,会出现异常
- 连接成功,说明客户端与服务端建立了通道,那么通过IO流就可以进行数据的传输,而Socket对象已经提供了输入流和输出流对象,通过getInputStream(),getOutputStream()获取即可
- 与服务端通讯结束后,关闭Socket
客户端
通过Socket建立对象并指定要连接的服务端主机以及端口1
2
3
4Socket s = new Socket(“192.168.1.1”, 9999);
OutputStream out = s.getOutputStream();
out.write("hello".getBytes());
s.close();
基本思路(服务端)
- 服务端需要明确它要处理的数据是从哪个端口进入的
- 当有客户端访问时,要明确是哪个客户端,可通过accept()获取已连接的客户端对象,并通过该对象与客户端通过IO流进行数据传输
- 当该客户端访问结束,关闭该客户端
服务端
建立服务端需要监听一个端口1
2
3
4
5
6
7
8
9ServerSocket ss = new ServerSocket(9999);
Socket s = ss.accept ();
InputStream in = s.getInputStream();
byte[] buf = new byte[1024];
int num = in.read(buf);
String str = new String(buf,0,num);
System.out.println(s.getInetAddress().toString() + ":" + str);
s.close();
ss.close();
基于TCP的发送返回示例
1 | import java.net.*; |
Tcp传输最容易出现的问题
- 客户端连接上服务端,两端都在等待,没有任何数据传输
- 通过例程分析:
因为read方法或者readLine方法是阻塞式 - 解决办法:
自定义结束标记
使用shutdownInput,shutdownOutput方法
1 | import java.io.*; |
TCP的并发
当A客户端连接上以后。被服务端获取到。服务端执行具体流程,这时B客户端连接,只有等待
因为服务端还没有处理完A客户端的请求,还有循环回来执行下次accept方法。所以暂时获取不到B客户端对象
那么为了可以让多个客户端同时并发访问服务端,服务端最好就是将每个客户端封装到一个单独的线程中,这样,就可以同时处理多个客户端请求
示例:并发上传文件1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85import java.io.*;
import java.net.*;
class PicClient {
public static void main(String[] args) throws Exception {
if(args.length != 1) {
System.out.println("请选择一个jpg格式的图片");
return ;
}
File file = new File(args[0]);
if(!(file.exists() && file.isFile())) {
System.out.println("该文件有问题,要么不存在,要么不是文件");
return ;
}
if(!file.getName().endsWith(".jpg")) {
System.out.println("图片格式错误,请重新选择");
return ;
}
if(file.length() > 1024 * 1024 * 5) {
System.out.println("文件过大,请重新选择");
return ;
}
Socket s = new Socket("192.168.1.2",10007);
FileInputStream fis = new FileInputStream(file);
OutputStream out = s.getOutputStream();
byte[] buf = new byte[1024];
int len = 0;
while((len = fis.read(buf)) != -1) {
out.write(buf,0,len);
}
//告诉服务端数据已写完
s.shutdownOutput();
InputStream in = s.getInputStream();
byte[] bufIn = new byte[1024];
int num = in.read(bufIn);
System.out.println(new String(bufIn, 0, num));
fis.close();
s.close();
}
}
class PicThread implements Runnable {
private Socket s;
PicThread(Socket s) {
this.s = s;
}
public void run() {
int count = 1;
String ip = s.getInetAddress().getHostAddress();
try {
System.out.println(ip + "···connected");
InputStream in = s.getInputStream();
File dir = new File("d:\\pic");
File file = new File(dir, ip + "(" + (count) + ")" + ".jpg");
while(file.exists())
file = new File(dir, ip + "(" + (count++) + ")" + ".jpg");
FileOutputStream fos = new FileOutputStream(file);
byte[] buf = new byte[1024];
int len = 0;
while((len = in.read(buf)) != -1) {
fos.write(buf,0,len);
}
OutputStream out = s.getOutputStream();
out.write("上传成功".getBytes());
fos.close();
s.close();
} catch (Exception e) {
throw new RuntimeException(ip+"上传失败");
}
}
}
class PicServer {
public static void main(String[] args) throws Exception {
ServerSocket ss = new ServerSocket(10007);
while(true) {
Socket s = ss.accept();
new Thread(new PicThread(s)).start();
}
}
}
URI
Uniform Resource Identifies(统一资源定位符)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19import java.net.*;
class Test {
public static void main(String[] args) throws MalformedURLException {
URL url = new URL("http://192.168.1.2/web/index.html?name=cj&age=20");
System.out.println("getProtocol() :" + url.getProtocol());
System.out.println("getHost() :" + url.getHost());
System.out.println("getPort() :" + url.getPort());
System.out.println("getPath() :" + url.getPath());
System.out.println("getFile() :" + url.getFile());
System.out.println("getQuery() :" + url.getQuery());
URLConnection conn = url.openConnection();
System.out.println(conn);
InputStream in = conn.getInputStream();
byte[] buf = new byte[1024];
int len = in.read(buf);
System.out.println(new String(buf, 0, len));
}
}