多线程,python中创建多线程模块是Threading模块,Python中Thread是比较底层的模块,Threading是对Thread模块的封装。

         首先使用threading模块来创建线程;

import threading


def say():
    print("-------子线程-------")

if __name__ == "__main__":
    for i in range(5):
        thread = threading.Thread(target=say)  # 创建多线程实例,Thread中target是需要子线程执行的函数
        thread.start()  # 启动多线程

      第二种方式,通过集成的方式来创建线程;

import threading
import time


class test_thread(threading.Thread): # 通过继承threading.Thread重写run方法的形式,创建子线程
    def run(self):
        for i in range(3):
            time.sleep(1)
            msg = "I'm	" + self.name
        print(msg)


def test():
    for i in range(5):
        t = test_thread() # 实例化线程对象
        t.start()


if __name__ == "__main__":
    test()

运行结果:

I'm	Thread-5
I'm	Thread-3
I'm	Thread-4
I'm	Thread-2
I'm	Thread-1

共享全局变量

    线程还有一个特点是,全局变量是共享的,不同于进程全局变量之间是不同享,想要使用被改变的变量,需要采取一定的措施(进程之间的通信),才能共享数据。

线程与进程的区别:


定义的不同

1.进程是系统进⾏资源分配和调度的⼀个独⽴单位.

2.线程是进程的⼀个实体,是CPU调度和分派的基本单位,它是⽐进程更⼩的 能独⽴运⾏的基本单位.线程⾃⼰基本上不拥有系统资源,只拥有⼀点在运 ⾏中必不可少的资源(如程序计数器,⼀组寄存器和栈),但是它可与同属⼀个进程的其他的线程共享进程所拥有的全部资源.

区别

⼀个程序⾄少有⼀个进程,⼀个进程⾄少有⼀个线程.

线程的划分尺度⼩于进程(资源⽐进程少),使得多线程程序的并发性⾼。

进程在执⾏过程中拥有独⽴的内存单元,⽽多个线程共享内存,从⽽极 ⼤地提⾼了程序的运⾏效率

线线程不能够独⽴执⾏,必须依存在进程中

优缺点

线程和进程在使⽤上各有优缺点:线程执⾏开销⼩,但不利于资源的管理和 保护;⽽进程正相反。

线程锁Lock


缺点 就是,线程是对全局变量随意遂改可能造成多线程之间对全局变量 的混乱 (即线程⾮安全)


怎么样解决线程非安全这个问题呢?这里就引出了线程锁的概念

    当多个线程⼏乎同时修改某⼀个共享数据的时候,需要进⾏同步控制
线程同步能够保证多个线程安全访问竞争资源,最简单的同步机制是引⼊互 斥锁。
互斥锁为资源引⼊⼀个状态:锁定/⾮锁定。

某个线程要更改共享数据时,先将其锁定,此时资源的状态为“锁定”,其他 线程不能更改;直到该线程释放资源,将资源的状态变成“⾮锁定”,其他的 线程才能再次锁定该资源。互斥锁保证了每次只有⼀个线程进⾏写⼊操作, 从⽽保证了多线程情况下数据的正确性。

threading模块中定义了Lock类,可以⽅便的处理锁定;

from threading import Thread, Lock
import time

g_num = 0


def test1():
    global g_num
    # 这个线程和test2线程都在抢着 对这个锁 进行上锁,如果有1方成功的上锁,那么导致另外
    # 一方会堵塞(一直等待)到这个锁被解开为止
    mutex.acquire()
    for i in range(1000000):
        g_num += 1
    mutex.release()  # 用来对mutex指向的这个锁 进行解锁,,,只要开了锁,那么接下来会让所有因为
    # 这个锁 被上了锁 而堵塞的线程 进行抢着上锁

    print("---test1---g_num=%d" % g_num)


def test2():
    global g_num
    mutex.acquire()
    for i in range(1000000):
        g_num += 1
    mutex.release()

    print("---test2---g_num=%d" % g_num)


