1 基本概述

采用threading.Lock()对象创建锁lock,即 lock = threading.Lock()

其相关方法主要有

lock.acquire() # lock the lock, possibly blocking until it can be obtained
lock.release() # unlock of the lock
lock.locked() # test whether the lock is currently locked

Lock(指令锁)是可用的最低级的同步指令;Lock处于锁定状态时,不被特定的线程拥有。

2 示例1

2.1 异常分析

import threading
import time

a = b =0

def value():
    while True:
        if a != b :
            print("a = %d,b =%d"%(a,b))

t=threading.Thread(target=value)
t.start()
while True:
    a += 1
    b += 1

t.join()

运行结果为一串数字

...
a = 3361552,b =3361551
a = 3361552,b =3361551
a = 3361552,b =3361551
a = 3361552,b =3361551
a = 3361552,b =3361551
...

说明:

(1)按照程序分析来讲,应该没有输出,

(2)打印出好多相同的数字

可能原因

(1)首先明确a b为全局变量,所以当全局变量a b的值发生变化时,进程中的a b值也会随之变化

(2)当主线程中while True 使的a b均+1 时,线程函数中的a b值也会发生变化;但是主线程中的a b值赋值过程有时间间隔, 而线程中没有阻塞函数,所以在a b赋值中间的一个时刻,存在有a != b 这个过程,满足了线程函数的打印条件;所以会有数值打印出来

(3)在运行过程中,CPU高达100%,此时while 赋值运行和线程函数的运行并不是完全一致的,所以出现了数值较大的波动,有的数值仅出现一次或几次,有的数值甚至出现几十次。

(4)会出现当a赋值,当b没有赋值的间隙,子线程执行打印到桌面

2.2 添加sleep时间

import threading
import time

a = b =0

def value():
    while True:
        if a != b :
            print("a = %d,b =%d"%(a,b))

t=threading.Thread(target=value)
t.start()
while True:
    time.sleep(1) #添加sleep(1)时间
    a += 1
    b += 1

t.join()

运行

a = 3,b =3
a = 6,b =6
a = 15,b =15
a = 16,b =16
a = 17,b =17
a = 18,b =18
a = 19,b =19
a = 20,b =20

说明:

(1)打印出的数据之间的间隔不是“等差”关系

(2)运行过程可能是,

(a)a+1 赋值后,b尚未赋值,中间时间内线程判定出 a != b 为 True

(b)在print打印前,b+1赋值已经完成,而sleep(1)后,循环赋值中的a+1被阻塞;

(c)所以此时打印出的 a b值想等。

(3)为了说明 2 的过程,在b += 1 前添加 sleep(1),则打印出a的值比b的值均大1;且a b相同值行数达上百行

(4)从 2 和 3 过程可以说明线

2.3 颠倒交换

import threading
import time

a = b =0

def value():
    while True:
        # 不添加global关键词,
        # 则会 UnboundLocalError : local variable 'a' referenced before assignment
        # 也即在赋值前没有定义局部变量 a b
        global a
        global b

        a += 1
        b += 1

t=threading.Thread(target=value)
t.start()
while True:
    if a != b:
        print("a = %d,b =%d" % (a, b))

t.join()

说明:

该出a b本是全局变量,而当不添加关键字global时则会出现未定义局部变量的报错。——暂时还没有搞懂为什么!

3 示例2

使用进程锁Lock

import threading

a = b =0

# 创建线程锁对象
lock = threading.Lock()

def value():
    while True:
        lock.acquire()
        if a != b :
          print("a = %d,b =%d"%(a,b))
        lock.release()

t=threading.Thread(target=value)
t.start()
while True:
    lock.acquire()
    a += 1
    b += 1
    lock.release()

t.join()

运行,文件一直没有打印出内容

说明

(1)在a b 赋值时加锁,另外一个再加锁,这时候才会阻塞,此时不会打印。

(2)不管是子线程中的while循环 ,还是主线程中的while循环,任意一个去掉 lock.acquire  lock.release 后,均不会出现阻塞。均会打印输出内容。

