多线程

>多线程

>>多任务

简单来说就是操作系统可以同时运行多个任务

并发:任务数大于CPU核数,通过操作系统的各种任务调度算法,实现多个任务‘一起’执行,比如在单核CPU下需要同时处理3个任务,这就是并发,单核CPU在开启任务一会立马开启任务二,任务三也同理,这段时间内交替执行任务的方式就是并发。此外单核CPU只会存在并发的情况,不存在并行的情况。

并行:就是在同一时刻一起执行多个任务,比如你有8核的CPU,每个CPU执行一个任务,其中任务数<=CPU核数,就可以进行并行的多任务执行

同步:同步是指完成事务的逻辑顺序执行的,即先执行第一个事务,如果阻塞了,会一直等待,直到这个事务完成以后在执行第二个事务。

异步:异步和同步是相对的,异步是指在调用这个事务以后,不会等待这个事务的处理结果,直接处理第二个事务去,通过状态,通知,回调来通知调用者处理结果

>>_thread模块

# _thread 实现多线程
'''
_thread 常用模块
def start_new_thread(function,args,kwargs=None) 启动一个新的线程  function 线程处理函数  args 传递给线程的参数 元组类型 kwargs 可选参数
def allocate_lock() 分配锁对象
def exit() 线程退出
def get_ident() 获取线程表示符
def interrupt_main() 终止主线程 会产生 KeyboardInterrupt异常
'''

# 利用_thread 模块创建多线程
import _thread
import time


def work(thread_name):
    print('哟西 这是第%s个线程' % thread_name)
    time.sleep(1)

def main():
    for i in range(1,5):
        _thread.start_new_thread(work,('--%s'% i,))
    time.sleep(30)  # 主进程添加延迟  保证子线程执行完成
if __name__ == '__main__':
    main()

>>Treading模块

# threading实现多线程
'''
常用方法
def init(self,group=None,target=None,name=None,args=(),kwargs=None,*,daemon=None)
def start(self) 线程启动
def run(self) 线程操作主体 若没设置target 处理函数 则执行方法
def join(self,timeout=None)
def name(self) 获取线程名称
def ident(self) 获取线程标识
def is_alive()  判断线程存活状态
'''

# threading 创建多线程
import threading
import time


def work(thread_name):
    print('哟西 这是第%s个线程' % thread_name)
    time.sleep(1)

def main():

        thraeds=[threading.Thread(target=work,args=('---%s'%i,))for i in range(1,5)]
        for thraed in thraeds:
            thraed.start()
        time.sleep(3)  # 主进程添加延迟  保证子线程执行完成
if __name__ == '__main__':
    main()
    
# 获取线程信息
import threading
import time


def work(thread_name):
    print('哟西 这是第%s个线程' % thread_name)
    print(threading.current_thread().ident, threading.current_thread().name)
    print('')
    time.sleep(1)

def main():

        thraeds=[threading.Thread(target=work,args=('---%s'%i,))for i in range(20)]
        for thraed in thraeds:
            thraed.start()

if __name__ == '__main__':
    main()
# 将线程设置成类  设置守护进程
import threading
import time
class myThread(threading.Thread):
    def __init__(self, threadname, delay, count):
        super().__init__(name=threadname)
        self._delay = delay
        self._count = count

    def run(self):
        for num in range(self._count):
            time.sleep(1)
            print('线程id:%s,线程名称:%s - %s'
                  % (threading.current_thread().ident, threading.current_thread().getName(), num))

def main():
    thr1=myThread('用户进程', 1, 10)
    thr1.start()
    thr2= myThread('守护进程', 1, 999)
    thr2.setDaemon(True)  # 将线程2 设置成守护进程
    thr2.start()

if __name__ == '__main__':
    main()

>>线程同步

# 线程同步&互斥锁
# Semaphore 实现资源并发控制
import threading
import time
def work_handle(sem):
    if sem.acquire():
        print('请%s前往窗口,办理个人业务 '% threading.current_thread().getName())
        time.sleep(3)  # 模拟银行办理业务时间
        sem.release()  # 释放资源
def main():
    sem = threading.Semaphore(2)
    threading_list=[threading.Thread(target=work_handle,args=(sem,),name='客户%s'% num) for num in range(10)]
    for thr in threading_list:
        thr.start()
