- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章详解python多线程、锁、event事件机制的简单使用由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
线程和进程 。
1、线程共享创建它的进程的地址空间,进程有自己的地址空间 。
2、线程可以访问进程所有的数据,线程可以相互访问 。
3、线程之间的数据是独立的 。
4、子进程复制线程的数据 。
5、子进程启动后是独立的 ,父进程只能杀掉子进程,而不能进行数据交换 。
6、修改线程中的数据,都是会影响其他的线程,而对于进程的更改,不会影响子进程 。
threading.Thread 。
Thread 是threading模块中最重要的类之一,可以使用它来创建线程。有两种方式来创建线程:一种是通过继承Thread类,重写它的run方法;另一种是创建一个threading.Thread对象,在它的初始化函数(__init__)中将可调用对象作为参数传入。 先来看看通过继承threading.Thread类来创建线程的例子:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
import
threading
import
time
class
MyThread(threading.Thread):
def
__init__(
self
, arg):
# super(MyThread, self).__init__() # 新式类继承原有方法写法
threading.Thread.__init__(
self
)
self
.arg
=
arg
def
run(
self
):
time.sleep(
2
)
print
(
self
.arg)
for
i
in
range
(
10
):
thread
=
MyThread(i)
print
(thread.name)
thread.start()
|
另外一种创建线程的方法:
1
2
3
4
5
6
7
8
9
10
11
|
import
threading
import
time
def
process(arg):
time.sleep(
2
)
print
(arg)
for
i
in
range
(
10
):
t
=
threading.Thread(target
=
process, args
=
(i,))
print
(t.name)
t.start()
|
Thread类还定义了以下常用方法与属性:
Thread.getName() 获取线程名称 。
Thread.setName() 设置线程名称 。
Thread.name 线程名称 。
Thread.ident 获取线程的标识符。线程标识符是一个非零整数,只有在调用了start()方法之后该属性才有效,否则它只返回None 。
判断线程是否是激活的(alive)。从调用start()方法启动线程,到run()方法执行完毕或遇到未处理异常而中断 这段时间内,线程是激活的 。
Thread.is_alive() Thread.isAlive() 。
Thread.join([timeout]) 调用Thread.join将会使主调线程堵塞,直到被调用线程运行结束或超时。参数timeout是一个数值类型,表示超时时间,如果未提供该参数,那么主调线程将一直堵塞到被调线程结束 。
Python GIL(Global Interpreter Lock) 。
GIL并不是Python的特性,它是在实现Python解析器(CPython)时所引入的一个概念。就好比C++是一套语言(语法)标准,但是可以用不同的编译器来编译成可执行代码。有名的编译器例如GCC,INTEL C++,Visual C++等。Python也一样,同样一段代码可以通过CPython,PyPy,Psyco等不同的Python执行环境来执行。像其中的JPython就没有GIL。然而因为CPython是大部分环境下默认的Python执行环境。所以在很多人的概念里CPython就是Python,也就想当然的把GIL归结为Python语言的缺陷。所以这里要先明确一点:GIL并不是Python的特性,Python完全可以不依赖于GIL.
线程锁的使用:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
# 锁:GIL 全局解释器 它是为了保证线程在运行过程中不被抢占
number
=
0
lock
=
threading.RLock()
# 创建锁
def
run(num):
lock.acquire()
# 加锁
global
number
number
+
=
1
print
(number)
time.sleep(
2
)
lock.release()
# 释放锁
for
i
in
range
(
10
):
t
=
threading.Thread(target
=
run, args
=
(i, ))
t.start()
|
Join & Daemon 。
主线程A中,创建了子线程B,并且在主线程A中调用了B.setDaemon(),这个的意思是,把主线程A设置为守护线程,这时候,要是主线程A执行结束了,就不管子线程B是否完成,一并和主线程A退出.这就是setDaemon方法的含义,这基本和join是相反的。此外,还有个要特别注意的:必须在start() 方法调用之前设置,如果不设置为守护线程,程序会被无限挂起.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
class
MyThread1(threading.Thread):
def
__init__(
self
):
threading.Thread.__init__(
self
)
def
run(
self
):
print
(
"thread start"
)
time.sleep(
3
)
print
(
'thread end'
)
print
(
'main start'
)
thread1
=
MyThread1()
# thread1.setDaemon(True) # 设置子线程是否跟随主线程一起结束
thread1.start()
time.sleep(
1
)
print
(
'satrt join'
)
# thread1.join() # 使主线程阻塞,直至子线程运行完毕再继续主线程
print
(
'end join'
)
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
def
run(n):
print
(
'[%s]------running----\n'
%
n)
time.sleep(
2
)
print
(
'--done--'
)
def
main():
for
i
in
range
(
5
):
t
=
threading.Thread(target
=
run, args
=
[i,])
t.start()
# t.join()
print
(
'starting thread'
, t.getName())
m
=
threading.Thread(target
=
main,args
=
[])
# m.setDaemon(True) # 将主线程设置为Daemon线程,它退出时,其它子线程会同时退出,不管是否执行完任务
m.start()
# m.join() # 使主线程阻塞,直至子线程运行完毕再继续主线程
print
(
"---main thread done----"
)
|
线程锁(互斥锁Mutex) 。
一个进程下可以启动多个线程,多个线程共享父进程的内存空间,也就意味着每个线程可以访问同一份数据,此时,如果2个线程同时要修改同一份数据,会出现什么状况?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
num
=
100
# 设定一个共享变量
def
subNum():
global
num
# 在每个线程中都获取这个全局变量
print
(
'--get num:'
, num)
time.sleep(
2
)
num
-
=
1
# 对此公共变量进行-1操作
thread_list
=
[]
for
i
in
range
(
100
):
t
=
threading.Thread(target
=
subNum)
t.start()
thread_list.append(t)
for
t
in
thread_list:
# 等待所有线程执行完毕
t.join()
print
(
'final num:'
, num)
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
# 加锁版本
def
subNum():
global
num
# 在每个线程中都获取这个全局变量
print
(
'--get num:'
, num)
time.sleep(
1
)
lock.acquire()
# 修改数据前加锁
num
-
=
1
# 对此公共变量进行-1操作
lock.release()
# 修改后释放
num
=
100
# 设定一个共享变量
thread_list
=
[]
lock
=
threading.Lock()
# 生成全局锁
for
i
in
range
(
100
):
t
=
threading.Thread(target
=
subNum)
t.start()
thread_list.append(t)
for
t
in
thread_list:
# 等待所有线程执行完毕
t.join()
print
(
'final num:'
, num)
|
Rlock与Lock的区别:
RLock允许在同一线程中被多次acquire。而Lock却不允许这种情况。否则会出现死循环,程序不知道解哪一把锁。注意:如果使用RLock,那么acquire和release必须成对出现,即调用了n次acquire,必须调用n次的release才能真正释放所占用的锁 。
Events 。
Python提供了Event对象用于线程间通信,它是由线程设置的信号标志,如果信号标志位真,则其他线程等待直到信号接触。 Event对象实现了简单的线程通信机制,它提供了设置信号,清除信号,等待等用于实现线程间的通信.
event = threading.Event() 创建一个event 。
1 设置信号 event.set() 。
使用Event的set()方法可以设置Event对象内部的信号标志为真。Event对象提供了isSet()方法来判断其内部信号标志的状态。 当使用event对象的set()方法后,isSet()方法返回真 。
2 清除信号 event.clear() 。
使用Event对象的clear()方法可以清除Event对象内部的信号标志,即将其设为假,当使用Event的clear方法后,isSet()方法返回假 。
3 等待 event.wait() 。
Event对象wait的方法只有在内部信号为真的时候才会很快的执行并完成返回。当Event对象的内部信号标志位假时, 则wait方法一直等待到其为真时才返回。也就是说必须set新号标志位真 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
def
do(event):
print
(
'start'
)
event.wait()
print
(
'execute'
)
event_obj
=
threading.Event()
for
i
in
range
(
10
):
t
=
threading.Thread(target
=
do, args
=
(event_obj,))
t.start()
event_obj.clear()
inp
=
input
(
'输入内容:'
)
if
inp
=
=
'true'
:
event_obj.
set
()
|
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
原文链接:https://segmentfault.com/a/1190000014619654 。
最后此篇关于详解python多线程、锁、event事件机制的简单使用的文章就讲到这里了,如果你想了解更多关于详解python多线程、锁、event事件机制的简单使用的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我在网上搜索但没有找到任何合适的文章解释如何使用 javascript 使用 WCF 服务,尤其是 WebScriptEndpoint。 任何人都可以对此给出任何指导吗? 谢谢 最佳答案 这是一篇关于
我正在编写一个将运行 Linux 命令的 C 程序,例如: cat/etc/passwd | grep 列表 |剪切-c 1-5 我没有任何结果 *这里 parent 等待第一个 child (chi
所以我正在尝试处理文件上传,然后将该文件作为二进制文件存储到数据库中。在我存储它之后,我尝试在给定的 URL 上提供文件。我似乎找不到适合这里的方法。我需要使用数据库,因为我使用 Google 应用引
我正在尝试制作一个宏,将下面的公式添加到单元格中,然后将其拖到整个列中并在 H 列中复制相同的公式 我想在 F 和 H 列中输入公式的数据 Range("F1").formula = "=IF(ISE
问题类似于this one ,但我想使用 OperatorPrecedenceParser 解析带有函数应用程序的表达式在 FParsec . 这是我的 AST: type Expression =
我想通过使用 sequelize 和 node.js 将这个查询更改为代码取决于在哪里 select COUNT(gender) as genderCount from customers where
我正在使用GNU bash,版本5.0.3(1)-发行版(x86_64-pc-linux-gnu),我想知道为什么简单的赋值语句会出现语法错误: #/bin/bash var1=/tmp
这里,为什么我的代码在 IE 中不起作用。我的代码适用于所有浏览器。没有问题。但是当我在 IE 上运行我的项目时,它发现错误。 而且我的 jquery 类和 insertadjacentHTMl 也不
我正在尝试更改标签的innerHTML。我无权访问该表单,因此无法编辑 HTML。标签具有的唯一标识符是“for”属性。 这是输入和标签的结构:
我有一个页面,我可以在其中返回用户帖子,可以使用一些 jquery 代码对这些帖子进行即时评论,在发布新评论后,我在帖子下插入新评论以及删除 按钮。问题是 Delete 按钮在新插入的元素上不起作用,
我有一个大约有 20 列的“管道分隔”文件。我只想使用 sha1sum 散列第一列,它是一个数字,如帐号,并按原样返回其余列。 使用 awk 或 sed 执行此操作的最佳方法是什么? Accounti
我需要将以下内容插入到我的表中...我的用户表有五列 id、用户名、密码、名称、条目。 (我还没有提交任何东西到条目中,我稍后会使用 php 来做)但由于某种原因我不断收到这个错误:#1054 - U
所以我试图有一个输入字段,我可以在其中输入任何字符,但然后将输入的值小写,删除任何非字母数字字符,留下“。”而不是空格。 例如,如果我输入: 地球的 70% 是水,-!*#$^^ & 30% 土地 输
我正在尝试做一些我认为非常简单的事情,但出于某种原因我没有得到想要的结果?我是 javascript 的新手,但对 java 有经验,所以我相信我没有使用某种正确的规则。 这是一个获取输入值、检查选择
我想使用 angularjs 从 mysql 数据库加载数据。 这就是应用程序的工作原理;用户登录,他们的用户名存储在 cookie 中。该用户名显示在主页上 我想获取这个值并通过 angularjs
我正在使用 autoLayout,我想在 UITableViewCell 上放置一个 UIlabel,它应该始终位于单元格的右侧和右侧的中心。 这就是我想要实现的目标 所以在这里你可以看到我正在谈论的
我需要与 MySql 等效的 elasticsearch 查询。我的 sql 查询: SELECT DISTINCT t.product_id AS id FROM tbl_sup_price t
我正在实现代码以使用 JSON。 func setup() { if let flickrURL = NSURL(string: "https://api.flickr.com/
我尝试使用for循环声明变量,然后测试cols和rols是否相同。如果是,它将运行递归函数。但是,我在 javascript 中执行 do 时遇到问题。有人可以帮忙吗? 现在,在比较 col.1 和
我举了一个我正在处理的问题的简短示例。 HTML代码: 1 2 3 CSS 代码: .BB a:hover{ color: #000; } .BB > li:after {
我是一名优秀的程序员,十分优秀!