Atualização 2018-11-28:
Aqui está um resumo dos experimentos com Python 2 e 3. Com
main.py - executa foo.py
foo.py - executa lib / bar.py
lib / bar.py - imprime expressões de caminho de arquivo
| Python | Run statement | Filepath expression |
|--------+---------------------+----------------------------------------|
| 2 | execfile | os.path.abspath(inspect.stack()[0][1]) |
| 2 | from lib import bar | __file__ |
| 3 | exec | (wasn't able to obtain it) |
| 3 | import lib.bar | __file__ |
Para o Python 2, pode ser mais claro alternar para pacotes para que possa ser usado from lib import bar
- basta adicionar__init__.py
arquivos às duas pastas.
Para o Python 3, execfile
não existe - a alternativa mais próxima é exec(open(<filename>).read())
, embora isso afete os quadros da pilha. É mais simples de usar import foo
e import lib.bar
- não é __init__.py
necessário arquivo.
Veja também Diferença entre importação e execfile
Resposta original:
Aqui está um experimento baseado nas respostas neste tópico - com o Python 2.7.10 no Windows.
Os baseados em pilha são os únicos que parecem fornecer resultados confiáveis. Os dois últimos têm a sintaxe mais curta , ou seja -
print os.path.abspath(inspect.stack()[0][1]) # C:\filepaths\lib\bar.py
print os.path.dirname(os.path.abspath(inspect.stack()[0][1])) # C:\filepaths\lib
Aqui está a estes sendo adicionados ao sys como funções! Crédito para @Usagi e @pablog
Com base nos três arquivos a seguir, e executando o main.py a partir de sua pasta com python main.py
(também tentou execfiles com caminhos absolutos e chamando de uma pasta separada).
C: \ filepaths \ main.py: execfile('foo.py')
C: \ filepaths \ foo.py: execfile('lib/bar.py')
C: \ filepaths \ lib \ bar.py:
import sys
import os
import inspect
print "Python " + sys.version
print
print __file__ # main.py
print sys.argv[0] # main.py
print inspect.stack()[0][1] # lib/bar.py
print sys.path[0] # C:\filepaths
print
print os.path.realpath(__file__) # C:\filepaths\main.py
print os.path.abspath(__file__) # C:\filepaths\main.py
print os.path.basename(__file__) # main.py
print os.path.basename(os.path.realpath(sys.argv[0])) # main.py
print
print sys.path[0] # C:\filepaths
print os.path.abspath(os.path.split(sys.argv[0])[0]) # C:\filepaths
print os.path.dirname(os.path.abspath(__file__)) # C:\filepaths
print os.path.dirname(os.path.realpath(sys.argv[0])) # C:\filepaths
print os.path.dirname(__file__) # (empty string)
print
print inspect.getfile(inspect.currentframe()) # lib/bar.py
print os.path.abspath(inspect.getfile(inspect.currentframe())) # C:\filepaths\lib\bar.py
print os.path.dirname(os.path.abspath(inspect.getfile(inspect.currentframe()))) # C:\filepaths\lib
print
print os.path.abspath(inspect.stack()[0][1]) # C:\filepaths\lib\bar.py
print os.path.dirname(os.path.abspath(inspect.stack()[0][1])) # C:\filepaths\lib
print
__file__
absoluto ou relativo?