简介:

        百度百科上介绍说:Socket原意是 “插座”,可以看成是在两个程序进行通讯连接中的端点,是连接应用程序和网络驱动程序的桥梁。Socket在应用程序中创建,通过绑定与网络驱动建立关系。套接字,是支持TCP/IP的网络通信的基本操作单元,可以看做是不同主机之间的进程进行双向通信的端点,简单的说就是通信的两方的一种约定,用套接字中的相关函数来完成通信过程。

        个人的理解:假设有一个有两个门的房间,这两个门都有各自的钥匙,且一个是面向服务器(server)的,另一个是面向客户(client)的,当服务器与客户分别拿到这两把钥匙时,二者就可以通过这个房间交换信息了。这对钥匙就相当于是一对socket,网络通路相当于是房间,但因为网络通路并不是只有一条,相当于存在无数个这样的房子,所以就有无数对这样的钥匙,这时每个socket应该有些信息来标注自己的身份,不然分不清了,与人们一般选择不同形状的钥匙一个道理。具体实现中socket使用了三种手段一起来为自己标注:一种是网络传输协议的种类(TCP or UDP),然后是对方的IP地址,最后是对方(客户端)或自己(服务器端)的端口号Port. 一旦服务器端与客户端的socket能配对上,则二者就可以通信了。

        这里简单说下TCP与UDP。TCP即传输控制协议(Transmission Control Protocol),是一种流套接字,是面向连接的,在通信之前,要建立并维持一条连接,直到通信结束,这种方式叫做“虚电路”,是一种可靠数据传输方式,通过握手机制保证数据传输的准确性。UDP即用户数据报协议(User Datagram Protocol),不需要在通信前提前建立一条连接,而在确定了合适大小的报文大小后,直接把数据放到网上,通过路由控制协议向目的地传输,不保证数据传输的正确性与顺序,速度比TCP快好多,广泛应用于视频音频等即时通信服务中。

        Python中使用socket模块进行client/server 网络编程。基于网络的socket有一个自己的address family name: AF_INET. socket模块可以选择TCP/UDP,目的地地址与端口,并有相关数据传输与接收函数。

实例:

由于是在同一台电脑上运行,所以开了两个Python: Python3.3(for server), Python3.2(for client)

TCP-服务器端:

from socket import*
from time import ctime

HOST='' #空表示可以接收多有有效ip地址的连接请求,bind()函数可以绑定在所有有效的ip地址上
PORT=8080 #随机选的一个端口号,必须大于1024
BUFSIZ=1024 #每次读取的数据量
ADDR=(HOST,PORT) #HOST与PORT共同组成了ADDR

tcpSerSock=socket(AF_INET, SOCK_STREAM) #创建了一个基于网络的、使用TCP的socket。AF_INET表示是基于网络的socket编程,SOCK_STREAM表示是TCP通信
tcpSerSock.bind(ADDR) #socket与ADDR绑定到一起,表示socket可以在这个PORT上接收HOST处的连接请求。还需要开始监听
tcpSerSock.listen(5) #开始监听,5表示同时最多可以有5个连接进来。开始等待连接到来

while True: #表示一直等待
    print("waiting for connecting...")
    (tcpCliSock,addr)=tcpSerSock.accept() #如果有连接进来,则生成一个新的socket: tcpCliSock,并得到源的地址
    print('...connected from: ',addr)

    while True: #开始接收数据,并处理
        data=tcpCliSock.recv(BUFSIZ).decode('utf8') #接收数据,并转化成utf-8格式,原始的是bytes格式
        if not data: #如果没有数据了,则跳出while循环,等待下一次连接到来
            break;
        print('[%s] %s'%(ctime(),data)) #向shell输出,增加了当前的时间
        tcpCliSock.send(('[%s] %s'%(ctime(),data)).encode('utf8')) #将数据转换成bytes型,再send回去

    tcpCliSock.close() #关闭tcpCliSock,一次通信在服务器端的工作完成,继续等待下一次连接的到来
tcpSerSock.close() #不会被执行到,可以加个try-except进去,让程序更友好



TCP-客户端:

from socket import*

HOST='localhost' #由于是在同一台电脑上,所以可以写localhost,如果是在两台电脑上,则这儿要写server的ip地址
PORT=8080
BUFSIZ=1024
ADDR=(HOST,PORT) #服务器的ADDR

tcpCliSock=socket(AF_INET,SOCK_STREAM)
tcpCliSock.connect(ADDR) #连接到服务器,成功后,服务器端输出'...coonnected from...'

while True:
    data=input('> ').encode('utf8')
    if not data: #直接敲回车,则跳出while循环
        break;
    tcpCliSock.send(data) 
    data=tcpCliSock.recv(BUFSIZ) 
    if not data:
        break;
    print(data.decode('utf8'))
tcpCliSock.close()


        服务器端将bytes格式转换成utf-8格式,而客户端不需要的原因是,客户端没有对数据进行处理,直接输出;而服务器端在原数据基础上增加了当前的时间,所以转换成uft-8格式,便于处理。utf-8是unicode,可以处理汉语。

TCP-通信结果:

>>> 
waiting for connecting... #先运行服务器,等待客户端的运行
...connected from:  ('127.0.0.1', 53808) #当客户端运行,则出现这一行,同时客户端出现'> '
[Wed Aug 14 20:50:19 2013] Python编程 #输出处理后的结果
waiting for connecting... #一轮通信完成,等待下一轮的到来

>>> 
> Python编程 #向服务器端发送数据:'Python编程'
[Wed Aug 14 20:50:19 2013] Python编程 #得到服务器端的反馈
>  #结束
>>>



UDP-服务器端:(与TCP大同小异)

from socket import *
from time import ctime

HOST=''
PORT=8080
BUFSIZ=1024
ADDR=(HOST,PORT)

udpSerSock=socket(AF_INET,SOCK_DGRAM) #SOCK_DGRAM表示是UDP通信
udpSerSock.bind(ADDR) #绑定到ADDR,由于没有'虚电路',无连接,所以不需要listen,只是被动等待数据到来

while True: #被动等待数据到来
    print('waiting for message...')
    (data,addr)=udpSerSock.recvfrom(BUFSIZ) #当有数据到来时,得到data与源addr。注意这里不能产生一个新的socket,因为UDP是无连接的
    data=data.decode('utf8') #bytes转换成utf-8
    udpSerSock.sendto(('[%s] %s'%(ctime(),data)).encode('utf8'),addr) #传输处理后的数据,需要写addr,仍然因为是无连接
    print('...received from and returned to:',addr)

udpSerSock.close()



UDP-客户端:

from socket import*

HOST='localhost'
PORT=8080
BUFSIZ=1024
ADDR=(HOST,PORT)

udpCliSock=socket(AF_INET,SOCK_DGRAM) #SOCK_DGRAM表示UDP通信

while True:
    data=input('> ').encode('utf8')
    if not data:
        break;
    udpCliSock.sendto(data,ADDR)
    (data,ADDR)=udpCliSock.recvfrom(BUFSIZ) #可以得到ADDR
    if not data:
        break;
    print(data.decode('utf8'))

udpCliSock.close()



UDP-通信结果:

>>> 
waiting for message...
...received from and returned to: ('127.0.0.1', 56259)
waiting for message...

>>> 
> Python编程
[Wed Aug 14 20:53:39 2013] Python编程
> 
>>>