Python中socket编程

一 网络通讯三要素

1. IP:网络中每一台计算机的唯一标识,通过IP地址找到指定的计算机。

分类:
IPv4: 172.25.254.100 ===> 32位的二进制格式, 点分十进制法; 232-1
IPv6: ===> 128位的二进制格式 , 冒分十六进制;
查看:

ip addr show br0
ifconfig
2. 端口:用于标识进程的逻辑地址,通过端口找到指定进程。

常见的port和服务的对应关系:/etc/services
已经被分配的port: 0-1024
自定义端口号的范围: 1024-65535
物理端口:网卡口
逻辑端口:用于标识进程的逻辑地址,不同进程使用的端口是不同的,计算机通过端口找到指定进程,有效端口为0 ~ 6 + 5535,其中1~1024是系统使用的端口或保留端口。

3. 协议:定义通信规则,符合协议则可以通信,不符合不能通信。

协议是定义的通信规则,一般有TCP协议和UDP协议
TCP协议是在通信的两台设备之间建立连接通道,对传输的数据大小没有限制,但是因为建立连接,可靠一些,但是速度会慢一些。TCP协议又称为三次握手协议,因为建立过程有三步,发送请求、获取反馈、建立连接。通常使用的蓝牙、打电话都是TCP协议。
UDP协议需要将数据打包,因为包有大小,所以对数据大小有限制,UDP是不用建立连接的,不保证待接收方一定会接收到消息,所以不可靠,但是因为不建立连接,速度要快一些。例如发短信。

二 socket编程

import socket
print(socket.gethostname())
#根据域名获取对应服务器的ip地址
print(socket.gethostbyname('www.baidu.com'))
#根据ip获取对应的主机名
print(socket.gethostbyaddr('114.114.114.114'))
print(socket.getaddrinfo('www.xunlei.com',80))

在这里插入图片描述

AddressFamily.AF_INET : ipv4 (用于服务器与服务器之间的网络通信)
socket.AF_INET6 : ipv6 (基于IPV6的服务器与服务器之间的网络通信)
SOCK_STREAM: TCP协议 (基于TCP的流式socket通信)
socket.SOCK_DGRAM: UDP协议 (基于UDP的数据报式socket通信)

三 socket实现web简易服务器

import  socket

def handle_request(sockObj):
    sockObj.send(b'HTTP/1.1  200 OK\r\n\r\n')
    with open('hello.html') as f:
        sockObj.send(f.read().encode('utf-8'))

if __name__ == '__main__':
    #1.创建一个socket对象
    server = socket.socket()
    #2.绑定ip和端口
    server.bind(('172.25.254.74', 9002))
    #3.监听是否有客户端连接
    server.listen(3)
    print("服务器端已经启动9002端口......")

    while True:
        #4.接受客户端连接
        sockObj, address = server.accept()
        print(sockObj, address)
        #5.接受客户端发送的消息
        recv_data = sockObj.recv(1024)
        # sockObj.send(b'HTTP/1.1  200 OK\r\n\r\n')
        # sockObj.send(b'<h1>westos</h1>')

        #6.与客户端进行交互, 返回给客户端信息
        handle_request(sockObj)
        sockObj.close()

在这里插入图片描述
在这里插入图片描述

四 socket实现TCP聊天室

TCP服务端:

# 1. 创建一个socket对象
import socket
server = socket.socket()
# 2. 绑定ip和端口
server.bind(('172.25.254.74', 9005))
# 3. 监听是否有客户端连接
server.listen()
print("服务器端已经启动9005端口......")
# 4. 接收客户端连接
sockObj, address = server.accept()

while True:
    # 5. 接收客户端发送的消息
    recv_data = sockObj.recv(1024).decode('utf-8')
    print("client>:%s" %(recv_data))
    if recv_data == 'quit':
        break
    # 6. 给客户端回复消息
    send_data = input("server>:")
    if send_data == 'quit':
        break
    sockObj.send(send_data.encode('utf-8'))
# 7. 关闭socket对象
sockObj.close()
server.close()

TCP客户端:

import  socket

HOST = '172.25.254.74'
PORT = 9005

# 1. 创建客户端的socket对象
client = socket.socket()
# 2. 连接服务端,需要指定端口和IP
client.connect((HOST, PORT))
while True:
    # 3. 给服务端发送数据
    send_data = input("client:>")
    client.send(send_data.encode('utf-8'))
    if send_data == 'quit':
        break

    # 4. 获取服务端返回的消息
    recv_data = client.recv(1024).decode('utf-8')
    if recv_data == 'quit':
        break
    print("server:>%s" %(recv_data))
# 5. 关闭socket连接
client.close()

服务端:
在这里插入图片描述
客户端:
在这里插入图片描述
程序只能连接一个客户端,并不能支持多并发,所以当出现第二个客户端时,就会出现挂起不动,所以想要完成多并发,可以通过前面提到的多协程来实现,当然还有其它的方法

