python-多线程、网络编程、正则表达式

news2024/12/23 2:10:12

目录

闭包

多线程

主线程

线程阻塞

同步锁

网络编程

 正则表达式

re.match函数

re.search方法

re.match与re.search的区别

re.findall()方法

正则表达式的特殊规则


闭包


account=0
def atm(num,flag):
    global account
    if flag:
        account=num+account
        print(account)
    else:
        account-=num
        print(account)

atm(300,True)  #存入300元
atm(100,False) #取出100元

这是一个简单atm存取钱逻辑,account表示账户余额,存在问题是,account是全局变量,可以被任意访问和修改,为了解决这个问题,引入了闭包

闭包:在函数嵌套的前提下,内部函数引用外部函数的变量,并且外部函数返回了内部函数,将这个内部函数称为闭包

def out(account):
    def atm(num, flag):
        nonlocal account   #nonlocal关键字 让外部函数参数是一个可修改的值
        if flag:
            account += num
            print("余额",account)
        else:
            account -= num
            print("余额",account)
    return atm

atm=out(100)  #起始余额是100 返回值是内部函数
atm(200,True) #存入200
atm(100,False) #取出100

这时account不是全局变量,不可以被任意访问和修改。

多线程

我们电脑可以运行多个程序,运行多个程序可以称为运行多个进程。一个进程可以多含多个线程,线程是cpu运行的基本单位。

python的多线程可以通过threading模块来实现

obj=threading.Thread(group,target,name,args,kwargs)

  • group:暂时无用,未来功能的预置参数
  • target:执行的任务名称
  • args:以元组的形式传入参数
  • kwargs:以字典形式传入参数
  • name:线程名称,一般不用设置

start()方法:线程执行


import time
import threading #1、导入threading模块
def sing(**kwargs):
    while True:
        print(kwargs["name"],"在唱歌")
        time.sleep(1)

def dance():
    while True:
        print("跳舞")
        time.sleep(1)

if __name__=='__main__':
    print("作为主函数运行")
    # 创建一个进程 这个进程执行的是唱歌这个函数 传入一个字典
    sing=threading.Thread(target=sing,kwargs={"name":"张三"})
    # 创建一个进程 这个进程执行的是跳舞这个函数
    dance = threading.Thread(target=dance)

    sing.start()  #线程启动运行
    dance.start()

主线程

在python中,主线程是第一个启动的线程。

~父线程:如果启动线程A中启动了一个线程B,A就是B的父线程。

~子线程:B就是A的子线程。

创建线程时有一个damon属性,用它来判断主线程。当daemon设置False时,子线程不会随主线程退出而退出,主线程会一直等着子线程执行完。当daemon设置True时,子线程会随主线程退出而退出,主线程结束,其他的子线程会强制退出。

import time
import threading #1、导入threading模块

num=0
def dance():
    global num
    while num < 10:
        num = 1 + num
        print("跳舞")
        time.sleep(1)
        
count=0
def sing(**kwargs):
    global count
    while count<10:
        count=1+count
        print("在唱歌")
        time.sleep(1)
        threading.Thread(target=dance).start()

if __name__=='__main__':
    # 创建一个进程 这个进程执行的是唱歌这个函数 传入一个字典
    sing=threading.Thread(target=sing,daemon=False)
    sing.start()  #线程启动运行

sing线程运行内启动了dance线程,sing线程就是父线程,dance线程为子线程。damon为False,两个线程交替运行,damon为True,主线程结束之后就会直接退出 ,不执行子线程

damon为True时,打印 "在唱歌" 之后,time.sleep(1)让sleep线程挂起1s,父线程运行结束,不会执行子线程


import time
import threading #1、导入threading模块

num=0
def dance():
    global num
    while num < 10:
        num = 1 + num
        print("跳舞")
        time.sleep(1)

count=0
def sing(**kwargs):
    global count
    while count<10:
        count=1+count
        print("在唱歌")
        time.sleep(1)
        threading.Thread(target=dance).start()

if __name__=='__main__':
    # 创建一个进程 这个进程执行的是唱歌这个函数 传入一个字典
    sing=threading.Thread(target=sing,daemon=False)
    sing.start()  #线程启动运行

线程阻塞

