美文网首页
day19-多线程

day19-多线程

作者: 2333_11f6 | 来源:发表于2018-11-29 17:44 被阅读0次

    19.1 耗时操作

    一个进程默认有一个线程,这个线程叫主线程。默认情况下,所有的代码都是在主线程中执行的。

    import time,datetime
    
    
    def download(file_name):
        print('开始下载:%s' % file_name, datetime.datetime.now())
        time.sleep(5)          # 程序执行到这会停5秒,再执行后面的代码
        print('%s下载结束' % file_name, datetime.datetime.now())
    
    
    
    # 在主线程中下载俩个电影
    download('小黄人')
    
    download('地心游记')
    

    19.2 多线程

    python中提供threading模块来支持多线程技术

    默认创建的线程叫主线程,其它的线程叫子线程,如果希望代码在子线程中执行,必须手动创建线程对象。

    import threading
    import time,datetime
    
    
    def download(file_name):
        print('开始下载:%s' % file_name, datetime.datetime.now())
        time.sleep(5)          # 程序执行到这会停5秒,再执行后面的代码
        print('%s下载结束' % file_name, datetime.datetime.now())
        print(threading.current_thread())  # 当前线程
    
    
    # # 在主线程中下载俩个电影
    # download('小黄人')    # <_MainThread(MainThread, started 14148)>
    #
    # download('地心游记') # <_MainThread(MainThread, started 14148)>
    

    1.创建线程对象
    a.Thread - 线程类
    b.Thread(target=函数名,args=参数列表) - 直接创建线程对象,返回线程对象
    函数名 - 需要在当前创建的子线程中执行的函数变量
    参数列表 - 元组,元组的元素是函数的参数元组,注意参数只要1个时别忘了逗号。

    t1 = threading.Thread(target=download, args=('小黄人', ))
    t2 = threading.Thread(target=download, args=('地心游记', ))
    

    2.在子线程中执行任务
    在这就是调用t1对应的线程中调用download函数,并传1个参数’小黄人‘

    t1.start()            # <Thread(Thread-1, started 16228)>
    t2.start()            # <Thread(Thread-2, started 15660)>
    

    19.3 线程类子类

    from threading import Thread,current_thread
    

    创建子线程除了直接创建Thread对象,还可以创建这个类的子类对象

    注意:一个进程有多个线程,进程会在所有的线程结束后才结束

    # 1.声明一个类,继承Thread
    
    
    class DownloadThread(Thread):
        # 想要给run方法传值,通过添加对象属性来传
        def __init__(self, filename):
            super().__init__()
            self.filename = filename
    
        # 重写run方法
        def run(self):
            # 这里的代码会在子线程中执行
            print('run')
            print(current_thread())
    
    
    # 3.创建线程对象
    t1 = DownloadThread('789')
    # 4.通过线程对象调用start在子线程中执行run方法
    t1.start()      # <DownloadThread(Thread-1, started 17764)>
    
    t1.run()  # 主线程执行<_MainThread(MainThread, started 18984)>
    

    19.4 join函数

    线程对象.join() - 等待线程对象执行完成

    import time,datetime,random
    from threading import Thread
    
    
    class DownThread(Thread):
        def __init__(self, file_name):
            super().__init__()
            self.file_name = file_name
    
        def run(self):
            print('开始下载:%s' % self.file_name)
            t = random.randint(5, 12)
            time.sleep(t)
            print('结束下载:%s' % self.file_name)
            print('耗时%d秒' % t)
    
    
    if __name__ == '__main__':
        t1 = DownThread('小黄人')
        t2 = DownThread('黄人')
    
        time1 = time.time()
        t1.start()
        t2.start()
    
        t1.join()
        t2.join()
        # t1和t2都执行完成后才执行 time2
        time2 = time.time()
        print('总用时%d秒' % (time2-time1))
    

    19.5 数据共享

    import time
    from threading import Thread
    import threading
    

    当多个线程同时对一个数据进行操作时,可能会出现数据混乱

    多个线程对一个数据进行操作,一个线程将数据读出来,还没来得及存进去,另一个线程又去调用该数据,这时可能产生数据安全隐患。

    解决方式是给数据加锁

    Thread - 创建子线程
    Lock - 锁;创建锁对象

    
    
    class Account(object):
        def __init__(self, balance, name):
            self.balance = balance              # 余额
            self.name = name
            self.lock = threading.Lock()      # 创建锁对象
    
        def save(self, num):
            print(self.name, '开始存钱')
            # 加锁
            self.lock.acquire()
            old_balance = self.balance
            time.sleep(3)
            self.balance = old_balance + num
            # 解锁
            self.lock.release()
    
        def draw(self, num):
            print(self.name, '开始取钱')
            # 加锁
            self.lock.acquire()
            old_balance = self.balance
            time.sleep(3)
            self.balance = old_balance - num
            # 解锁
            self.lock.release()
    
    
    account1 = Account(1000, '小黄人')
    
    # 支付宝存钱
    t1 = Thread(target=account1.save, args=(1000, ))
    # 银行卡取钱
    t2 = Thread(target=account1.draw, args=(500, ))
    
    t1.start()
    t2.start()
    
    t1.join()
    t2.join()
    
    print(account1.balance)
    

    相关文章

      网友评论

          本文标题:day19-多线程

          本文链接:https://www.haomeiwen.com/subject/gpvfcqtx.html