# 创建一把互斥锁,这个锁默认是没有上锁的
mutex = Lock()

p1 = Thread(target=test1)
p1.start()
p2 = Thread(target=test2)
p2.start()

print("---g_num=%d---" % g_num)

运行结果:

C:\Users\Admin\PycharmProjects\BlockChain\venv\Scripts\python.exe C:/Users/Admin/PycharmProjects/BlockChain/Thread/threadingtest.py
---g_num=172105---
---test1---g_num=1000000
---test2---g_num=2000000

  总结:

  锁的好处:
     确保了某段关键代码只能由⼀个线程从头到尾完整地执⾏,这里就不会出现如果一个线程更改完数据后,没有生效时,而第二个线程再次更改数据,避免了多线程共享数据的出错。
锁的坏处:

      阻⽌了多线程并发执⾏,包含锁的某段代码实际上只能以单线程模式执 ⾏,效率就⼤⼤地下降了 由于可以存在多个锁,不同的线程持有不同的锁,并试图获取对⽅持有 的锁时,可能会造成死锁

使用线程互斥锁是解决了多线程共享数据可能出差的问题,但是还有可能会造成死锁。

死锁

     在线程间共享多个资源的时候,如果两个线程分别占有⼀部分资源并且同时 等待对⽅的资源,就会造成死锁。尽管死锁很少发⽣,但⼀旦发⽣就会造成应⽤的停⽌响应。下⾯看⼀个死锁 的例⼦。

import threading
import time


class MyThread1(threading.Thread):
    def run(self):
        if mutexA.acquire():
            print(self.name + '----do1---up----')
            time.sleep(1)
        if mutexB.acquire(): # mutexB在等待MyTherad2中 mutexBjeisu
            print(self.name + '----do1---down----')
        mutexB.release() #释放 mutexB锁
        mutexA.release() #释放 mutexA


class MyThread2(threading.Thread):
    def run(self):
        if mutexB.acquire():
            print(self.name + '----do2---up----')
        time.sleep(1)
        if mutexA.acquire():# mutexA在等待MyTherad1中 mutexA解锁
            print(self.name + '----do2---down----')
        mutexA.release() # 释放mutexA锁
        mutexB.release() # 释放mutexB锁


mutexA = threading.Lock()
mutexB = threading.Lock()

if __name__ == '__main__':
    t1 = MyThread1()
    t2 = MyThread2()
    t1.start()
    t2.start()

运行结果:

C:\Users\Admin\PycharmProjects\BlockChain\venv\Scripts\python.exe C:/Users/Admin/PycharmProjects/BlockChain/Thread/threadingtest.py
Thread-1----do1---up----
Thread-2----do2---up----

通过上面的例子可以看出,双反都在等待对方锁的释放,执行结果就没有再往下执行,这就造成了死锁。程序会一直在处于等待的状态。

怎么样来避免这种死锁的发生?这里引入同步的概念

同步

同步就是让多个进程有序的执行,从而避免死锁的发生。例

from threading import Thread,Lock
from time import sleep

class Task1(Thread):
    def run(self):
        while True:
            if lock1.acquire():
                print("------Task 1 -----")
                sleep(0.5)
                lock2.release()

class Task2(Thread):
    def run(self):
        while True:
            if lock2.acquire():
                print("------Task 2 -----")
                sleep(0.5)
                lock3.release()

class Task3(Thread):
    def run(self):
        while True:
            if lock3.acquire():
                print("------Task 3 -----")
                sleep(0.5)
                lock1.release()

#使用Lock创建出的锁默认没有“锁上”
lock1 = Lock()
#创建另外一把锁,并且“锁上”
lock2 = Lock()
lock2.acquire()
#创建另外一把锁,并且“锁上”
lock3 = Lock()
lock3.acquire()

t1 = Task1()
t2 = Task2()
t3 = Task3()

t1.start()
t2.start()
t3.start()

在创建锁时,将其他两把锁上锁,只让一个线程来执行,执行完成后接着把下一把锁解开,让三个线程有序的执行。