热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:46.PythonSocket编程

篇首语:本文由编程笔记#小编为大家整理,主要介绍了46.PythonSocket编程相关的知识,希望对你有一定的参考价值。复习:

篇首语:本文由编程笔记#小编为大家整理,主要介绍了46. Python Socket编程相关的知识,希望对你有一定的参考价值。


复习:消息队列

为了防止消息丢失,或者是调用方,不需一直等待响应方的结果。

threadtest.py

import codecs
from queue import Queue
from threading import Thread
import time
class Produce(Thread):
    def __init__(self, queue):
        super(Produce, self).__init__()
        self.fileName = "../firstlession/passwd"
        self.fileList = list()
        self.queue = queue
    def run(self):
        with codecs.open(self.fileName) as f:
            self.fileList += f.readlines()
        for line in self.fileList:
            self.queue.put(line)
class Consumer(Thread):
    def __init__(self, queue):
        self.queue = queue
        super(Consumer, self).__init__()
        self.newPasswd = "newpasswd.txt"
        self.fileList = list()
        self.stat = 1
    def run(self):
        while 1:
            if self.queue.empty():
                time.sleep(2)
                self.stat += 1
                if self.stat == 5:
                    break
            else:
                self.stat = 1
                data = self.queue.get()
                self.fileList.append(data)
        with codecs.open(self.newPasswd, 'w') as f:
            f.writelines(self.fileList)


调用

# thread1.py

from queue import Queue
from onlive.secondlesson.threadtest import Produce, Consumer
def main():
    q = Queue()
    produce = Produce(q)
    consumer = Consumer(q)
    produce.start()
    consumer.start()
if __name__ == '__main__':
    main()


socket 简介

TCP的可靠性实现:

(1)校验码

(2)接收方反馈

(3)信息包附带序号


UDP:

(1)快 不需要花费时间建立和关闭连接

(2)快 偶尔丢失一两个消息包无所谓,但是TCP会严格检查

(3)快 UDP的限制是一个信息包不超过64KB的数据


TCP和UDP区别:

UDP不建立连接,只保证数据的完整性,数据传输快,但是不保证数据是否真的被收到,也不保证数据是否只接收一次,也不保证次序。

TCP则相反。

服务端是用来给一个或多个客户端提供服务的,当客户端发起请求,开始等待服务端的返回结果,服务端接受完请求以后,根据自己的逻辑处理请求,并返回给客户端,客户端接收到返回结果以后,关闭和服务端的连接。

[备注]:只要是发送数据的就是"写",只要是接收数据的就是"读"。

最常用的客户端和服务端有两种模式:C/S模式(mysql) 和   B/S模式(百度、京东、淘宝网站等)


socket流程:

(1)服务端创建完一个socket以后

(2)需要绑定一个IP:PORT

(3)对其进行监听(listen)-->【监听的方法内需要带一个数字,这个数字表示同时有多少个客户端可以来访问服务端】

(4)然后接收请求(accept)--> 【客户端每创建一个连接,调用 "connect函数" 后,服务端就需要生成一个新的socket连接和客户端进行传输,传输完成后关闭客户端连接、关闭服务端;服务端永远要比客户端多一个socket连接,如果说客户端是n个socket连接,那么服务端就要创建n+1个socket连接,因为刚开始启动服务端的时候,服务端就要创建一个socket连接,每当从客户端过来一个连接,服务端就要创建一个socket连接跟客户端进行交互,所以服务端比客户端多一个连接。】

技术分享图片


socket常用函数讲解:

创建套接字:

s = socket.socket(address family, socket type)


address family:

socket.AF_INET 默认ipv4

socket.AF_INET6 ipv6

socket.AF_UNIX 只用于单一unix系统间进行通信

socket type:

socket.SOCK_STREAM 流式socket TCP

socket.SOCK_DGRAM 数据报式socket UDP


TCP 方式 socket:

(1)创建socket:

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

(2)绑定地址:

address = ('0.0.0.0', 8009)

s.bind(address) 或者 s.bind((0.0.0.0, 8009)) 两种方式相同;

【注意:address 必须是一个元组,容易错误,address = (host, port)】

【host:服务端ip,字符串类型,如果为0.0.0.0,代表本机的任意一个IP】

【port:服务端提供的端口,整形,0-1024为系统保留(不选这里面的端口)】

(3)监听消息:

s.listen(badklog)

backlog 代表可以同时接受多少个socket连接

(4)接受连接:

conn, addr = s.accept()

接受连接并返回元组(conn,addr),其中conn是新的套接字对象,每个新的连接就创建一个新的对象。可以用来接受和发送数据,addr是客户端的地址:包含host和port。

(5)发送数据:

s.send(string) 发送字符串到连接的套接字,可能未将指定内容全部发送;

s.sendall(string) 内部递归调用send,将所有内容发送出去,建议使用。

(6)接收数据:

data = s.recv(bufsize)

接收套接字数据,数据以字符串形式返回,bufsize指定最多接收的数据量,可以使用1024, 2048

如果不知道接收的数量有多少,可以能几个字节,可能几兆,一般通过循环接收。


