python 全局變量的import機制


      在之前學習python設計模式(工廠模式實踐篇),希望使用全局變量代替c++的宏完成服務自動注冊功能時,遇到過一個問題,全局變量的定義和使用放在同一個可執行腳本中的問題。先把有問題的代碼曬一下:

IServer.py

from abc import ABCMeta, abstractmethod
print __name__

class IServer:
    def __init__(self):
        pass

    @abstractmethod
    def DoWithA(self):
        pass

    @abstractmethod
    def DoWithB(self):
        pass

 IServer_A.py

import IServer
serverType ='1001'
print __name__
dir()
from CreatFactory import GLOBAL_class_dic dir() class IServer_A(IServer.IServer): def __init__(self): pass def DoWithA(self): print 'Server_A do with interface A' def DoWithB(self): print 'Server_A do with interface B' global GLOBAL_class_dic print 'the id of GLOBAL_class_dic in A is:',id(GLOBAL_class_dic) GLOBAL_class_dic[serverType] = IServer_A print 'GLOBAL_class_dic in a is:', GLOBAL_class_dic

IServer_B.py

import IServer
serverType ='1002'from CreatFactory import GLOBAL_class_dic
print __name__

class IServer_B(IServer.IServer):
    def __init__(self):
        pass

    def DoWithA(self):
        print 'Server_B do with interface A'

    def DoWithB(self):
        print 'Server_B do with interface B'

print 'the id of GLOBAL_class_dic in B is:',id(GLOBAL_class_dic)
GLOBAL_class_dic[serverType] = IServer_B
print 'GLOBAL_class_dic in b is:', GLOBAL_class_dic

CreatFactory.py

#coding:UTF-8
import os;
import sys;
import threading
from misc import *

global GLOBAL_class_dic

GLOBAL_class_dic ={1:1}
print 'GLOBAL_class_dic in define is:', GLOBAL_class_dic
print 'the id of GLOBAL_class_dic in define is:', id(GLOBAL_class_dic)

dir()
import IServer_A import IServer_B def CreateServer(serverType):
global GLOBAL_class_dic print 'GLOBAL_class_dic in use is:', GLOBAL_class_dic print 'the id of GLOBAL_class_dic in USE is:', id(GLOBAL_class_dic) if GLOBAL_class_dic.has_key(serverType): return GLOBAL_class_dic[serverType] else: return 'no' if __name__ == '__main__': pass # 接收到報文后,根據報文的內容,從db中獲取到serverType,假設獲取到的serverType=1001 print 'main' print 'GLOBAL_class_dic in main A is:', GLOBAL_class_dic serverType = '1002' server = CreateServer(serverType) print 'GLOBAL_class_dic in main B is:', GLOBAL_class_dic print 'server :',server server.DoWithA(server())

    代碼內已經加了調試的部分信息, 運行CreatFactory.py。調用DoWithA失敗,提示AttributeError: 'str' object has no attribute 'DoWithA'。運行結果如下:

 
         

D:\Python27\python.exe "D:/DesignMode/Server --00/CreatFactory.py"
GLOBAL_class_dic in define is: {1: 1}
the id of GLOBAL_class_dic in define is: 36230176
['GLOBAL_class_dic', 'Misc', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'binascii', 'inspect', 'minidom', 'os', 'struct', 'sys', 'threading']
IServer
IServer_A
['IServer', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'serverType']
GLOBAL_class_dic in define is: {1: 1}
the id of GLOBAL_class_dic in define is: 36230032
['GLOBAL_class_dic', 'Misc', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'binascii', 'inspect', 'minidom', 'os', 'struct', 'sys', 'threading']
1
['IServer', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'serverType']
['GLOBAL_class_dic', 'IServer', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'serverType']
IServer_B
the id of GLOBAL_class_dic in B is: 36230032
GLOBAL_class_dic in b is: {1: 1, '1002': <class IServer_B.IServer_B at 0x022C2ED8>}
['GLOBAL_class_dic', 'IServer', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'serverType']
the id of GLOBAL_class_dic in A is: 36230032
GLOBAL_class_dic in a is: {1: 1, '1002': <class IServer_B.IServer_B at 0x022C2ED8>, '1001': <class IServer_A.IServer_A at 0x02273420>}
main
GLOBAL_class_dic in main A is: {1: 1}
GLOBAL_class_dic in use is: {1: 1}
the id of GLOBAL_class_dic in USE is: 36230176
GLOBAL_class_dic in main B is: {1: 1}
server : no
Traceback (most recent call last):
File "D:/DesignMode/Server --00/CreatFactory.py", line 38, in <module>
server.DoWithA(server())
AttributeError: 'str' object has no attribute 'DoWithA'

 
         

Process finished with exit code 1

 

     從運行的結果,可以看到:GLOBAL_class_dic 被定義了2次。有兩個不同的id,第一次定義分配了一塊內存,第二次不明原因的又重新分配了一塊內存,然后服務的自動注冊全部注冊在這塊內存中,等到main函數使用的使用,又使用的是第一次申請的內存,所以導致程序運行失敗。那問題就來了,為什么會被重新又分配了一次?

     之所以會被重新定義一次全局變量,是因為在執行CreatFactory.py時,最開始定義了全局變量,此時該命名空間可使用的函數和變量打印:['GLOBAL_class_dic', 'Misc', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'binascii', 'inspect', 'minidom', 'os', 'struct', 'sys', 'threading',然后在import IServer_A,在IServer_A.py中,import IServer后,在from CreatFactory import GLOBAL_class_dic打印出可使用的函數和變量時,['IServer', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'serverType'],就沒有GLOBAL_class_dic,程序發現沒有,就又重新聲明了一遍。似乎問題原因已經找到了。

    python在導入的時候,有2種場景,一種就是在文件前普通的import語句,還有一種就是特殊的場景:__main__模塊是相對於Python的導入系統。在最開始運行CreatFactory.py文件時,__name__打印的值是__main__,而再子類再次導入時,會在當前命名空間查找是否已經導入__name__=CreatFactory,發現這個模塊不存在,故此又導入了一遍,全局變量由此又被重新定義分配了內存,后期全局變量在子類業務的使用就都使用該值,而在main函數里,使用的又是當前的作用域內的第一次定義的全局變量。

  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM