>  기사  >  백엔드 개발  >  Python으로 학습 노트 가져오기

Python으로 학습 노트 가져오기

高洛峰
高洛峰원래의
2017-02-03 16:05:201524검색

머리말

Python에서 모듈을 구성하는 방법에는 두 가지가 있습니다. 하나는 간단한 Python 파일, 파일 이름은 모듈 이름, 다른 하나는 패키지, 패키지는 여러 Python 파일이 포함된 디렉터리입니다. . 디렉터리에 __init__.py 파일이 있어야 하므로 디렉터리 이름은 모듈 이름입니다. import

import 구문

가져오기 구문에는 두 가지 유형이 있습니다

1. 모듈 직접 가져오기

import Module
import Module as xx

2. , 함수, 변수 등)

from Module import Name
from Module immport Name as yy

구문은 객체의 별칭을 설정하는 데 사용되며(여기서 객체는 모듈, 클래스, 함수 등을 나타냄) import는 현재 파일의 네임스페이스에 개체 이름

다음 디렉터리 구조가 있다고 가정

├── A.py
└── pkg
 ├── B.py
 └── __init__.py

현재 디렉터리에서 다음 명령문이 유효합니다

import A
import pkg
import pkg.B
from pkg import B

논의를 단순화하기 위해 다음에서는 구문

가져오기 단계


로드된 모든 모듈 정보에 대한 예를 제공하지 않습니다. python은 sys.modules 구조에 저장됩니다. 모듈을 import할 때 다음 단계가 수행됩니다.

import A인 경우 sys.modules에 이미 A가 있는지 확인하고, 있으면 로드하지 마세요. 그렇지 않은 경우 A에 대한 모듈 객체를 생성하고 A를 로드합니다

A에서 가져온 것이라면 B를 가져오고 먼저 A에 대한 모듈 객체를 생성한 다음 A를 구문 분석하고 여기에서 B를 찾아 A의 객체에 채웁니다. __dict__


nested import


모듈을 가져올 때 모듈 A, B, C가 여러 번 가져오게 될지 걱정될 수 있습니다. A는 B와 C를 가져와야 하고 B는 C를 가져와야 합니다. 이런 방식으로 A는 C를 한 번만 가져옵니다. 그러나 가져오기 자체는 B를 가져올 때 한 번 수행됩니다. 그러나 위에서 언급한 가져오기 단계에 따르면 두 번째 Import 시 모듈이 Load된 것으로 확인되어

Import는 반복되지 않으나 이때 Error

#filename: A.py
from B import BB
class AA:pass
 
#filename: B.py
from A import AA
class BB:pass

와 같은 상황이 발생합니다. , A.py를 실행하든 B.py를 실행하든 ImportError 예외가 발생합니다. A.py를 실행한다고 가정하면 그 이유는 다음과 같습니다

B에서 A.py 파일을 실행할 때. import BB, B.py가 먼저 스캔되고 A의 네임스페이스에서 B에 대한 모듈 객체가 생성되어 B에서 BB를 찾으려고 합니다

B.py를 먼저 스캔합니다. 이때 A에서 import AA를 실행합니다. , A.py

를 스캔하고 A.py의 첫 번째 줄을 스캔하여 B import BB를 실행합니다. 1단계에서 B에 대해 모듈 객체를 생성했으므로 B에서 직접 가져옵니다. 이때 모듈 객체의 __dict__는 BB를 얻을 수 없다는 것이 명백하므로 예외가 발생합니다.


이 상황을 해결하는 방법은 두 가지가 있는데,

B import BB에서 import B로 변경하거나 A import AA에서 import A로 변경

A.py 또는 B.py에서 두 줄의 코드를 교환


간단히 말하면, import는 디렉터리에 __init__.py 파일이 있을 때 import

package import


를 시도해야 한다는 점에 유의하세요. 패키지

패키지를 가져오는 것은 단일 파일을 가져오는 것과 같습니다.

단일 파일을 가져올 때 파일의 클래스, 함수 및 변수를 가져옵니다. 모두 가져오기 개체로 사용할 수 있습니다

패키지를 가져올 때 하위 패키지, 패키지에 포함된 파일, __init__.py에 있는 클래스, 함수, 변수를 모두 가져오기 개체로 사용할 수 있습니다

다음과 같은 디렉터리 구조가 있다고 가정합니다

pkg
├── __init__.py
└── file.py

__init__.py의 내용은 다음과 같습니다

argument = 0
class A:pass

해도 괜찮습니다. pkg

>>> import pkg
>>> import pkg.file
>>> from pkg import file
>>> from pkg import A
>>> from pkg import argument

그러나 다음 명령문은 잘못되었습니다

