头哥Python魔鬼训练:线程同步之报数

任务描述

排队报数是现实生活中很常见的一种统计人数的手段。它需要队列中的每个人互相协作,后一名比前一名报的数字多一,最后一名报的数字即为队列的总人数。 本关利用线程模拟队列中的个人,最终实现一个队列报数的过程。

编程要求

本关的编程任务是,补全右侧编辑器中BeginEnd区间的代码,具体要求如下:

  • 本关将根据测试输入创建多个线程,每个线程相当于队列中的一个人,他们报的数用全局变量x存储;
  • 学员需要编写run()方法,使得每个线程将自己该报的数输出;
  • 注意在输出语句之前,加入time.sleep(0.1)防止输出过快造成顺序混乱的情况。
测试说明

测试过程:

  • 平台将运行用户补全的代码文件,并生成若干组测试数据;

  • 接着根据程序的输出判断程序是否正确。

以下是测试样例:

测试输入:10 预期输出:12345678910

先上答案,后面详细讲解

# -*- coding: utf-8 -*-

import threading
import time
lock = threading.Lock()
class mythread(threading.Thread):
    x = 0
    def __init__(self):
        threading.Thread.__init__(self)
    def run(self):
        global x
    #*********begin*********#
        lock.acquire()
        mythread.x += 1
        print(mythread.x,end='')
        time.sleep(0.1)
        lock.release()
    #********* end*********#
num = input()
t1 = []
for i in range(int(num)):
    t = mythread()
    t1.append(t)
x = 0
for i in t1:
    i.start()

lock = threading.Lock()
创建了一个Lock对象,它是一种线程同步机制,用于控制多个线程对共享资源的访问。


class mythread(threading.Thread):
    x = 0
    def __init__(self):
        threading.Thread.__init__(self)

定义了一个继承自`threading.Thread`的子类`mythread`,并初始化了一个类属性`x`为0,该属性将作为全局变量来使用。`__init__`方法调用了父类的构造函数。


    def run(self):
        global x
        lock.acquire()
        mythread.x += 1
        print(mythread.x)
        time.sleep(0.1)
        lock.release()

这是线程类的`run`方法,它定义了线程的执行逻辑:
1) 声明`x`为全局变量
2) `lock.acquire()`获取锁对象,确保同一时间只有一个线程可以执行该代码块
3) `mythread.x += 1`将全局变量`x`加1
4) `print(mythread.x)`打印新的`x`值
5) `time.sleep(0.1)`等待0.1秒,防止输出过快导致混乱
6) `lock.release()`释放锁对象,允许其他线程执行该代码块


num = input()
t1 = []
for i in range(int(num)):
    t = mythread()
    t1.append(t)

1) 用户输入一个数字`num`
2) 创建一个空列表`t1`
3) 根据输入的`num`创建相应数量的`mythread`实例,并添加到`t1`列表中


x = 0
for i in t1:
    i.start()
1) 重置全局变量`x`为0 
2) 遍历`t1`列表,对每个线程实例调用`start()`方法启动线程

总的来说,这段代码模拟了一个队列报数的场景。用户输入一个数字,代表队列中人的数量。程序根据输入创建相应数量的线程实例,每个线程相当于队列中的一个人。线程的`run`方法通过获取锁、修改全局变量`x`和打印`x`的值来模拟报数过程。最终输出的就是按顺序报出的数字序列。

  • 8
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python 中,我们可以使用 threading 模块来创建多线程程序。线程同步是指多个线程之间协调工作,以避免竞态条件(race condition)和死锁(deadlock)等问题。其中一个经典的例子就是报数问题。 假设有两个线程 A 和 B,它们交替报数,从 1 开始,每次加 1,直到 10,然后停止。线程 A 报数时输出 "A: 1","A: 2","A: 3",...,线程 B 报数时输出 "B: 1","B: 2","B: 3",...,并且 A 和 B 的报数交替进行。 下面是一种使用 threading.Condition 来实现线程同步的方法: ```python import threading class Counter: def __init__(self): self.count = 0 self.cond = threading.Condition() def inc(self): with self.cond: while self.count >= 10: self.cond.wait() self.count += 1 print(threading.current_thread().name, ':', self.count) self.cond.notify_all() def worker(counter): for i in range(10): counter.inc() counter = Counter() t1 = threading.Thread(target=worker, args=(counter,)) t2 = threading.Thread(target=worker, args=(counter,)) t1.start() t2.start() t1.join() t2.join() ``` 在这个例子中,Counter 类封装了一个计数器和一个 Condition 对象。每个线程运行 worker 函数,该函数通过 counter.inc() 方法来递增计数器。在 inc() 方法中,使用 with self.cond: 语句来获取 Condition 锁,并在 while 循环中等待计数器小于 10。如果计数器已经达到 10,当前线程会调用 self.cond.wait() 方法进入等待状态,直到其他线程通过 self.cond.notify_all() 方法来唤醒它。如果计数器小于 10,当前线程会递增计数器,并输出当前线程名和计数器值,然后调用 self.cond.notify_all() 方法来唤醒其他线程。最后,主线程等待两个工作线程结束后退出。 这样,就可以实现线程同步,保证两个线程交替报数,避免了竞态条件和死锁问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值