软件开发架构

  涉及到两个程序之间通讯的应用大致可以分为两种:

  第一种是应用类:qq、微信、网盘、优酷这一类是属于需要安装的桌面应用

  第二种是web类:比如百度、知乎、博客园等使用浏览器访问就可以直接使用的应用

  这些应用的本质其实都是两个程序之间的通讯。而这两个分类又对应了两个软件开发的架构~

C/S架构

  C/S即:Client与Server ,中文意思:客户端与服务器端架构,这种架构也是从用户层面(也可以是物理层面)来划分的。

  服务端定义: 是  要一直运行  等待服务别人

  客户端定义: 是  我用的时候  我才使用服务

  这里的客户端一般泛指客户端应用程序EXE,程序需要先安装后,才能运行在用户的电脑上,对用户的电脑操作系统环境依赖较大。

python客户端和服务端通信 python客户端和服务端区别_IP

B/S架构

  B/S即:Browser与Server,中文意思:浏览器端与服务器端架构,这种架构是从用户层面来划分的。

  B/S火的原因————统一入口

  Browser浏览器,其实也是一种Client客户端,只是这个客户端不需要大家去安装什么应用程序,只需在浏览器上通过HTTP请求服务器端相关的资源(网页资源),客户端Browser浏览器就能进行增删改查。

python客户端和服务端通信 python客户端和服务端区别_python客户端和服务端通信_02

  一个程序如何在网络上找到另一个程序?

  首先,程序必须要启动,其次,必须有这台机器的地址,我们都知道我们人的地址大概就是国家\省\市\区\街道\楼\门牌号这样字。那么每一台联网的机器在网络上也有自己的地址,它的地址是怎么表示的呢?

  就是使用一串数字来表示的,例如:192.168.100.166

python客户端和服务端通信 python客户端和服务端区别_IP_03

python客户端和服务端通信 python客户端和服务端区别_客户端_04

IP地址是指互联网协议地址(英语:Internet Protocol Address,又译为网际协议地址),是IP Address的缩写。IP地址是IP协议提供的一种统一的地址格式,它为互联网上的每一个网络和每一台主机分配一个逻辑地址,以此来屏蔽物理地址的差异。

IP地址是一个32位的二进制数,通常被分割为4个“8位二进制数”(也就是4个字节)。IP地址通常用“点分十进制”表示成(a.b.c.d)的形式,其中,a,b,c,d都是0~255之间的十进制整数。例:点分十进IP地址(100.4.5.6),实际上是32位二进制数(01100100.00000100.00000101.00000110)。

ip地址定义

python客户端和服务端通信 python客户端和服务端区别_IP_03

python客户端和服务端通信 python客户端和服务端区别_客户端_04

"端口"是英文port的意译,可以认为是设备与外界通讯交流的出口。

端口

  因此ip地址精确到具体的一台电脑,而端口精确到具体的程序。

  osi七层模型

  须知一个完整的计算机系统是由硬件、操作系统、应用软件三者组成,具备了这三个条件,一台计算机系统就可以自己跟自己玩了(打个单机游戏,玩个扫雷啥的)

  如果你要跟别人一起玩,那你就需要上网了,什么是互联网?

  互联网的核心就是由一堆协议组成,协议就是标准,比如全世界人通信的标准是英语,如果把计算机比作人,互联网协议就是计算机界的英语。所有的计算机都学会了互联网协议,那所有的计算机都就可以按照统一的标准去收发信息从而完成通信了。

  osi七层模型

  人们按照分工不同把互联网协议从逻辑上划分了层级:

python客户端和服务端通信 python客户端和服务端区别_python客户端和服务端通信_07

socket概念

  socket层

python客户端和服务端通信 python客户端和服务端区别_IP_08

  理解socket

TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。

python客户端和服务端通信 python客户端和服务端区别_IP_03

python客户端和服务端通信 python客户端和服务端区别_客户端_04

其实站在你的角度上看,socket就是一个模块。我们通过调用模块中已经实现的方法建立两个进程之间的连接和通信。
也有人将socket说成ip+port,因为ip是用来标识互联网中的一台主机的位置,而port是用来标识这台机器上的一个应用程序。
所以我们只要确立了ip和port就能找到一个应用程序,并且使用socket模块来与之通信。

