Socket 一些事
1. 网络基础
1.1 网络分层
计算机网络分为五层:物理层、数据链路层、网络层、运输层、应用层。
其中:
网络层:负责根据IP找到目的地址的主机
运输层:通过端口把数据传到目的主机的目的进程,来实现进程与进程之间的通信
1.2 端口号(PORT)
端口号规定为16位,即允许一个IP主机有 2 的 16 次方 65535 个不同的端口。其中:
- 0~1023:分配给系统的端口号。我们不可以乱用
- 1024~49151:登记端口号。主要是让第三方应用使用,但是必须在 IANA (互联网数字分配机构)按照规定手续登记,
- 49152~65535:短暂端口号。是留给客户进程选择暂时使用,一个进程使用完就可以供其他进程使用。
在Socket使用时,可以用1024~65535的端口号
1.3 C/S结构
即客户端/服务器结构,是软件系统体系结构
作用:充分利用两端硬件环境的优势,将任务合理分配到Client端和Server端来实现,降低了系统的通讯开销。
Socket 正是使用这种结构建立连接的,一个套接字接客户端,一个套接字接服务器。
基于TCP或UDP协议的 Socket 使用.png1.4 TCP协议
Transmission Control Protocol,即传输控制协议,是一种传输层通信协议
基于 TCP 的应用层协议有 FTP、Telnet、SMTP、HTTP、POP3 与 DNS。
1.4.1 特点:面向连接、面向字节流、全双工通信、可靠。
面向连接:指的是要使用TCP传输数据,必须先建立TCP连接,传输完成后释放连接,就像打电话一样必须先拨号建立一条连接,打完后挂机释放连接。
全双工通信:即一旦建立了TCP连接,通信双方可以在任何时候都能发送数据。
可靠的:指的是通过TCP连接传送的数据,无差错,不丢失,不重复,并且按序到达。
面向字节流:流,指的是流入到进程或从进程流出的字符序列。简单来说,虽然有时候要传输的数据流太大,TCP报文长度有限制,不能一次传输完,要把它分为好几个数据块,但是由于可靠性保证,接收方可以按顺序接收数据块然后重新组成分块之前的数据流,所以TCP看起来就像直接互相传输字节流一样,面向字节流。
1.4.2 TCP建立连接
必须进行三次握手:若A要与B进行连接,则必须
第一次握手:建立连接。客户端发送连接请求报文段,将 SYN 位置为 1,Sequence Number 为 x;然后,客户端进入 SYN_SEND 状态,等待服务器的确认。即 A 发送信息给 B
第二次握手:服务器收到客户端的 SYN 报文段,需要对这个 SYN 报文段进行确认。即 B 收到连接信息后向 A 返回确认信息
第三次握手:客户端收到服务器的 (SYN+ACK) 报文段,并向服务器发送 ACK 报文段。即 A 收到确认信息后再次向 B 返回确认连接信息
此时,A告诉自己上层连接建立;B收到连接信息后告诉上层连接建立。
TCP 三次握手过程.png
TCP三次握手 = 一条TCP连接建立完成 = 可以开始发送数据。
三次握手期间任何一次未收到对面回复都要重发。
最后一个确认报文段发送完毕以后,客户端和服务器端都进入 ESTABLISHED 状态。
1.4.3 为什么 TCP 建立连接需要三次握手?
答:防止服务器端因为接收了早已失效的连接请求报文从而一直等待客户端请求,从而浪费资源
“早已失效的连接请求报文段”的产生在这样一种情况下:Client 发出的第一个连接请求报文段并没有丢失,而是在某个网络结点长时间的滞留了,以致延误到连接释放以后的某个时间才到达 server ,但 Server 收到此失效的连接请求报文段后,就误认为是 Client 再次发出的一个新的连接请求。于是就向 Client 发出确认报文段,同意建立连接。
假设不采用“三次握手”:只要 Server 发出确认,新的连接就建立了。由于现在 Client 并没有发出建立连接的请求,因此不会向 Server 发送数据。但 Server 却以为新的运输连接已经建立,并一直等待 Client 发来数据。这样,Server 的资源就白白浪费掉了。
采用“三次握手”的办法可以防止上述现象发生:
Client 不会向 Server 的确认发出确认, Server 由于收不到确认,就知道 Client 并没有要求建立连接,所以Server不会等待Client发送数据,资源就没有被浪费。
1.4.4 TCP 释放连接
TCP释放连接需要四次挥手过程,现在假设 A 主动释放连接:(数据传输结束后,通信的双方都可释放连接)
第一次挥手:A 发送释放信息到 B ;(发出去之后,A->B发送数据这条路径就断了)
第二次挥手:B 收到 A 的释放信息之后,回复确认释放的信息:我同意你的释放连接请求
第三次挥手:B 发送“请求释放连接“信息给 A
第四次挥手:A 收到 B 发送的信息后向 B 发送确认释放信息:我同意你的释放连接请求
B 收到确认信息后就会正式关闭连接;
A等待 2MSL 后依然没有收到回复,则证明 B 端已正常关闭,于是 A 关闭连接。
TCP 四次挥手过程
1.4.5 为什么 TCP 释放连接需要四次挥手?
答:为了保证双方都能通知对方“需要释放连接”,即在释放连接后都无法接收或发送消息给对方
需要明确的是:TCP 是全双工模式,这意味着是双向都可以发送、接收的
释放连接的定义是:双方都无法接收或发送消息给对方,是双向的
当主机 1 发出“释放连接请求”( FIN 报文段)时,只是表示主机 1 已经没有数据要发送 / 数据已经全部发送完毕。但是,这个时候主机 1 还是可以接受来自主机 2 的数据。
当主机 2 返回“确认释放连接”信息( ACK 报文段)时,表示它已经知道主机 1 没有数据发送了,但此时主机2还是可以发送数据给主机1。
当主机 2 也发送了 FIN 报文段时,即告诉主机 1 我也没有数据要发送了。
此时,主机 1 和 2 已经无法进行通信:主机1无法发送数据给主机 2 ,主机 2 也无法发送数据给主机 1 ,此时,TCP 的连接才算释放。
1.5 UDP 协议
定义:User Datagram Protocol,即用户数据报协议,是一种传输层通信协议。
基于UDP的应用层协议有TFTP、SNMP与DNS。
1.5.1 特点:无连接的、不可靠的、面向报文、没有拥塞控制
无连接的:和 TCP 要建立连接不同,UDP 传输数据不需要建立连接,就像写信,在信封写上收信人名称、地址就可以交给邮局发送了,至于能不能送到,就要看邮局的送信能力和送信过程的困难程度了。
不可靠的:因为 UDP 发出去的数据包发出去就不管了,不管它会不会到达,所以很可能会出现丢包现象,使传输的数据出错。
面向报文:数据报文,就相当于一个数据包,应用层交给 UDP 多大的数据包,UDP 就照样发送,不会像 TCP 那样拆分。
没有拥塞控制:拥塞,是指到达通信子网中某一部分的分组数量过多,使得该部分网络来不及处理,以致引起这部分乃至整个网络性能下降的现象,严重时甚至会导致网络通信业务陷入停顿,即出现死锁现象,就像交通堵塞一样。TCP 建立连接后如果发送的数据因为信道质量的原因不能到达目的地,它会不断重发,有可能导致越来越塞,所以需要一个复杂的原理来控制拥塞。而 UDP 就没有这个烦恼,发出去就不管了。
1.5.2 应用场景
很多的实时应用(如 IP 电话、实时视频会议、某些多人同时在线游戏等)要求源主机以很定的速率发送数据,并且允许在网络发生拥塞时候丢失一些数据,但是要求不能有太大的延时,UDP 就刚好适合这种要求。所以说,只有不适合的技术,没有真正没用的技术。
2. Socket
2.1 Socket 定义
Socket 定义.pngSocket 是一个对 TCP / IP 协议进行封装的编程调用接口(API)。
即通过 Socket,我们才能在 Andorid 平台上通过 TCP/IP 协议进行开发,Socket 不是一种协议,而是一个编程调用接口(API),属于传输层(主要解决数据如何在网络中传输)。
Socket 成对出现,一对套接字:
Socket ={(IP地址1:PORT端口号),(IP地址2:PORT端口号)}
2.2 Socket 原理
Socket 的使用类型主要有两种:
- 流套接字(streamsocket) :基于 TCP 协议,采用 流的方式 提供可靠的字节流服务
- 数据报套接字(datagramsocket):基于 UDP 协议,采用 数据报文 提供数据打包发送的服务
具体原理图如下:
原理图2.3 Socket 与 Http 对比
- Socket 属于传输层,因为 TCP / IP 协议属于传输层,解决的是数据如何在网络中传输的问题 。
- HTTP 协议 属于 应用层,解决的是如何包装数据 。
由于二者不属于同一层面,所以本来是没有可比性的。但随着发展,默认的 Http 里封装了下面几层的使用,所以才会出现 Socket & HTTP 协议的对比:(主要是工作方式的不同):
-
Http:采用 请求—响应 方式。
即建立网络连接后,当 客户端 向 服务器 发送请求后,服务器端才能向客户端返回数据。
可理解为:是客户端有需要才进行通信 -
Socket:采用 服务器主动发送数据 的方式
即建立网络连接后,服务器可主动发送消息给客户端,而不需要由客户端向服务器发送请求。
可理解为:是服务器端有需要才进行通信
2.4 Socket 使用步骤
Socket 通信模型.pngSocket 可基于 TCP 或者 UDP 协议,但 TCP 更加常用。所以下面的使用步骤 & 实例的 Socket 将基于 TCP 协议:
步骤1:创建客户端 & 服务器的连接
// 创建Socket对象 & 指定服务端的IP及端口号
Socket socket = new Socket("192.168.1.172", 8989);
// 判断客户端和服务器是否连接成功
socket.isConnected());
步骤2:客户端 & 服务器 通信
通信包括:客户端 接收服务器的数据 & 发送数据 到 服务器
<-- 操作1:接收服务器的数据 -->
// 步骤1:创建输入流对象 InputStream
InputStream is = socket.getInputStream()
// 步骤2:创建输入流读取器对象并传入输入流对象
// 该对象作用:获取服务器返回的数据
InputStreamReader isr = new InputStreamReader(is);
BufferedReader br = new BufferedReader(isr);
// 步骤3:通过输入流读取器对象接收服务器发送过来的数据
br.readLine();
<-- 操作2:发送数据 到 服务器 -->
// 步骤1:从 Socket 获得输出流对象 OutputStream
// 该对象作用:发送数据
OutputStream outputStream = socket.getOutputStream();
// 步骤2:写入需要发送的数据到输出流对象中
// 特别注意:数据的结尾加上换行符才可让服务器端的readline()停止阻塞
outputStream.write((mEdit.getText().toString()+"\n").getBytes("utf-8"));
// 步骤3:发送数据到服务端
outputStream.flush();
步骤3:断开客户端 & 服务器 连接
// 断开 客户端发送到服务器 的连接,即关闭输出流对象 OutputStream
os.close();
// 断开 服务器发送到客户端 的连接,即关闭输入流读取器对象 BufferedReader
br.close();
// 最终关闭整个Socket连接
socket.close();
2.5 简单示例
在 Eclipse 下创建一个Java项目,代码:
public class SocketServer {
public static void main(String[] args) throws IOException {
//1.创建一个服务器端Socket,即ServerSocket,指定绑定的端口,并监听此端口
ServerSocket serverSocket = new ServerSocket(8989);
InetAddress address = InetAddress.getLocalHost();
String ip = address.getHostAddress();
Socket socket = null;
//2.调用accept()等待客户端连接
System.out.println("~~~服务端已就绪,等待客户端接入~~~,服务端ip地址: " + ip);
socket = serverSocket.accept();
//3.连接后获取输入流,读取客户端信息
InputStream is=null;
InputStreamReader isr=null;
BufferedReader br=null;
OutputStream os=null;
PrintWriter pw=null;
is = socket.getInputStream(); //获取输入流
isr = new InputStreamReader(is,"UTF-8");
br = new BufferedReader(isr);
os = socket.getOutputStream();
os.write(("123456" + "\n").getBytes("utf-8"));
String info = null;
while((info=br.readLine())!=null){//循环读取客户端的信息
System.out.println("客户端发送过来的信息:" + info);
}
socket.shutdownInput();//关闭输入流
socket.close();
}
}
在 Android Studio 下创建一个项目,代码:
public class MainActivity extends Activity {
/**
* 主线程Handler
* 用于将从服务器获取的消息显示出来
*/
private Handler mMainHandler;
/**
* Socket变量
*/
private Socket socket;
/**
* 线程池
* 为了方便展示,此处直接采用线程池进行线程管理,而没有一个个开线程
*/
private ExecutorService mThreadPool;
/**
* 接收服务器消息 变量
* 输入流对象
*/
InputStream is;
/**
* 接收服务器消息 变量
* 输入流读取器对象
*/
InputStreamReader isr;
BufferedReader br;
/**
* 接收服务器消息 变量
* 接收服务器发送过来的消息
*/
String response;
/**
* 发送消息到服务器 变量
* 输出流对象
*/
OutputStream outputStream;
private Button btnConnect, btnDisconnect, btnReceive, btnSend;
private TextView mReceive;
private EditText mEdit;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btnConnect = (Button) findViewById(R.id.btn_connect);
btnDisconnect = (Button) findViewById(R.id.btn_disconnect);
btnSend = (Button) findViewById(R.id.btn_send);
btnReceive = (Button) findViewById(R.id.btn_receive);
mEdit = (EditText) findViewById(R.id.edit_send);
mReceive = (TextView) findViewById(R.id.tv_receive);
mThreadPool = Executors.newCachedThreadPool();
// 实例化主线程,用于更新接收过来的消息
mMainHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case 0:
mReceive.setText(response);
break;
default:
}
}
};
//创建客户端和服务器的连接
btnConnect.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 利用线程池直接开启一个线程 & 执行该线程
mThreadPool.execute(new Runnable() {
@Override
public void run() {
try {
// 创建Socket对象 & 指定服务端的IP 及 端口号
socket = new Socket("192.168.1.113", 8989);
// 判断客户端和服务器是否连接成功
System.out.println(socket.isConnected());
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
});
//接收服务器消息
btnReceive.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 利用线程池直接开启一个线程和执行该线程
mThreadPool.execute(new Runnable() {
@Override
public void run() {
try {
// 步骤1:创建输入流对象
is = socket.getInputStream();
// 步骤2:创建输入流读取器对象 并传入输入流对象
// 该对象作用:获取服务器返回的数据
isr = new InputStreamReader(is);
br = new BufferedReader(isr);
// 步骤3:通过输入流读取器对象 接收服务器发送过来的数据
response = br.readLine();
// 步骤4:通知主线程,将接收的消息显示到界面
Message msg = Message.obtain();
msg.what = 0;
mMainHandler.sendMessage(msg);
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
});
//发送消息给服务器
btnSend.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 利用线程池直接开启一个线程和执行该线程
mThreadPool.execute(new Runnable() {
@Override
public void run() {
try {
// 步骤1:从Socket 获得输出流对象OutputStream
// 该对象作用:发送数据
outputStream = socket.getOutputStream();
// 步骤2:写入需要发送的数据到输出流对象中
outputStream.write((mEdit.getText().toString() + "\n").getBytes("utf-8"));
// 特别注意:数据的结尾加上换行符才可让服务器端的readline()停止阻塞
// 步骤3:发送数据到服务端
outputStream.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
});
//断开客户端和服务器的连接
btnDisconnect.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
try {
// 断开 客户端发送到服务器 的连接,即关闭输出流对象OutputStream
outputStream.close();
// 断开 服务器发送到客户端 的连接,即关闭输入流读取器对象
br.close();
// 最终关闭整个Socket连接
socket.close();
// 判断客户端和服务器是否已经断开连接
System.out.println(socket.isConnected());
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
}