join方法,两个A,B并发运行的线程,A线程join()之后,A线程阻塞,直到B线程运行结束之后,A线程恢复运行


import time
import threading #1、导入threading模块
class Thread(threading.Thread):

    def __init__(self,name):  #构造函数
        threading.Thread.__init__(self)
        self.name=name

    def run(self) -> None:  #线程运行时 直接执行 run()方法
        for i in range(0,10):
            print(i,self.name,time.ctime(time.time()))

thread1=Thread("线程A")
thread1.start()

for i in range(0,10):
    print("线程B")
    if(i==2):
        thread1.join()
        

线程A,线程B并发运行,在线程B的i=2时,线程B阻塞,线程A一直运行,直到线程A运行结束,线程B才会恢复运行。

其他方法

run():用以表示线程活动的方法

start():启动线程

join():等待至线程终止

isAlive():返回线程是否活动的

getName():返回线程名称

setName() : 设置线程名称

同步锁

锁机制 让一个可变数据,在被修改期间不可以被其他线程读取,保证数据读取正确

使用Thread对象的Lock和Rlock可以实现简单的线程同步,这两个对象都有acquire方法和release方法,对于那些需要每次只允许一个线程操作的数据,可以将其操作放到acquire和release方法之间,即acquire相当于上锁,而release相当于解锁。

一个场景:两个窗口一起卖100张车票,在没有锁时,ticket数据在被窗口一修改的同时,被窗口二读取到修改前的数据,那么就会导致 这两个窗口一起卖出第i张票的情况,这是不合理的

import time
import threading #1、导入threading模块
class Thread(threading.Thread):

    def __init__(self,name):
        threading.Thread.__init__(self)
        self.name=name

    def run(self) -> None:
        global ticket
        while ticket > 0:
            print("%s%s%d%s" % (self.name, "卖出了第", ticket, "张票"))
            ticket = ticket - 1
            time.sleep(1)

ticket =10  #设置全局变量初始值

thread1=Thread("窗口一")
thread1.start()

thread2=Thread("窗口二")
thread2.start()

加锁:加上同步锁,保证ticket数据在修改时,不可以被其他进程访问


import time
import threading #1、导入threading模块
class Thread(threading.Thread):

    def __init__(self,name):
        threading.Thread.__init__(self)
        self.name=name

    def run(self) -> None:
        global ticket
        while ticket > 0:
            lock.acquire()  # 加锁 保证ticket被一个线程持有 其他线程不得访问这个变量
            if(ticket<=0):
                break
            print("%s%s%d%s" % (self.name, "卖出了第", ticket, "张票"))
            ticket = ticket - 1
            lock.release()  # 解锁

ticket =100  #设置全局变量初始值
lock=threading.RLock() #获取锁
thread1=Thread("窗口一")
thread1.start()

thread2=Thread("窗口二")
thread2.start()

网络编程

Socket(套接字)负责实现网络编程

创建服务器

"""
服务器程序
"""

# 1、导入Socket模块
import socket

# 2、创建Socket对象
service = socket.socket()

# 3、绑定 ip地址和端口
# bind() 绑定地址(host,port)到套接字, 在 AF_INET下,以元组(host,port)的形式表示地址。
service.bind(("localhost", 8888))

# 4、监听端口
# 开始 TCP 监听。参数 指定在拒绝连接之前,操作系统可以挂起的最大连接数量。该值至少为 1,大部分应用程序设为 5 就可以了
service.listen(2)  # 可连接次数2

while True:
    # 5、接收客户端信息
    # s.accept()	被动接受TCP客户端连接,(阻塞式)等待连接的到来  返回元组信息(con,address).con表示连接信息,address表示客户端信息
    con, address = service.accept()
    print("客户端信息: ", address)

    # 6、接收客户端消息
    # s.recv()	接收 TCP 数据,数据以字符串形式返回,bufsize 指定要接收的最大数据量。flag 提供有关消息的其他信息,通常可以忽略。

    data = con.recv(1024).decode("utf-8")
    if (data == "exit"):
        con.send("退出".encode())
        con.close()
        break
    # 以1024字节空间接收客户端发生的数据
    print("客户端发送的数据: ", data)

    # 7、服务器发送数据
    str = input("服务器接受到了客户端信息,输入回应信息: ")
    con.send(str.encode())
    # 8、关闭连接
    con.close()