pythonl练习笔记——threading创建进程锁Lock()的更多相关文章

  1. python从入门到放弃之进程锁lock

    # ### lock (互斥锁)"""# 应用在多进程当中# 互斥锁lock : 互斥锁是进程间的get_ticket互相排斥进程之间,谁先抢占到资源,谁就先上锁,等到解 ...

  2. 13.1、多进程:进程锁Lock、信号量、事件

    进程锁: 为什么要有进程锁:假如现在有一台打印机,qq要使用打印机,word文档也要使用打印机,如果没有使用进程锁,可能会导致一些问题,比如QQ的任务打印到一半,Word插进来,于是打印出来的结果是各 ...

  3. python学习笔记——多进程中的锁Lock

    1 进程锁 python编程中,引入了对象互斥锁的概念,来保证共享数据操作的完整性. 每个对象都对应于一个可称为“互斥锁”的标记,这个标记用来保证在任一时刻,只能有一线程访问对象. 在python中我 ...

  4. 进程锁Lock

    例1: 10个人去买票,先去查询余票,有票就去购买.代码如下: # ticket.py {"ticket": 1} # 只有一张票 import json import time ...

  5. 进程锁 Lock

    进程锁其实是锁的输入终端,屏幕,太多的进程,为了防止打印错乱而锁,作用不大 from multiprocessing import Process, Lock def f(l, i): l.acqui ...

  6. pythonl练习笔记——threading线程中的事件Event

    1 事件Event 使用方法:e = threading.Event() Event对象主要用于线程间通信,确切地说是用于主线程控制其他线程的执行. Event事件提供了三个方法:wait等待.cle ...

  7. python 多进程锁Lock和共享内存

    多进程锁 lock = multiprocessing.Lock() 创建一个锁 lock.acquire() 获取锁 lock.release() 释放锁 with lock: 自动获取.释放锁 类 ...

  8. python线程互斥锁Lock(29)

    在前一篇文章 python线程创建和传参 中我们介绍了关于python线程的一些简单函数使用和线程的参数传递,使用多线程可以同时执行多个任务,提高开发效率,但是在实际开发中往往我们会碰到线程同步问题, ...

  9. python笔记9 线程进程 threading多线程模块 GIL锁 multiprocessing多进程模块 同步锁Lock 队列queue IO模型

    线程与进程 进程 进程就是一个程序在一个数据集上的一次动态执行过程.进程一般由程序.数据集.进程控制块三部分组成.我们编写的程序用来描述进程要完成哪些功能以及如何完成:数据集则是程序在执行过程中所需要 ...

随机推荐

  1. 我所遭遇过的游戏中间件--Scaleform

    我所遭遇过的游戏中间件---Scaleform Scaleform帮助开发人员利用现代系统的三维硬件加速性能创建电影品质的菜单.游戏内HUD,动画纹理.迷你游戏以及移动游戏与应用.Scaleform作 ...

  2. SonarQube代码质量管理平台 的安装、配置与使用

    SonarQube是管理代码质量一个开放平台,可以快速的定位代码中潜在的或者明显的错误,下面将会介绍一下这个工具的安装.配置以及使用. 准备工作: 1.jdk(不再介绍) 2.sonarqube:ht ...

  3. 添加sqljdbc4的maven依赖

    sqljdbc是微软sql server的jdbc驱动 使用sqljdbc需要从微软的官方网站下载jar包: http://www.microsoft.com/en-us/download/detai ...

  4. 我对REST的理解

    1:rest的由来 REST即表述性状态传递(英文:Representational State Transfer,简称REST) 通俗点说:资源在网络中以某种表现形式进行状态转移. 源于REST之父 ...

  5. UML建模学习1:UML统一建模语言简单介绍

    一什么是UML? Unified Modeling Language(UML又称为统一建模语言或标准建模语言)是国际对象管理组织OMG制定的一个通 用的.可视化建模语言标准.能够用来描写叙述(spec ...

  6. Jacoco覆盖率工具使用调研

    JaCoCo Java Code Coverage Library Jacoco是一个开源的覆盖率工具.Jacoco可以嵌入到Ant .Maven中,并提供了EclEmma Eclipse插件,也可以 ...

  7. 【js】用正则表达式对文字进行局部替换

    比如有个字符串http://www.55188.com/thread-8306254-2-3.html,需要把8306254后面的2替换成其它数字,其它保持不变,该如何办呢?请看代码: var url ...

  8. Linux - 系统路径加入命令

    系统路径加入命令 本文地址: http://blog.csdn.net/caroline_wendy 1. 在根文件夹建立bin文件夹: $mkdir bin 2. 改动".bash_pro ...

  9. Discuz常见小问题-如何快速安装和配置

    下载PHPNOW 可以解压到本地的某个目录,最好不要有中文路径,然后查看Readme进行安装,双击Setup.cmd 安装结束之后,会要求输入一个初始化的密码,不要忘记,会自动弹出一个测试页面,可以测 ...

  10. 理解进程调度时机跟踪分析进程调度与进程切换的过程(Linux)

    ----------------------------------------------------------------------------------- 理解进程调度时机跟踪分析进程调度 ...