>>> import pkg.A
>>> import pkg.argument

ImportError: xxx라는 모듈이 없습니다. 왜냐하면 import A.B를 실행할 때 A와 B는 모두 모듈(파일 또는 패키지)이어야 합니다

상대 가져오기 및 절대 가져오기


절대 가져오기 형식은 import A.B 또는 from A 가져오기입니다. B, 상대 가져오기 형식은 from .import B 또는 from ..A import B, .는 현재 모듈을 나타내고, ..는 상위 수준 모듈을 나타내고, ...은 상위 수준 모듈을 나타냅니다. 여러 패키지가 있는 경우 한 패키지의 콘텐츠를 다른 패키지로 가져와야 할 수 있으며 이로 인해 절대 가져오기가 발생하며 이는 종종 오류가 발생할 가능성이 가장 높은 경우입니다.

디렉토리 구조는 다음과 같습니다

app
├── __inti__.py
├── mod1
│ ├── file1.py
│ └── __init__.py
├── mod2
│ ├── file2.py
│ └── __init__.py
└── start.py

app/start.py의 내용은 import mod1.file1


app/mod1의 내용입니다. /file1.py는 ..mod2 import file2

분석을 용이하게 하기 위해 이제 모든 py 파일(__init__.py 포함)의 첫 번째 줄에 print __file__, __name__

을 추가합니다. app/mod1/file1.py에서 상대 가져오기를 사용합니다. app/mod1에서 python file1.py를 실행하거나 app에서 python mod1/file1.py를 실행하면 오류 ValueError: Attemptedrelative import in non-package

이 보고됩니다. app에서 python을 실행하면 - m mod1.file1 또는 python start.py가 오류 ValueError: Attemptedrelative import between toplevel package

구체적인 이유는 나중에 논의하겠습니다. 모듈을 가져올 때 몇 가지 규칙을 살펴보세요

그것이 명확하지 않은 경우 패키지 구조를 지정할 때 Python은 __name__을 기반으로 패키지의 모듈 구조를 결정합니다. 모듈 자체이며 패키지 구조가 없습니다. A.B.C 구조인 경우 최상위 모듈은 A. 입니다.

기본적으로 이 원칙을 따르세요

절대 가져오기인 경우 모듈은 자체 하위 모듈만 가져올 수 있거나 최상위 모듈과 동일한 수준의 모듈 및 하위 모듈만 가져올 수 있습니다

상대 가져오기인 경우 모듈은 패키지 구조를 가져야 하며 최상위 모듈 내부의 모듈만 가져올 수 있습니다.


디렉터리 구조는 다음과 같습니다

A
├── B1
│ ├── C1
│ │ └── file.py
│ └── C2
└── B2

其中A,B1,B2,C1,C2都为包,这里为了展示简单没有列出__init__.py文件,当file.py的包结构为A.B1.C1.file(注意,是根据__name__来的,而不是磁盘的目录结构,在不同目录下执行file.py时对应的包目录结构都是不一样的)时,在file.py中可采用如下的绝对的导入

import A.B1.C2
import A.B2

和如下的相对导入

from .. import C2
from ... import B2

什么情况下会让file.py的包结构为A.B1.C1.file呢,有如下两种

在A的上层目录执行python -m A.B1.C1.file, 此时明确指定了包结构

在A的上层目录建立文件start.py,在start.py里有import A.B1.C1.file,然后执行python start.py,此时包结构是根据file.py的__name__变量来的

再看前面出错的两种情况,第一种执行python file1.py和python mod1/file1.py,此时file.py的__name__为__main__ ,也就是说它本身就是顶层模块,并没有包结构,所以会报错

第二种情况,在执行python -m mod1.file1和python start.py时,前者明确告诉解释器mod1是顶层模块,后者需要导入file1,而file1.py的__name__为mod1.file1,顶层模块为也mod1,所以在file1.py中执行from ..mod2 import file2时会报错 ,因为mod2并不在顶层模块mod1内部。通过错误堆栈可以看出,并不是在start.py中绝对导入时报错,而是在file1.py中相对导入报的错

那么如何才能偶正确执行呢,有两种方法,一种是在app上层目录执行python -m app.mod1.file1,另一种是改变目录结构,将所有包放在一个大包中,如下

   
app
├── pkg
│ ├── __init__.py
│ ├── mod1
│ │ ├── __init__.py
│ │ └── file1.py
│ └── mod2
│ ├── __init__.py
│ └── file2.py
└── start.py

start.py内容改成import pkg.mod1.file1,然后在app下执行python start.py

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家学习或者使用python能带来一定的帮助,如有疑问大家可以留言交流。

更多python中import学习备忘笔记相关文章请关注PHP中文网!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.