2019-04-17-day034-线程与数据共享
内容回顾
锁
- 互斥锁
- 能够保护数据的安全性
- 保证对于数据的修改操作同一时刻多个进程只有一个进程执行
- 进程数据不安全 : 同时修改文件/数据库/其他共享资源的数据
###队列 -- 实现了进程之间的通信(IPC) - 进程队列 -- 进程安全
- 从multiprocessing导入Queue
- q = Queue()
- put/get/put_nowait(丢数据)/get_nowait
- 基于 管道 + 锁,管道进程不安全
- 管道 基于文件级别的socket实现的
import queue
from multiprocessing import Queue
q = Queue(5)
q.put(1)
q.put(1)
q.put(1)
q.put(1)
q.put(1)
print('________')
try:
q.put_nowait(1)
except queue.Full:
pass
print('********')
print(q.get())
print(q.get())
print(q.get())
print(q.get())
print(q.get())
try:
print(q.get_nowait())
except queue.Empty:
pass
生产者消费者模型
- 一个生产数据和消费数据的完整的流程解耦成为两个部分 : 生产 和 消费
- 由于生产速度和消费速度不一致,所以需要我们调整生产者和消费者的个数来达到效率平衡
互斥锁
from multiprocessing import Lock
# 互斥锁:在同一个进程内,也有锁的竞争关系
# 在同一个进程中连续acquire多次会产生死锁
lock = Lock()
lock.acquire() * 拿走钥匙
print(123)
lock.acquire() * 又想要钥匙 卡住
print(456)
lock.release()
lock.release()
数据共享
from multiprocessing import Process,Manager,Lock
def func(dic,lock):
with lock:
dic['count'] -= 1
if __name__ == '__main__':
lock = Lock()
m = Manager()
dic = m.dict({'count':100})
p_l = []
for i in range(100):
p = Process(target=func,args=(dic,lock))
p.start()
p_l.append(p)
for p in p_l:p.join()
print(dic)
- 100减100次1这么慢? 不是减操作造成的 而是开启进程 管理进程 销毁进程拖慢了程序的执行速度
- 为什么在这里出现了数据不安全的现象?
- 什么情况下会出现数据不安全 : Manager类当中对字典\列表 += -= *= /=
- 如何解决 : 加锁
线程
- 线程是进程中的一个单位
- 进程是计算机中最小的资源分配单位
- 线程是计算机中被CPU调度的最小单位
开启进程 关闭进程 切换进程都需要时间
- 你的电脑的资源还是有限的
- 开启过多的进程会导致你的计算机崩溃
- count_cpu*2 = 8
- 同时对8个客户端服务
- qps 每秒的请求数 2w
- 2000/8 = 250台机器
数据隔离的
- ftp server端
- qq server端
###并发 :同一时刻能同时接收多个客户端的请求
###线程 - 轻型进程 轻量级的进程
- 在同一个进程中的多个线程是可以共享一部分数据的
- 线程的开启\销毁\切换都比进程要高效很多
多核利用
- 多个进程可不可以利用多核(多个CPU) 可以
- 多个线程可不可以利用多核(多个CPU) 可以
多进程和多线程之间的区别
- 进程 数据隔离 开销大
- 线程 数据共享 开销小
python当中的多线程
- 不能访问多个cpu
- 是Cpython解释器导致的,GIL锁
- GIL锁 = 全局解释器锁,导致了同一时刻只能有一个线程访问CPU
- jpython pypy解释器中的多线程是可以访问多核的
from dis import dis
def func():
a =[]
a.append(1)
dis(func)
python --> 字节码 --> 机器码
python代码机器语言
利用多核 意味着 多个CPU可以同时计算线程中的代码
IO操作
- accept负责从网络上读取数据
- open() 调用操作系统的指令开启一个文件
分布式\多进程
web框架
- django flask tornado twistwed scrapy sanic
threading模块
import threading
threading 和 multiprocessing
先有的threading模块
- 没有池的功能
- multiprocessing完全模仿threading模块完成的
- 实现了池的功能
- concurrent.futures
- 实现了线程池\进程池
import os
import time
from threading import Thread
def func():
time.sleep(1)
print('in func',os.getpid())
print('in main',os.getpid())
for i in range(20):
* func()
Thread(target=func).start()
传参数
import os
import time
from threading import Thread
def func(i):
time.sleep(1)
print('in func',i,os.getpid())
print('in main',os.getpid())
for i in range(20):
* func()
Thread(target=func,args=(i,)).start()
if name == '__main__'在开启线程的时候可以不加
- 在线程部分不需要通过import来为新的线程获取代码
- 因为新的线程和之前的主线程共享同一段代码
- 不需要import 也就不存在在子线程中又重复了一次创建线程的操作
- 所以就不必要if name == '__main__'
开销
import time
from multiprocessing import Process
from threading import Thread
def func(a):
a += 1
if __name__ == '__main__':
start = time.time()
t_l = []
for i in range(100):
t = Thread(target=func,args=(i,))
t.start()
t_l.append(t)
for t in t_l:t.join()
print('thread :',time.time() - start)
start = time.time()
t_l = []
for i in range(100):
t = Process(target=func, args=(i,))
t.start()
t_l.append(t)
for t in t_l: t.join()
print('Process',time.time() - start)
多个线程之间的全局变量是共享的
from threading import Thread
tn = 0
def func():
global tn
tn += 1
t_l = []
for i in range(100):
t = Thread(target=func)
t.start()
t_l.append(t)
for t in t_l:t.join()
print(tn)
进程之间数据隔离
from multiprocessing import Process
pn = 0
def func():
global pn
pn += 1
if __name__ == '__main__':
p_l = []
for i in range(100):
p = Process(target=func)
p.start()
p_l.append(p)
for p in p_l:p.join()
print(pn)
线程中的几个其他方法
import os
from threading import Thread,currentThread
def func():
t = currentThread()
print(t.name,t.ident,os.getpid())
tobj = Thread(target=func)
tobj.start()
print('tobj :',tobj)
t2 = currentThread()
print(t2.name,t2.ident,os.getpid())
lst = [1,2,3,4,5,6,7,8,9,10]
按照顺序把列表中的每一个元素都计算一个平方
使用多线程的方式
并且将结果按照顺序返回
import time
import random
from threading import Thread,currentThread
dic = {}
def func(i):
t = currentThread()
time.sleep(random.random())
dic[t.ident] = i**2
t_lst = []
for i in range(1,11):
t = Thread(target=func,args=(i,))
t.start()
t_lst.append(t)
for t in t_lst:
t.join()
print(dic[t.ident])
from threading import active_count # 返回当前有多少个正在工作的线程
import time
import random
from threading import Thread,currentThread
dic = {}
def func(i):
t = currentThread()
time.sleep(random.random())
dic[t.ident] = i**2
for i in range(10):
Thread(target=func,args=(i,)).start()
print(active_count()) * ???
线程有terminate么?
- 没有terminate 不能强制结束
- 所有的子线程都会在执行完所有的任务之后自动结束
作业
- 通读博客 :操作系统 进程 线程
- 把课上的代码都搞明白
- 多线程实现一个并发的tcp协议的socket server
- 明天默写和线程
2019-04-17-day034-线程与数据共享的更多相关文章
- 2019.04.17 读书笔记 checked与unchecked
在普通的编程中,我们是很容易去分析数据的大小,然后给出合理的类型,但是在很多数据库的累计中,缺存在很多隐患,特别是研发时,数据量小,求和也不会溢出,当程序运行几年后,再来一次大求和,隐形的BUG就出来 ...
- Sqlite && EF Code FIRST 终极解决方案 2019.5.17
Sqlite && EF Code FIRST 终极解决方案 2019.5.17 包括根据模型自动生成数据库,初始化数据,模型改变时的自动数据迁移等 2019.12.25 更新 支持E ...
- 关于asp.net执行exe程序时权限不够的解决办法(2015.04.17更新)
一,本文背景 长话短说:asp.net项目中需要用到PDF转换成SWF文件,用户上传后自动调用pdf2swf.exe转换. 但有个问题,执行时权限不够,导致一直报错(滚动条一直在往下滚,刷屏中),见下 ...
- PowerBI更新2019/04 - 解决方案架构 - PowerBI Solution Architecture(一图胜万字!)
Power BI 架构图 (2019/04) 1) Power BI Desktop 是一个免费的工具.它可以用来准备和管理数据模型:包括链接各种数据:做数据清洗:定义关系:定义度量值和层级关系:应用 ...
- 每日一练ACM 2019.04.13
2019.04.13 第1002题:A+B Proble Ⅱ Problem DescriptionI have a very simple problem for you. Given two in ...
- 2019/01/17 基于windows使用fabric将gitlab的文件远程同步到服务器(git)
觉得django项目把本地更新push到gitlab,再执行fabric脚本从gitlab更新服务器项目挺方便的,当然从本地直接到服务器就比较灵活. 2019/01/17 基于windows使用fab ...
- 最新版IntelliJ IDEA2019.1破解教程(2019.04.08更新)
[原文链接]:https://www.tecchen.xyz/idea-crack.html 我的个人博客:https://www.tecchen.xyz,博文同步发布到博客园. 由于精力有限,对文章 ...
- Python 验证线程是数据共享的
import os import time from threading import Thread # from multiprocessing import Process #通过对全局变量的修改 ...
- 并发编程8 线程的创建&验证线程之间数据共享&守护线程&线程进程效率对比&锁(死锁/递归锁)
1.线程理论以及线程的两种创建方法 2.线程之间是数据共享的与join方法 3.多线程和多进程的效率对比 4.数据共享的补充线程开启太快 5.线程锁 互斥锁 同步锁 6.死锁现象和递归锁 7.守护线程 ...
- Java基础学习总结(17)——线程
一.线程的基本概念 线程理解:线程是一个程序里面不同的执行路径 每一个分支都叫做一个线程,main()叫做主分支,也叫主线程. 程只是一个静态的概念,机器上的一个.class文件,机器上的一个.exe ...
随机推荐
- 【题解】Luogu P3287 [SCOI2014]方伯伯的玉米田
原题传送门 一眼就能看出来这是一道dp题 显而易见每次操作的右端点一定是n,每株玉米被拔高的次数随位置不下降 用f(i,j) 表示以第i 株玉米结尾它被拔高了j 次的最长序列长度. \(f(i,j)= ...
- linux基础之awk
gawk - pattern scanning and processing language 基本用法: gawk [options] 'program' FILE... program: PATT ...
- Windows Server 2008 R2 下载地址
以下资源均来自微软 MSDN,是原汁原味的原版系统资源,值得系统爱好者收藏.以下多数链接是 ed2k 链接,推荐使用国外开源的 eMule 下载,亦可使用迅雷,但使用 eMule 更有利于共享资源. ...
- linux centos7添加ip黑名单禁止某个ip访问
centos7用的是firewall 添加单个黑名单只需要把ip添加到 /etc/hosts.deny 格式 sshd:$IP:deny vim /etc/hosts.deny 添加你要禁止的i ...
- cmd 查看端口
windows + R 调出cmd窗口 输入 netstat -nao 查看端口使用情况 输入netstat -nao | findstr 80查找指定端口 杀死指定端口的进程taskkill /pi ...
- dom常用操作
创建节点:document.createElement(元素名), document.createTextNode(文本内容) 添加节点:parent.appendChild(newChild) 移除 ...
- java jdk动态代理学习记录
转载自: https://www.jianshu.com/p/3616c70cb37b JDK自带的动态代理主要是指,实现了InvocationHandler接口的类,会继承一个invoke方法,通过 ...
- Linux下Java环境安装
本节主要讲解Linux(Centos 6.5)下Java环境的安装 1. 卸载机器上默认安装的JDK 在Linux环境下一般会默认安装jdk,为了自己项目的开发部署,一般情况要重新装jdk,而且自己装 ...
- SAM文件格式
帮朋友处理sam各式文件,又记不住sam各式每列代表的什么内容,干脆转个帖子留着以后查询. 在SAM输出的结果中每一行都包括十二项通过Tab分隔,从左到右分别是: 1 序列的名字 2 概括出一个合适的 ...
- GeoServer服务器环境的搭建
.java 的安装 下载地址:http://www.oracle.com/technetwork/java/javase/downloads/index.html 我下载的java 1.8版本 1. ...