service.close()

创建客户端


"""
客户端程序
"""
#1、导入socket模块
import socket


#2、创建Socket对象
service=socket.socket()

#3、开启Tcp连接
#s.connect()	主动初始化TCP服务器连接,。一般address的格式为元组(hostname,port),如果连接出错,返回socket.error错误
service.connect(("localhost",8888))

#4、发送数据  提供参数应该是byte类型
str=input("输入发送数据: ")
service.send(str.encode())

print("客户端接收服务器返回结果: ",service.recv(1024).decode("UTF-8"))
#5、关闭连接
service.close()

 正则表达式

正则表达式是一个特殊的字符序列,它能帮助你方便的检查一个字符串是否与某种模式匹配。

re 模块使 Python 语言拥有全部的正则表达式功能。

re.match函数

re.match 尝试从字符串的起始位置匹配一个模式,如果不是起始位置匹配成功的话,match() 就返回 none。

函数语法

re.match(pattern, string, flags=0)

函数参数说明:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。

匹配成功 re.match 方法返回一个匹配的对象,否则返回 None。

我们可以使用 group(num) 或 groups() 匹配对象函数来获取匹配表达式。

import  re

str="hello world hi"
result=re.match("hello world",str)
print(result)
print(result.span())  #得到目标字符串 在str字符串中中匹配结果的位置
print(result.group()) #到的匹配字符串

re.search方法

re.search 扫描整个字符串并返回第一个成功的匹配。

函数语法:

re.search(pattern, string, flags=0)

函数参数说明:

参数描述
pattern匹配的正则表达式
string要匹配的字符串。
flags标志位,用于控制正则表达式的匹配方式,如:是否区分大小写,多行匹配等等。
import  re

str="hi hello world hi"
result=re.search("hello world",str)
print(result)
print(result.span())  #得到目标字符串 在str字符串中中匹配结果的位置
print(result.group()) #到的匹配字符串

re.match与re.search的区别

re.match只匹配字符串的开始,如果字符串开始不符合正则表达式,则匹配失败,函数返回None;而re.search匹配整个字符串,直到找到一个匹配。

re.findall()方法

在字符串中找到正则表达式所匹配的所有子串,并返回一个列表,如果有多个匹配模式,则返回元组列表,如果没有找到匹配的,则返回空列表。

注意: match 和 search 是匹配一次 findall 匹配所有。

正则表达式的特殊规则

正则表达式可以包含一些可选标志修饰符来控制匹配的模式。修饰符被指定为一个可选的标志。多个标志可以通过按位 OR(|) 它们来指定。

修饰符描述
re.I使匹配对大小写不敏感
re.L做本地化识别(locale-aware)匹配
re.M多行匹配,影响 ^ 和 $
re.S使 . 匹配包括换行在内的所有字符
re.U根据Unicode字符集解析字符。这个标志影响 \w, \W, \b, \B.
模式描述
^匹配字符串的开头
$匹配字符串的末尾。
.匹配任意一个字符(除了\n),\. 表示匹配.本身
[]匹配[]内的字符
\w匹配字母数字及下划线
\W匹配非字母数字及下划线
\s匹配任意空白字符
\S匹配任意非空字符
\d匹配任意数字,等价于 [0-9]
\D匹配任意非数字
\A匹配字符串开始
\Z匹配字符串结束,如果是存在换行,只匹配到换行前的结束字符串。
\z匹配字符串结束
*匹配前一个规则的字符0次到无数次
+匹配前一个规则的字符1次到无数次
匹配前一个规则的字符0次或者1次
{m}匹配前一个规则的字符m次
{m,}匹配前一个规则的字符最少m次
{m,n}匹配前一个规则的字符m次到n次

 


import re

#1、匹配账户 只能有数字或者英文字母组成 长度6-16位
rule='^[0-9a-zA-z]{6,16}$'
str="1234abajmhkkkhJ"
result=re.match(rule,str)
print("匹配结果",result)