站在你的角度上看socket

  套接字(socket)的发展史

  套接字起源于 20 世纪 70 年代加利福尼亚大学伯克利分校版本的 Unix,即人们所说的 BSD Unix。 因此,有时人们也把套接字称为“伯克利套接字”或“BSD 套接字”。一开始,套接字被设计用在同 一台主机上多个应用程序之间的通讯。这也被称进程间通讯,或 IPC。套接字有两种(或者称为有两个种族),分别是基于文件型的和基于网络型的。 

  基于文件类型的套接字家族

  套接字家族的名字:AF_UNIX

  unix一切皆文件,基于文件的套接字调用的就是底层的文件系统来取数据,两个套接字进程运行在同一机器,可以通过访问同一个文件系统间接完成通信

  基于网络类型的套接字家族

  套接字家族的名字:AF_INET

  (还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现,所有地址家族中,AF_INET是使用最广泛的一个,python支持很多种地址家族,但是由于我们只关心网络编程,所以大部分时候我么只使用AF_INET)

  tcp协议和udp协议

  TCP(Transmission Control Protocol)可靠的、面向连接的协议(eg:打电话)、传输效率低全双工通信(发送缓存&接收缓存)、面向字节流。使用TCP的应用:Web浏览器;电子邮件、文件传输程序。

  UDP(User Datagram Protocol)不可靠的、无连接的服务,传输效率高(发送前时延小),一对一、一对多、多对一、多对多、面向报文,尽最大努力服务,无拥塞控制。使用UDP的应用:域名系统 (DNS);视频流;IP语音(VoIP)。

 

python客户端和服务端通信 python客户端和服务端区别_客户端_11

套接字(socket)初使用

 基于TCP协议的socket

  tcp是基于链接的,必须先启动服务端,然后再启动客户端去链接服务端

  server端

1 import socket
 2 sk = socket.socket()
 3 sk.bind(('127.0.0.1',8898))  #把地址绑定到套接字
 4 sk.listen()          #监听链接
 5 conn,addr = sk.accept() #接受客户端链接
 6 ret = conn.recv(1024)  #接收客户端信息
 7 print(ret)       #打印客户端信息
 8 conn.send(b'hi')        #向客户端发送信息
 9 conn.close()       #关闭客户端套接字
10 sk.close()        #关闭服务器套接字(可选)

  client端

1 import socket
2 sk = socket.socket()           # 创建客户套接字
3 sk.connect(('127.0.0.1',8898))    # 尝试连接服务器
4 sk.send(b'hello!')
5 ret = sk.recv(1024)         # 对话(发送/接收)
6 print(ret)
7 sk.close()            # 关闭客户套接字

  问题:有的同学在重启服务端时可能会遇到

python客户端和服务端通信 python客户端和服务端区别_套接字_12

  解决方法:

#加入一条socket配置,重用ip和端口
import socket
from socket import SOL_SOCKET,SO_REUSEADDR
sk = socket.socket()
sk.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加
sk.bind(('127.0.0.1',8898))  #把地址绑定到套接字
sk.listen()          #监听链接
conn,addr = sk.accept() #接受客户端链接
ret = conn.recv(1024)   #接收客户端信息
print(ret)              #打印客户端信息
conn.send(b'hi')        #向客户端发送信息
conn.close()       #关闭客户端套接字
sk.close()        #关闭服务器套接字(可选)

 模拟与服务器聊天

1 # servery
 2 
 3 import socket
 4 sk = socket.socket()
 5 sk.bind(('127.0.0.1', 8080))
 6 sk.listen()
 7 conn, adds = sk.accept()
 8 while 1:
 9     ret = conn.recv(1024).decode('utf-8')
10     print('client:'+ret)
11     if ret == 'bye':
12         conn.send(b'bye')
13         break
14     info = input('server:>>>').encode('utf-8')
15     conn.send(info)
16 conn.close()
17 sk.close()
18 
19 
20 # client
21 
22 import socket
23 sk = socket.socket()
24 sk.connect(('127.0.0.1', 8080))
25 while 1:
26     ret = input('client>>>').encode('utf-8')
27     sk.send(ret)
28     ret = sk.recv(1024).decode('utf-8')
29     print('server:' + ret)
30     if ret == 'bye':
31         sk.send(b'bye')
32         break
33 sk.close()

服务端与客户端一对多聊天

1 # server
 2 import socket
 3 sk = socket.socket()
 4 sk.bind(('127.0.0.1',8080))
 5 sk.listen()
 6 while 1:
 7     conn, addr = sk.accept()
 8     while 1:
 9         ret = conn.recv(1024).decode('utf-8')
10         print(ret)
11         if ret == 'bye':
12             conn.send(b'bye')
13             break
14         info = input('admin>>>').encode('utf-8')
15         conn.send(info)
16     conn.close()
17 conn.close()
18 sk.close
19 
20 
21 # client
22 import socket
23 sk = socket.socket()
24 sk.connect(('127.0.0.1', 8080))
25 while 1:
26     info = input('>>>').encode('utf-8')
27     sk.send(info)
28     ret = sk.recv(1024).decode('utf-8')
29     print(ret)
30     if ret == 'bye':
31         sk.send(b'bye')
32         break
33 sk.close()
34 
35 # client2
36 import socket
37 sk = socket.socket()
38 sk.connect(('127.0.0.1', 8080))
39 while 1:
40     info = input('2>>>').encode('utf-8')
41     sk.send(info)
42     ret = sk.recv(1024).decode('utf-8')
43     print(ret)
44     if ret == 'bye':
45         sk.send(b'bye')
46         break
47 sk.close()