协程方式建立服务端
def handle_request(sockObj):
    while True:
        # 5. 接收客户端发送的消息
        recv_data = sockObj.recv(1024).decode('utf-8')
        print("client>:%s" % (recv_data))
        if recv_data == 'quit':
            break
        # 6. 给客户端回复消息
        # send_data = "hello client"
        # print("server>: %s" %(send_data))
        send_data = input("server>:")
        if send_data == 'quit':
            break
        sockObj.send(send_data.encode('utf-8'))


from gevent import  monkey
monkey.patch_all()
import  gevent

# 1. 创建一个socket对象
import socket

server = socket.socket()
# 2. 绑定ip和端口
server.bind(('172.25.254.250', 9006))
# 3. 监听是否有客户端连接
server.listen()
print("服务器端已经启动9006端口......")

while True:
    # 4. 接收客户端连接
    sockObj, address = server.accept()
    # handle_request(sockObj)
    gevent.spawn(handle_request, sockObj)
# 7. 关闭socket对象
sockObj.close()
server.close()

每一次接收客户端连接后,便开启一个协程,以此来实现同时接收多个客户端

五 socket实现UDP聊天室

UDP服务端

import socket
HOST = '172.25.254.74'
PORT = 6001
# 1. 创建socket对象
server =  socket.socket(type=socket.SOCK_DGRAM)
# 2. 绑定IP和port
server.bind((HOST, PORT))
print("等待客户端的UDP请求.......")
# 3. 接收客户端发送的消息
data, address = server.recvfrom(1024)
print("接收到客户端的消息:", data.decode("utf-8"))
print("客户端连接的socket地址:",  address)
# 4. 给客户端回复消息
server.sendto(b'hello client', address)
# 5. 关闭socket对象
server.close()

在这里插入图片描述
UDP客户端:

import socket
HOST = '172.25.254.74'
PORT = 6001

# 1. 创建socket对象
client =  socket.socket(type=socket.SOCK_DGRAM)
# 2. 发送消息给服务端
client.sendto(b"hello server", (HOST, PORT))
# 3. 接收服务端返回的信息
data, address = client.recvfrom(1024)
print("接收服务端的消息:", data)
# 4. 关闭socket对象
client.close()

在这里插入图片描述
执行结束,接收到服务端的连接成功通知
在这里插入图片描述

通过socket获取网页内容:
import socket
from urllib.request import urlopen

# 获取网页内容
#print(urlopen('http://www.baidu.com').read())

client = socket.socket()
client.connect(('www.baidu.com', 80))
client.send(b'GET / HTTP/1.1\r\nHost: www.baidu.com\r\nConnection: close\r\n\r\n')
recv_data = client.recv(1024*100)
print(recv_data.decode('utf-8'))
client.close()

执行结果:

HTTP/1.1 200 OK
Accept-Ranges: bytes
Cache-Control: no-cache
Content-Length: 14615
Content-Type: text/html
Date: Sat, 26 Jan 2019 02:36:18 GMT
Etag: "5c36c624-3917"
Last-Modified: Thu, 10 Jan 2019 04:12:20 GMT
P3p: CP=" OTI DSP COR IVA OUR IND COM "
Pragma: no-cache
Server: BWS/1.1
Set-Cookie: BAIDUID=0DCB53D24218A3DE6E219CC2FCAF6F43:FG=1; expires=Thu, 31-Dec-37 23:55:55 GMT; max-age=2147483647; path=/; domain=.baidu.com
Set-Cookie: BIDUPSID=0DCB53D24218A3DE6E219CC2FCAF6F43; expires=Thu, 31-Dec-37 23:55:55 GMT; max-age=2147483647; path=/; domain=.baidu.com
Set-Cookie: PSTM=1548470178; expires=Thu, 31-Dec-37 23:55:55 GMT; max-age=2147483647; path=/; domain=.baidu.com
Vary: Accept-Encoding
X-Ua-Compatible: IE=Edge,chrome=1
Connection: close

<!DOCTYPE html><!--STATUS OK-->
<html>
<head>
	<meta http-equiv="content-type" content="text/html;charset=utf-8">
	<meta http-equiv="X-UA-Compatible" content="IE=Edge">
	<link rel="dns-prefetch" href="//s1.bdstatic.com"/>
	<link rel="dns-prefetch" href="//t1.baidu.com"/>
	<link rel="dns-prefetch" href="//t2.baidu.com"/>
	<link rel="dns-prefetch" href="//t3.baidu.com"/>
	<link rel="dns-prefetch" href="//t10.baidu.com"/>
	<link rel="dns-prefetch" href="//t11.baidu.com"/>
	<link rel="dns-prefetch" href="//t12.baidu.com"/>
	<link rel="dns-prefetch" href="//b1.bdstatic.com"/>
	<title>百度一下,你就知道</title>
	<link href="http://s1.bdstatic.com/r/www/cache/static/home/css/index.css" rel="stylesheet" type="text/css" />
	<!--[if lte IE 8]><style index="index" >#content{height:480px\9}#m{top:260px\9}</style><![endif]-->

可知其是一个httml页面,解析即可

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值