Python进阶基础学习(多线程)
Python进阶学习笔记(一)
threading模块
threading.thread(target = (函数)) 负责定义子线程对象
threading.enumerate() 负责查看子线程对象
测试代码如下:
import time
import threading
def sing():
for i in range(5):
print('-----唱歌-----')
time.sleep(1) def dance():
for i in range(5):
print('-----跳舞-----')
time.sleep(1) def main():
t1 = threading.Thread(target=sing) #子线程1
t2 = threading.Thread(target=dance) #子线程2
t1.start()
t2.start()
while True:
print(threading.enumerate())
if len(threading.enumerate()) == 1:
break
time.sleep(1)
if __name__ == '__main__':
main()
补充:
线程的调度是随机的,主线程如果在子线程之前GG,你们子线程也GG
threading.thread()不会创建线程,只有调用其实例.start()之后才会创建线程
通过类来创建线程
测试代码如下:
import threading
import time
class Person(threading.Thread):
def run(self) -> None:
for i in range(3):
print('aaa' + self.name)
time.sleep(1) if __name__ == '__main__':
p = Person()
p.start()
for i in range(3):
print('bbb')
time.sleep(1)
补充:
threading.thread是一个类,通过继承并且override run函数,即可声明进程
多线程共享全局变量
测试代码如下:
import threading
import time
num = 100
def test1():
global num
num += 1
print(num)
def test2():
print(num)
def main():
t1 = threading.Thread(target=test1)
t2 = threading.Thread(target=test2)
t1.start()
time.sleep(1)
test2()
if __name__ == '__main__':
main()
若创建线程是需要带参数,则是用args
测试代码如下:
num = [1,1,2,3] def test1(args):
args.append('hhs')
print(args) def main():
t1 = threading.Thread(target=test1,args=(num,))
t1.start()
print(num) if __name__ == '__main__':
main()
补充:args后边传的是一个元组,少了逗号会报错
多线程共享全局变量出现的问题
测试代码如下:
import threading
import time g_num = 0
def count1(n):
global g_num
for i in range(n):
g_num += 1 def count2(n):
global g_num
for i in range(n):
g_num += 1 def main():
t1 =threading.Thread(target=count1,args=(100000,))
t2 = threading.Thread(target=count2, args=(100000,))
t1.start()
t2.start()
time.sleep(2)
print('正常num为200000,实际num为%d' % g_num) if __name__ == '__main__':
main()
补充:输出结果为:175586,与正常结果不相同

