from threading import Thread
import time
g_num = 0
def test1():
global g_num
for i in range(1000000):
g_num += 1
print("---test1---g_num=%d" % g_num)
def test2():
global g_num
for i in range(1000000):
g_num += 1
print("---test2---g_num=%d" % g_num)
p1 = Thread(target=test1)
p1.start()
# time.sleep(3) # 如果此时加入休眠三秒,将会等待p1线程执行完,此时p2线程的结果会符合预期,而主线程继续向下执行,其结果不会与p2相同
p2 = Thread(target=test2)
p2.start()
print("---g_num=%d---" % g_num)
返回:
---g_num=787091---
---test1---g_num=1263307
---test2---g_num=1792617
解析:
但是由于是多线程访问,
在num=0时,t1取得num=0。而此时cpu根据调度策略系统把t1调度为”sleeping”状态,把t2转换为”running”状态,t2也获得num=0。然后t2对得到的值进行加1并赋给num,使得num=1。然后系统又把t2调度为”sleeping”,把t1转为”running”。线程t1又把它之前得到的0加1后赋值给num。这样,明明t1和t2都完成了1次加1工作,但结果仍然是num=1。
或者说num = 0 时,线程t1得以被cpu执行,t1将num的值改为1后,cpu执行t2,此时t2拿到num的值是修改后的,也就是1,但是刚刚获取到值,cpu又把执行权给了t1,此时t1拿到值并加1,然后t2执行,因为t2在让出执行权之前拿到的值是1所以将1加上1得到2,所以num本应成为3,但是此时仍为2、当然cpu的调度可能性有很多种情况,可能t1已经将num加到了100,而t2有将其改为1也是有可能的,完全取决于cpu的调度