if __name__ == '__main__':
    main()

>>多线程共享全局变量

# 共享全局变量
import threading
num =0
def thr1():
    global num
    for i in range(0,1000000):
        num = num+1

def thr2():
    global num
    for i in range(0,1000000):
        num = num+1

if __name__ == '__main__':
    for i in range(5):
        num = 0
        t1 = threading.Thread(target=thr1)
        t2 = threading.Thread(target=thr2)
        t1.start()
        t2.start()
        t1.join()
        t2.join()
        print('计算结果为%s' % num)
        
》》》计算结果为1647829
	计算结果为2000000
	计算结果为2000000
	计算结果为1000000
	计算结果为1645209
    # 创建两个线程 各自加了100000次 累计输出应该是200000 调用了5次应该每次的输出应该一样的 为什么不一样呢,应为像个线程同时运行,不知到谁先谁后,累加1时,线程1 刚累加等于1  想赋值给 num 结果 线程2 提前赋值给了num 结果就是1了  累计次数越多 误差越大 

&互斥锁解决资源问题(lock()和Rlock())

# 互斥锁

import threading
num =0

def thr1(lock):
    lock.acquire()
    global num
    for i in range(0,1000000):
        num = num+1
    lock.release()

def thr2(lock):
    lock.acquire()
    global num
    for i in range(0,1000000):
        num = num+1
    lock.release()
if __name__ == '__main__':
    lock = threading.Lock()
    for i in range(5):
        num = 0
        t1 = threading.Thread(target=thr1,args=(lock,))
        t2 = threading.Thread(target=thr2,args=(lock,))
        t1.start()
        t2.start()
        t1.join()
        t2.join()
        print('计算结果为%s' % num)
import threading
num =0

def thr1(lock):
    lock.acquire()
    global num
    for i in range(0,1000000):
        num = num+1
    lock.release()

def thr2(lock):
    lock.acquire()
    global num
    for i in range(0,1000000):
        num = num+1
    lock.release()
if __name__ == '__main__':
    lock = threading.RLock()
    for i in range(5):
        num = 0
        t1 = threading.Thread(target=thr1,args=(lock,))
        t2 = threading.Thread(target=thr2,args=(lock,))
        t1.start()
        t2.start()
        t1.join()
        t2.join()
        print('计算结果为%s' % num)

>>死锁

一直等待对方释放锁的情况就是死锁
# 死锁

import threading

def get_value(index,lock,list1):
    lock.acquire()

    if index >= len(list1):
        print('超出索引范围')

        return
    data = list1[index]
    print(data)
    lock.release()

def main():
    lock = threading.Lock()
    for i in range(10):
        t =threading.Thread(target=get_value,args=(3,lock,[1,2]))
        t.start()

if __name__ == '__main__':
    main()
# 解决死锁问题 可以在acquire()里面添加超时时间 超过时间自动解锁 

# 死锁

import threading

def get_value(index,lock,list1):
    lock.acquire(timeout=2)

    if index >= len(list1):
        print('超出索引范围')

        return
    data = list1[index]
    print(data)
    lock.release()

def main():
    lock = threading.Lock()
    for i in range(10):
        t =threading.Thread(target=get_value,args=(3,lock,[1,2]))
        t.start()

if __name__ == '__main__':
    main()

>>定时调度

# 定时调度是一种基于线程任务的操作管理 可以实现在末段时间的任务的重复执行

# 通过sched模块实现定时调度操作
'''
delay=0 调度任务启动的延迟时间 如果是0 表示立即启动
priority=0 多个调度任务的启动优先级 
action=func 设置任务的调度处理函数
argument = (参数,) 调度函数的参数 必须填可迭代对象  
'''
import sched


def work_1():
    print('我是第一个任务')

def work_2():
    print('我是第二个任务')

def work_3():
    print('我是第三个任务')

def main():
    sch =sched.scheduler()
    s1=sch.enter(delay=2,priority=1,action=work_1,)
    s2= sch.enter(delay=6, priority=3, action=work_2)
    s3 = sch.enter(delay=4, priority=2, action=work_3)
    sch.run()
if __name__ == '__main__':

    main()
posted @ 2022-04-11 23:53  饭兜  阅读(35)  评论(0编辑  收藏  举报