解析问题:
1.cpu是一句句执行的
2.若把g_num += 1解析成很多句,如果线程1只执行1-2句,第三句还没赋值回去,就调用了线程2,则会出现相加的结果不等于预期的结果
互斥锁threading.Lock()
用于解决上面共享全局变量的问题
变量名.acquire() 上锁
变量名.release() 解锁
给要执行的代码上锁,自己运行的时候别的线程无法运行
测试代码如下:
import threading
import time num = 0
mutex = threading.Lock()
def count1(n):
global num
for i in range(n):
mutex.acquire()
num += 1
mutex.release() def count2(n):
global num
for i in range(n):
mutex.acquire()
num += 1
mutex.release() def main():
t1 =threading.Thread(target=count1,args=(100000,))
t2 = threading.Thread(target=count2, args=(100000,))
t1.start()
t2.start()
time.sleep(2)
print('正常num为200000,实际num为%d' % num) if __name__ == '__main__':
main()
Python进阶基础学习(多线程)的更多相关文章
- Python入门基础学习 二
Python入门基础学习 二 猜数字小游戏进阶版 修改建议: 猜错的时候程序可以给出提示,告诉用户猜测的数字偏大还是偏小: 没运行一次程序只能猜测一次,应该提供多次机会给用户猜测: 每次运行程序,答案 ...
- Python入门基础学习 三
Python入门基础学习 三 数据类型 Python区分整型和浮点型依靠的是小数点,有小数点就是浮点型. e记法:e就是10的意思,是一种科学的计数法,15000=1.5e4 布尔类型是一种特殊的整形 ...
- Python入门基础学习 一
Python入门基础学习 一 Python下载及安装 下载地址:https://www.python.org/,选择最新的版本下载 稍等一会,安装完成. 简单语句 从idle启动Python:IDLE ...
- python进阶强化学习
最近学习了慕课的python进阶强化训练,将学习的内容记录到这里,同时也增加了很多相关知识. 主要分为以下九个模块: 基本使用 迭代器和生成器 字符串 文件IO操作 自定义类和类的继承 函数装饰器和类 ...
- Python入门基础学习(文件与异常处理)
Python基础学习笔记(七) 捕获异常的语法格式: 文件的基本操作: 打开文件 读.写文件 关闭文件 read方法 --读取文件: open函数的第一个参数是要打开的文件名(文件名区分大小写) 如果 ...
- Python入门基础学习(时间模块,随机模块)
Python基础学习笔记(六) time模块: 时间的三种表示方法: 1.格式化字符串 2.时间戳 用来表示和1970年的时间间隔,单位为s 3.元组 struct_time 9个元素 time的st ...
- Python入门基础学习(模块,包)
Python基础学习笔记(五) 模块的概念:模块是python程序架构的一个核心概念 每个以拓展名py结尾的python源代码文件都是一个模块 模块名同样也是一个标识符,需要符合标识符的命名规则 在模 ...
- Python入门基础学习(面向对象)
Python基础学习笔记(四) 面向对象的三个基本特征: 封装:把客观事物抽象并封装成对象,即将属性,方法和事件等集合在一个整体内 继承:允许使用现有类的功能并在无须重新改写原来的类情况下,对这些功能 ...
- Python入门基础学习(函数)
Python基础学习笔记(三) 函数的概念: 所谓函数,就是把具有独立功能的代码块组织为一个小模块,在需要的时候调用 函数的使用包含两个步骤: 1.定义函数 --封装独立的功能 2.调用函数 --享受 ...
随机推荐
- tl-wr742n无线路由器怎么设置
tl wr742n无线路由器的设置方法如下: 1.宽带总线(猫出来的网线)连接路由器的WAN口. 2.将网线一头连接路由器任意LAN口,一头连接电脑,启动电脑和路由器设备,也可以通过手机连接无线路由器 ...
- [WPF 自定义控件]自定义控件库系列文章
Kino.Toolkit.Wpf Kino.Toolkit.Wpf是一组简单实用的WPF控件与工具,用于介绍自定义控件的入门.相关博客地址如下: 开始一个自定义控件库项目 介绍开始一个自定义控件库项目 ...
- Hibernate 框架 -HQL 语法
HQL ( Hibernate Query Language ) 查询语言是面向对象的查询语言,也是在 Hibernate 中最常见的.其语法和 SQL 语法有一些相似,功能十分强大,几乎支持除特殊 ...
- JavaScript 自定义html元素鼠标右键菜单
自定义html元素鼠标右键菜单 实现思路 在触发contextmenu事件时,取消默认行为(也就是阻止浏览器显示自带的菜单),获取右键事件对象,来确定鼠标的点击位置,作为显示菜单的left和top值 ...
- 【好书推荐】《剑指Offer》之硬技能(编程题1~6)
本文例子完整源码地址:https://github.com/yu-linfeng/BlogRepositories/tree/master/repositories/sword 前一篇<[好书推 ...
- Docker动态添加端口,不需要重新建立镜像
Docker容器在运行期间有时可能会需要修改或者添加暴露的端口,但是有时候运行的容器又不想再另外建立一个新的镜像.这时可以找到docker容器的存放地方,然后直接修改配置文件. 我们的容器都是保存在/ ...
- Linux笔记15 使用Apache服务部署静态网站。
配置服务文件参数Linux系统中的配置文件 服务目录 /etc/httpd 主配置文件 /etc/httpd/conf/httpd.conf 网站数据目录 /var/www/html 访问日志 /va ...
- diango中的MTV——FBV/CBV以及装饰器的复用问题解决
MVC M: model 模型 与数据库交互 V: view 视图 HTML C:controller 控制器 流程 和 业务逻辑 MTV M:model ORM T:template 模板 HTML ...
- 痞子衡嵌入式:飞思卡尔i.MX RTyyyy系列MCU启动那些事(2)- Boot配置(BOOT Pin/eFUSE)
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是飞思卡尔i.MX RTyyyy系列MCU的Boot配置. 在上一篇文章 Boot简介 里痞子衡为大家介绍了Boot基本原理以及i.MXR ...
- 使用navicat连接只开放内网ip连接的数据库
无法通过Navicat来连接MySQL,比较常见的两种问题? 服务器上自己安装的MySQL数据库,且未开通外网登录账号 直接购买服务商的MySQL数据库不创建公网访问,只有内网访问 背景: 公司数 ...