#2、匹配QQ号 要求10位数字 第一位不是0
rule='^[1-9][0-9]{9}$'     #第一位数字1-9,剩余数字0-9并且长度是9
str="9089776555"
result=re.match(rule,str)
print("匹配结果",result)

#匹配邮箱地址  10位数字 后面跟着@符号 后缀 QQ 或者 136  后面跟着.com

rule='^[0-9]{10}[@](QQ|136){1}(.com){1}$'
str="1234567890@136.com"     
result=re.match(rule,str)
print("匹配结果",result)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/113938.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

免费开源的高精度OCR文本提取,支持 100 多种语言、自动文本定位和脚本检测,几行代码即可实现离线使用(附源码)

免费开源的高精度OCR文本提取,支持 100 多种语言、自动文本定位和脚本检测,几行代码即可实现离线使用(附源码)。 要从图像、照片中提取文本吗?是否刚刚拍了讲义的照片并想将其转换为文本?那么您将需要一个可以通过 OCR(光学字符识别)识别文本的应用程序。 图片文字识…

html圣诞树代码

一、前言 想做一个圣诞树&#xff0c;通过html实现了下 二、效果展示 三、代码 <!DOCTYPE html> <html> <head> <meta http-equiv"Content-Type" content"text/html; charsetutf-8" /> <meta name"viewport" cont…

245. 你能回答这些问题吗——线段树

给定长度为 N 的数列 A&#xff0c;以及 M 条指令&#xff0c;每条指令可能是以下两种之一&#xff1a; 1 x y&#xff0c;查询区间 [x,y] 中的最大连续子段和&#xff0c; 2 x y&#xff0c;把 A[x] 改成 y。 对于每个查询指令&#xff0c;输出一个整数表示答案。 输入格式…

RabbitMQ 第二天 高级 9 RabbitMQ 集群搭建 9.3 集群管理 9.5 负载均衡-HAProxy

RabbitMQ 【黑马程序员RabbitMQ全套教程&#xff0c;rabbitmq消息中间件到实战】 文章目录RabbitMQ第二天 高级9 RabbitMQ 集群搭建9.3 集群管理9.5 负载均衡-HAProxy9.5.1 安装HAProxy9.5.2 配置HAProxy第二天 高级 9 RabbitMQ 集群搭建 9.3 集群管理 rabbitmqctl join_cl…

MariaDB上市:MySQL之父奋斗13年终敲钟 要写代码写到100岁

雷递网 雷建平 12月24日云数据库公司MariaDB日前与特殊目的公司Angel Pond Holdings完成合并&#xff0c;并在纽交所上市&#xff0c;新公司更名为MariaDB。MariaDB是2022年初与Angel Pond Holdings达成合并协议&#xff0c;对新公司的作价为6.72亿美元。MariaDB是MySQL之父Mic…

【技术应用】java基于UNIX域套接字(unix domain socket)连接redis

【技术应用】java基于UNIX域套接字unix domain socket连接redis一、前言二、实现思路三、代码实现1、java socket基于redis.sock连接redis2、Lettuce框架基于redis.sock连接redis一、前言 在公司工作中经常涉及到一些中小型项目&#xff0c;这些项目都会涉及使用redis数据库&a…

Redis5.0+——持久化——RDBAOF

Redis持久化-RDB 1.实现目标&#xff1a; 在redis持久化时&#xff0c;持久化dump.rdb文件放入到redis解压目录下的data目录下的6379目录下 2.前期准备 1.在redis-5.0.3解压目录下新建data数据目录 2.编辑前面配置的/etc/redis.conf配置文件 修改持久化文件位置 (1) 进入安…

MySQL热备之PXB备份与恢复

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&#x1f61…

【语音处理】使用块反射器的基于DFT的系统中用于旁瓣抑制的正交预编码(Matlab代码实现)

&#x1f468;‍&#x1f393;个人主页&#xff1a;研学社的博客 &#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜…

Java中的多线程(下)

作者&#xff1a;~小明学编程 文章专栏&#xff1a;JavaEE 格言&#xff1a;热爱编程的&#xff0c;终将被编程所厚爱。 目录 多线程案例 单例模式 饿汉模式 懒汉模式 阻塞式队列 为什么要引入阻塞队列 Java中的阻塞队列 模拟实现阻塞队列 定时器 标准库中的定时器 …