UDP 方式 socket:

s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

s.sendto(string)

data, address = s.recvfrom(bufsize)


客户端:

客户端首先也要创建socket套接字

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)


客户端连接服务端函数:

s.connect(address) #连接到address的套接字

result - connect_ex(address) #成功返回0,失败返回错误码


通用:

s.close() #关闭套接字

s.getsocketname()   #获取套接字的名字

s.settimeout(timeout) #设置套接字超时时间,timeout为float类型,单位秒

s.gettimeout() #获得套接字超时时间

s.setblocking(flag) #flage为bool值

setblocking(True) is equivalent to settimeout(None); #不设置超时时间,一直阻塞在那里

setblocking(False) is equivalent to settimeout(0.0); #设置超时时间为0,如果设置False,accept和recv一旦无数据,则报错。

s.fileno() # 返回套接字的文件描述符(一个小整数)。这对于select.select()是有用的。


socket例子:


写socket工具:

util.py

import socket
import time
class InitSocketTest(object):
    def __init__(self, host, port, type):
        self.host = host
        self.port = port
        self.address = (host, port)
        self.type = type
        self.s = None
        self.creatsocket()
    def creatsocket(self):
        if self.type.upper() == "TCP":
            self.s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        elif self.type.upper == "UDP":
            self.s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
        else:
            print("you must input the InitSocket(type) is 'UDP|TCP' ")
class SocketServerTest(InitSocketTest):
    def __init__(self, host, port, type,  backlog):
        self.backlog = backlog
        super(SocketServerTest, self).__init__(host, port, type)
        self.clientAddress = None
    def run(self):
        self.s.bind(self.address)
        self.s.listen(self.backlog)
        print("server starting…………")
        conn, self.clientAddress = self.s.accept()
        print("accept connect from {0}".format(self.clientAddress))
        for i in range(1, 10):
            conn.sendall("i = {0}".format(str(i)).encode("utf-8"))
        self.s.close()
class ClientSocketTest(InitSocketTest):
    def run(self):
        self.s.connect(self.address)
        stat = 1
        while 1:
            data = self.s.recv(2048)
            if len(data)>0:
                stat = 1
                print(data.decode("utf-8"))
            else:
                stat += 1
                time.sleep(1)
                if stat == 5:
                    break


# testserver.py

from onlive.sockettest.util import SocketServerTest    
if __name__ == '__main__':    
    socketServer = SocketServerTest(host="0.0.0.0", port=9999, type="tcp", backlog=5)    
    socketServer.run()


testclient.py

from onlive.sockettest.util import ClientSocketTest    
if __name__ == '__main__':    
    socketClient = ClientSocketTest(host="127.0.0.1", port=9999, type="tcp")    
    socketClient.run()



推荐阅读
  • 第七课主要内容:多进程多线程FIFO,LIFO,优先队列线程局部变量进程与线程的选择线程池异步IO概念及twisted案例股票数据抓取 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • 本文介绍了如何在wxpython中将matplotlib图表嵌入到自定义窗体中的方法。通过调用FigureCanvasWx类,可以实现在自定义窗体中显示matplotlib图表。同时,还介绍了与此相关的一些类和参数。 ... [详细]
  • 本文介绍了解决二叉树层序创建问题的方法。通过使用队列结构体和二叉树结构体,实现了入队和出队操作,并提供了判断队列是否为空的函数。详细介绍了解决该问题的步骤和流程。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • 李逍遥寻找仙药的迷阵之旅
    本文讲述了少年李逍遥为了救治婶婶的病情,前往仙灵岛寻找仙药的故事。他需要穿越一个由M×N个方格组成的迷阵,有些方格内有怪物,有些方格是安全的。李逍遥需要避开有怪物的方格,并经过最少的方格,找到仙药。在寻找的过程中,他还会遇到神秘人物。本文提供了一个迷阵样例及李逍遥找到仙药的路线。 ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • 本文介绍了操作系统的定义和功能,包括操作系统的本质、用户界面以及系统调用的分类。同时还介绍了进程和线程的区别,包括进程和线程的定义和作用。 ... [详细]
  • STL迭代器的种类及其功能介绍
    本文介绍了标准模板库(STL)定义的五种迭代器的种类和功能。通过图表展示了这几种迭代器之间的关系,并详细描述了各个迭代器的功能和使用方法。其中,输入迭代器用于从容器中读取元素,输出迭代器用于向容器中写入元素,正向迭代器是输入迭代器和输出迭代器的组合。本文的目的是帮助读者更好地理解STL迭代器的使用方法和特点。 ... [详细]
  • 本文介绍了贝叶斯垃圾邮件分类的机器学习代码,代码来源于https://www.cnblogs.com/huangyc/p/10327209.html,并对代码进行了简介。朴素贝叶斯分类器训练函数包括求p(Ci)和基于词汇表的p(w|Ci)。 ... [详细]
  • linux进阶50——无锁CAS
    1.概念比较并交换(compareandswap,CAS),是原⼦操作的⼀种,可⽤于在多线程编程中实现不被打断的数据交换操作࿰ ... [详细]
author-avatar
易柔宛_968
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有