socket()
本地的进程间通信(IPC)有很多种方式,但可以总结为下面4类:
-
消息传递(管道、FIFO、消息队列)
-
同步(互斥量、条件变量、读写锁、文件和写记录锁、信号量)
-
共享内存(匿名的和具名的)
-
远程过程调用(Solaris门和Sun RPC)
不过这个不是重点!提问,在网络中进程之间如何通信?
首要解决的问题是如何唯一标识一个进程,否则通信无从谈起!在本地可以通过进程PID来唯一标识一个进程,但是在网络中这是行不通的。其实TCP/IP协议族已经帮我们解决了这个问题,网络层的“ip地址”可以唯一标识网络中的主机,而传输层的“协议+端口”可以唯一标识主机中的应用程序(进程)。这样利用三元组(ip地址,协议,端口)就可以标识网络的进程了,网络中的进程通信就可以利用这个标志与其它进程进行交互。
使用TCP/IP协议的应用程序通常采用应用编程接口:UNIX BSD的套接字(socket)和UNIX System V的TLI(已经被淘汰),来实现网络进程之间的通信。就目前而言,几乎所有的应用程序都是采用socket,而现在又是网络时代,网络中进程通信是无处不在,这就是我为什么说“一切皆socket”。
所以,在下面的理解中,我们可以把socket 简单的理解为“ip + 协议 + 端口”
其中ip就是ip地址,协议就是TCP+UDP,端口范围是1-65535
需要说明的是socket和file有类似的地方,这是因为,在发展的初期他们都是按UNIX的思路创造出来的,“一切皆文件”,所以对于socket也有 读/写IO,打开,关闭这样的操作。
socket和file的区别:
- file模块是针对某个指定文件进行【打开】【读写】【关闭】
- socket模块是针对 服务器端 和 客户端Socket 进行【打开】【读写】【关闭】
实现socket通常分成两个部分
client | server |
socket() | socket() |
bind() | |
listen() | |
connect() | accept() |
write() | |
read() | |
write() | |
read() | |
close() | |
close() |
上图片:
知道了沟通过程,剩下就是语法,然后就可以开始进行socket的编写了。
sk = socket.socket(socket.AF_INET,socket.SOCK_STREAM,0)
参数一:地址簇
socket.AF_INET IPv4(默认)
socket.AF_INET6 IPv6socket.AF_UNIX 只能够用于单一的Unix系统进程间通信
参数二:类型
socket.SOCK_STREAM 流式socket , for TCP (默认)
socket.SOCK_DGRAM 数据报式socket , for UDPsocket.SOCK_RAW 原始套接字,普通的套接字无法处理ICMP、IGMP等网络报文,而SOCK_RAW可以;其次,SOCK_RAW也可以处理特殊的IPv4报文;此外,利用原始套接字,可以通过IP_HDRINCL套接字选项由用户构造IP头。
socket.SOCK_RDM 是一种可靠的UDP形式,即保证交付数据报但不保证顺序。SOCK_RAM用来提供对原始协议的低级访问,在需要执行某些特殊操作时使用,如发送ICMP报文。SOCK_RAM通常仅限于高级用户或管理员运行的程序使用。
socket.SOCK_SEQPACKET 可靠的连续数据包服务参数三:协议
0 (默认)与特定的地址家族相关的协议,如果是 0 ,则系统就会根据地址格式和套接类别,自动选择一个合适的协议
#!/usr/bin/env python # -- coding: utf-8 -- __author__ = 'EchoRep' ‘’‘ 创建服务端 ’‘’ import socket def main(): # 创建socket对象 sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 监听端口 sock.bind(('127.0.0.1',8001)) # 开始监听, sock.listen(5) while True: # 阻塞,deng 。。。。 # 直到有请求连接 print '....' connection, address = sock.accept() # connection,代表客户端socket对象, # address,客户端IP地址 #handle_request(connection) buf = connection.recv(1024) print buf #通过链接接受数据 connection.send("HTTP/1.1 200 OK ") connection.send("Hello, World") connection.close() if __name__ == '__main__': main()
#!/usr/bin/env python # -- coding: utf-8 -- __author__ = 'EchoRep' ‘’‘ 创建客户端 ’‘’ import socket while True: objClient = socket.socket() objClient.connect(("localhost",8001)) objClient.send("hihihi") serverDate = objClient.recv(1024) print serverDate objClient.close()