1,Thread Local类的用法     

     python 中多线程的并发跟其他语言一样,需要考虑多线程并发访问去全局变量所带来的问题,python的local类解决了这个问题,通过它让每个线程内部有一个相对独立的local保存数据,某一个线程修改了数据,不影响其他线程中保存的数据。



1 from threading import Thread
 2 import threading
 3 import time
 4 local_value=threading.local()
 5 local_value.x='en'
 6 class threadtest(Thread):
 7 
 8      def __init__(self,num):
 9          Thread.__init__(self)
10          self.num=num
11          
12      def run(self):
13          local_value.x=self.getName()
14          for i in range(self.num):
15              time.sleep(1)
16              print(local_value)
17          
18 
19 threadtest(3).start()
20 threadtest(2).start()



执行结果是:



Thread-1
Thread-2
Thread-1
Thread-2
Thread-1
[Finished in 3.2s]



这表明,虽然两个线程公用一个local_value.x但是确实相互独立的变量。

另外,



local_value.x=self.getName()放在__init__()中时,运行会报错,说'_thread._local' object has no attribute 'x',目测是因为在初始化__init__的时候,线程还没有建立local_value对象,只有在__init__执行完之后,才可以开始赋值(不过对于这一点我不确定,看源码也看晕了)。



2,Thread join()方法的用法及含义:

join()源码中的解释:



"""Wait until the thread terminates.

        This blocks the calling thread until the thread whose join() method is
        called terminates -- either normally or through an unhandled exception
        or until the optional timeout occurs.

        When the timeout argument is present and not None, it should be a
        floating point number specifying a timeout for the operation in seconds
        (or fractions thereof). As join() always returns None, you must call
        isAlive() after join() to decide whether a timeout happened -- if the
        thread is still alive, the join() call timed out.

        When the timeout argument is not present or None, the operation will
        block until the thread terminates.

        A thread can be join()ed many times.

        join() raises a RuntimeError if an attempt is made to join the current
        thread as that would cause a deadlock. It is also an error to join() a
        thread before it has been started and attempts to do so raises the same
        exception.

        """



当某个子线程调用join()时,主线程将阻塞,直至该线程执行完毕,

当每个子线程调用都join(),且不设置参数时



from threading import Thread
import threading
import time
local_value=threading.local()
class threadtest(Thread):
    
     def __init__(self,num):
         Thread.__init__(self)
         self.num=num
         #local_value.x=self.getName()
     def run(self):
         local_value.x=self.getName()
         for i in range(self.num):
             time.sleep(1)
             print(local_value.x)
from threading import current_thread
        
#print(current_thread(),local_value.__dict__,'--')
a=threadtest(2)
b=threadtest(2)
c=threadtest(2)
a.start()
a.join()
b.start()
b.join()        
c.start()
c.join()
for i in range(2):
    time.sleep(1)
    print(current_thread())



结果是:



Thread-1
Thread-1
Thread-2
Thread-2
Thread-3
Thread-3
<_MainThread(MainThread, started 6688)>
<_MainThread(MainThread, started 6688)>
[Finished in 8.2s]



即:哪个子线程先调用join,就先完成它的任务,依次执行,不仅是主线程,其他的子线程也必须为调用了join的线程让路,让它执行完毕,在执行下一个调用了join的子线程,最后才轮到主线程上台(主线程:卧槽,我才是老大啊!)

 

如果每个子线程都设置时间参数,但是时间参数小于完成任务所需的时间的话



from threading import Thread
import threading
import time
local_value=threading.local()
class threadtest(Thread):
    
     def __init__(self,num):
         Thread.__init__(self)
         self.num=num
         #local_value.x=self.getName()
     def run(self):
         local_value.x=self.getName()
         for i in range(self.num):
             time.sleep(1)
             print(local_value.x)
from threading import current_thread
        
#print(current_thread(),local_value.__dict__,'--')
a=threadtest(6)
b=threadtest(6)
#c=threadtest(6)
a.start()
b.start()
a.join(2)
b.join(2)
#c.start()
#c.join(2)
#以下是主线程
for i in range(2):
    #ime.sleep(1)
    print(current_thread())



 

结果为:


Thread-1
Thread-2
Thread-1
Thread-2
Thread-1
Thread-2
Thread-1
Thread-2
<_MainThread(MainThread, started 8652)>
<_MainThread(MainThread, started 8652)>
Thread-1
Thread-2
Thread-1
Thread-2
[Finished in 6.2s]


从结果可以看出,子线程开始确实阻塞了主线程,但是没有阻塞其他子线程,所以两个子线程仍然是并行执行,一共阻塞了4秒钟(线程1,和线程2一起),四秒钟后程序开始执行join之后的代码,也就是主线程的代码,从这之后,主线程和子线程之间是并行关系。

 

总结一下(不要告诉我你只看总结)

1,子线程只要调用了join函数,就会阻塞主线程,不管有没有设置参数,

2,子线程如果调用join函数,设置了时间参数,主线程在设置的时间内被阻塞,子线之间一直是并行的,时间到了后,程序将会执行join()之后的代码。

3,如果子线程调用join函数,没有设置时间参数,那么子线程之间不会并行,而是有优先成那个调用了join的子线程。

如果不调用join,那么主线程不会为子线程阻塞,所有线程并发执行。