1. 程式人生 > >python3基礎筆記(六)模塊與包

python3基礎筆記(六)模塊與包

__main__ 過程 __init__ 而不是 pro .com manage sys.path 腳本

模塊&包

模塊(modue)的概念:

在計算機程序的開發過程中,隨著程序代碼越寫越多,在一個文件裏代碼就會越來越長,越來越不容易維護。

為了編寫可維護的代碼,我們把很多函數分組,分別放到不同的文件裏,這樣,每個文件包含的代碼就相對較少,很多編程語言都采用這種組織代碼的方式。在Python中,一個.py文件就稱之為一個模塊(Module)。

使用模塊有什麽好處?

最大的好處是大大提高了代碼的可維護性。

其次,編寫代碼不必從零開始。當一個模塊編寫完畢,就可以被其他地方引用。我們在編寫程序的時候,也經常引用其他模塊,包括Python內置的模塊和來自第三方的模塊。

所以,模塊一共三種:

  • python標準庫
  • 第三方模塊
  • 應用程序自定義模塊

另外,使用模塊還可以避免函數名和變量名沖突。相同名字的函數和變量完全可以分別存在不同的模塊中,因此,我們自己在編寫模塊時,不必考慮名字會與其他模塊沖突。但是也要註意,盡量不要與內置函數名字沖突。

模塊導入方法

1 import 語句

1 import module1[, module2[,... moduleN]

當我們使用import語句的時候,Python解釋器是怎樣找到對應的文件的呢?答案就是解釋器有自己的搜索路徑,存在sys.path裏。  

1 2 [‘‘, ‘/usr/lib/python3.4‘, ‘/usr/lib/python3.4/plat-x86_64-linux-gnu‘, ‘/usr/lib/python3.4/lib-dynload‘, ‘/usr/local/lib/python3.4/dist-packages‘, ‘/usr/lib/python3/dist-packages‘]  

因此若像我一樣在當前目錄下存在與要引入模塊同名的文件,就會把要引入的模塊屏蔽掉。

2 from…import 語句

1 from
modname import name1[, name2[, ... nameN]]

這個聲明不會把整個modulename模塊導入到當前的命名空間中,只會將它裏面的name1或name2單個引入到執行這個聲明的模塊的全局符號表。

3 From…import* 語句

1 from modname import *

這提供了一個簡單的方法來導入一個模塊中的所有項目。然而這種聲明不該被過多地使用。大多數情況, Python程序員不使用這種方法,因為引入的其它來源的命名,很可能覆蓋了已有的定義。

4 運行本質 

1 2 #1 import test #2 from test import add  

無論1還是2,首先通過sys.path找到test.py,然後執行test腳本(全部執行),區別是1會將test這個變量名加載到名字空間,而2只會將add這個變量名加載進來。  

包(package)

如果不同的人編寫的模塊名相同怎麽辦?為了避免模塊名沖突,Python又引入了按目錄來組織模塊的方法,稱為包(Package)。

舉個例子,一個abc.py的文件就是一個名字叫abc的模塊,一個xyz.py的文件就是一個名字叫xyz的模塊。

現在,假設我們的abcxyz這兩個模塊名字與其他模塊沖突了,於是我們可以通過包來組織模塊,避免沖突。方法是選擇一個頂層包名:

技術分享圖片

引入了包以後,只要頂層的包名不與別人沖突,那所有模塊都不會與別人沖突。現在,view.py模塊的名字就變成了hello_django.app01.views,類似的,manage.py的模塊名則是hello_django.manage。

請註意,每一個包目錄下面都會有一個__init__.py的文件,這個文件是必須存在的,否則,Python就把這個目錄當成普通目錄(文件夾),而不是一個包。__init__.py可以是空文件,也可以有Python代碼,因為__init__.py本身就是一個模塊,而它的模塊名就是對應包的名字。

調用包就是執行包下的__init__.py文件

註意點(important)

技術分享圖片

1--------------

在nod1裏import hello是找不到的,有同學說可以找到呀,那是因為你的pycharm為你把myapp這一層路徑加入到了sys.path裏面,所以可以找到,然而程序一旦在命令行運行,則報錯。有同學問那怎麽辦?簡單啊,自己把這個路徑加進去不就OK啦:

1 2 3 4 5 import sys,os BASE_DIR=os.path.dirname(os.path.dirname(os.path.abspath(__file__))) sys.path.append(BASE_DIR) import hello hello.hello1()

2 --------------

1 2 if __name__==‘__main__‘: print(‘ok‘)

“Make a .py both importable and executable”

如果我們是直接執行某個.py文件的時候,該文件中那麽”__name__ == ‘__main__‘“是True,但是我們如果從另外一個.py文件通過import導入該文件的時候,這時__name__的值就是我們這個py文件的名字而不是__main__。

這個功能還有一個用處:調試代碼的時候,在”if __name__ == ‘__main__‘“中加入一些我們的調試代碼,我們可以讓外部模塊調用的時候不執行我們的調試代碼,但是如果我們想排查問題的時候,直接執行該模塊文件,調試代碼能夠正常運行!s

3   

技術分享圖片

##-------------cal.py
def add(x,y):
 
    return x+y
##-------------main.py
import cal      #from module import cal
 
def main():
 
    cal.add(1,2)
     
##--------------bin.py
from module import main
 
main.main()

  

# from module import cal 改成 from . import cal同樣可以,這是因為bin.py是我們的執行腳本,
# sys.path裏有bin.py的當前環境。即/Users/yuanhao/Desktop/whaterver/project/web這層路徑,
# 無論import what ,  解釋器都會按這個路徑找。所以當執行到main.py時,import cal會找不到,因為
# sys.path裏沒有/Users/yuanhao/Desktop/whaterver/project/web/module這個路徑,而
#  from  module/.  import cal 時,解釋器就可以找到了。

  

python3基礎筆記(六)模塊與包