docker的虚悬镜像是什么?

虚悬镜像是什么? 答:仓库名、标签都是<none>的镜像,俗称:dangling image 我们使用Dockerfile写一个: 1:编写 from ubuntu CMD echo action is success2:构建 docker build . 注意没有 -t 产生原因: 1:构建时候因为编写错误导致 2:删除的时候 对于这样…

数据溢出的二进制原理

char 类型的数据占一个字节&#xff0c;一个字节有 8 位&#xff0c;最高位为符号位&#xff0c;1表示负数&#xff0c;0表示正数。在计算机中&#xff0c;数据用补码表示&#xff0c;正数的补码是它本身&#xff0c;负数的补码为 “符号位不变&#xff0c;其他位取反后再加1”…

Spring Cloud 系列之OpenFeign:(4)集成OpenFeign

目录 传送门 服务间调用 集成OpenFeign 说明文档 添加pom依赖 启用OpenFeign 声明OpenFeign接口 改造远程调用 定义OpenFeign接口 测试OpenFeign调用 传送门 Spring Cloud Alibaba系列之nacos&#xff1a;(1)安装 Spring Cloud Alibaba系列之nacos&#xff1a;(2)单…

离散数学数理逻辑部分【2】

文章目录命题逻辑等值演算公式的使用【重点】析取范式和合取范式【重点】范式存在定义【了解】求公式A的范式的步骤&#xff1a;【重点】极大项和极小项【重点】主合取范式和主析取范式【重点】等式演算求主析取范式【重点】真值表求主析取范式【了解】主范式的应用【重点】推理…

Python : 使用python实现教务管理系统(GUI界面+数据库)

一、设计目的 1.熟悉Python和相关软件的操作。 2.基于本学期所学Python知识&#xff0c;熟练应用掌握&#xff0c;制作符合要求的教务管理系统。 3.会对程序运行中的错误代码进行分析&#xff0c;找出合理的解决方案。 4.掌握tkinter开发流程&#xff0c;布局方法和主要组件&a…

C语言位域

如果程序的结构中包含多个开关量&#xff0c;只有 TRUE/FALSE 变量&#xff0c;如下&#xff1a; struct {unsigned int widthValidated;unsigned int heightValidated; } status; 这种结构需要 8 字节的内存空间&#xff0c;但在实际上&#xff0c;在每个变量中&#xff0c;…

JavaScript游戏开发(4)(笔记)

文章目录八、角色动作状态的管理8.1 准备部分8.2 角色状态改变的基本方式8.3 完善整个代码8.4 存在的问题九、简单的横板动作卷轴游戏9.1 准备部分9.2 输入管理器9.3 状态管理器9.4 背景管理器9.5 敌人管理器9.6 碰撞检测、UI绘制9.7 更多的角色状态与特效9.8 完善游戏附录素材…

跳槽、换房、不忘输出,与你分享我匆忙的 2022~

前些日子下班回家的瞬间&#xff0c;忽然想起去年春节还在跟老爸吐露职场的困境和对房子的无奈。哪曾想过了不到半年的时间竟全部解决&#xff0c;令我不禁感叹人生的捉摸不透。 让我姑且花点文字记录下&#xff0c;与你分享我这一年的匆匆忙忙&#xff5e; 目录前瞻&#xf…

Listener监听器 | 监听域对象创建和销毁、使用监听器统计网站在线人数

目录 一&#xff1a;监听域对象创建和销毁 1、什么是监听器&#xff1f;监听器有什么用&#xff1f; 2、Servlet规范中提供了哪些监听器&#xff1f; 3、实现一个监听器的步骤 4、HttpSessionBindingListener 5、HttpSessionIdListener & HttpSessionActivationList…

【编译原理】实验二:NFA到DFA

目录 实验二 NFA 到 DFA 一、实验目的 二、预备知识 三、实验内容 NFA向DFA的转换的思路 NFA和DFA之间的联系 NFAToDFA.h 文件 main.c 文件 RegexpToPost.c 文件 PostToNFA.c 文件 NFAFragmentStack.c 文件 PostToNFA.h 文件 NFAFragmentStack.h 文件 NFAStateStack.h 文件 dem…