현재 스크립트 디렉토리를 올바르게 결정하는 방법은 무엇입니까?
파이썬에서 현재 스크립트 디렉토리를 결정하는 가장 좋은 방법은 무엇입니까?
파이썬 코드를 호출하는 많은 방법으로 인해 좋은 해결책을 찾기가 어렵다는 것을 알았습니다.
몇 가지 문제가 있습니다 :
__file__
로 스크립트를 실행하면 정의되지 않습니다exec
.execfile
__module__
모듈에서만 정의됩니다
사용 사례:
./myfile.py
python myfile.py
./somedir/myfile.py
python somedir/myfile.py
execfile('myfile.py')
(다른 스크립트에서 다른 디렉토리에 있으며 다른 현재 디렉토리를 가질 수 있습니다.
완벽한 솔루션이 없다는 것을 알고 있지만 대부분의 경우를 해결하는 최선의 방법을 찾고 있습니다.
가장 많이 사용되는 방법 os.path.dirname(os.path.abspath(__file__))
이지만 다른 스크립트를 사용하여 스크립트를 실행하면 실제로 작동하지 않습니다 exec()
.
경고
현재 디렉토리를 사용하는 모든 솔루션이 실패합니다. 이는 스크립트가 호출되는 방식에 따라 다를 수 있거나 실행중인 스크립트 내에서 변경 될 수 있습니다.
os.path.dirname(os.path.abspath(__file__))
실제로 당신이 얻는 최고의 것입니다.
exec
/ execfile
; 로 스크립트를 실행하는 것은 일반적이지 않습니다 . 일반적으로 모듈 인프라를 사용하여 스크립트를로드해야합니다. 이러한 방법을 사용해야하는 경우, 내가 설정하는 것이 좋습니다 __file__
에서 globals
하면 해당 파일 이름을 읽을 수 있도록 스크립트에 전달합니다.
실행 코드에서 파일 이름을 얻는 다른 방법은 없습니다. CWD가 완전히 다른 위치에있을 수 있습니다.
를 통해 스크립트가 호출되는 경우를 실제로 다루려면 모듈을 execfile(...)
사용 inspect
하여 파일 이름 (경로 포함)을 추론 할 수 있습니다 . 내가 아는 한, 이것은 나열된 모든 경우에 적용됩니다.
filename = inspect.getframeinfo(inspect.currentframe()).filename
path = os.path.dirname(os.path.abspath(filename))
#!/usr/bin/env python
import inspect
import os
import sys
def get_script_dir(follow_symlinks=True):
if getattr(sys, 'frozen', False): # py2exe, PyInstaller, cx_Freeze
path = os.path.abspath(sys.executable)
else:
path = inspect.getabsfile(get_script_dir)
if follow_symlinks:
path = os.path.realpath(path)
return os.path.dirname(path)
print(get_script_dir())
CPython, Jython, Pypy에서 작동합니다. 스크립트를 사용하여 스크립트를 실행하면 작동합니다 execfile()
( sys.argv[0]
그리고 __file__
기반 솔루션은 여기서 실패합니다). 스크립트가 실행 가능한 zip 파일 (/ eg) 안에 있으면 작동합니다 . 스크립트가 PYTHONPATH=/path/to/library.zip python -mscript_to_run
zip 파일에서 "가져온"( ) 경우 작동 합니다. 이 경우 아카이브 경로를 반환합니다. 스크립트가 독립 실행 형 실행 파일 ( sys.frozen
) 로 컴파일 된 경우 작동합니다 . 심볼릭 링크에서 작동합니다 ( realpath
심볼릭 링크 제거). 대화식 인터프리터에서 작동합니다. 이 경우 현재 작업 디렉토리를 반환합니다.
Python 3.4 이상에서는 더 간단한 pathlib
모듈을 사용할 수 있습니다 .
from inspect import currentframe, getframeinfo
from pathlib import Path
filename = getframeinfo(currentframe()).filename
parent = Path(filename).resolve().parent
os.path...
방법은 파이썬 2의 '할 일'입니다.
Python 3에서는 다음과 같이 스크립트 디렉토리를 찾을 수 있습니다.
from pathlib import Path
cwd = Path(__file__).parents[0]
사용 os.path.dirname(os.path.abspath(__file__))
되는 경우에 대한 실제 필요성이 있는지 여부를 매우 신중하게 사용 하고 조사하십시오 exec
. 스크립트를 모듈로 사용할 수없는 경우 문제가 발생할 수 있습니다.
Zen of Python # 8 을 명심하십시오 . 유용 사례에 대한 좋은 주장이 있다고 생각 exec
되면 문제의 배경에 대해 더 자세히 알려주십시오.
할 것이다
import os
cwd = os.getcwd()
당신이 원하는 것을합니까? "현재 스크립트 디렉토리"가 정확히 무엇을 의미하는지 잘 모르겠습니다. 제공 한 사용 사례에 대해 예상되는 결과는 무엇입니까?
import os
import sys
def get_script_path():
return os.path.dirname(os.path.realpath(sys.argv[0]))
my_script_dir = get_script_path()
print my_script_dir
이것은 스택의 최상위에 스크립트의 디렉토리를 제공합니다 (즉, 파이썬이 아닌 실행중인 디렉토리 C:/
).
먼저 익명의 코드를 삽입하는 방법에 대해 이야기하고 있다면 유스 케이스가 누락 된 것입니다.
code.compile_command()
code.interact()
imp.load_compiled()
imp.load_dynamic()
imp.load_module()
__builtin__.compile()
loading C compiled shared objects? example: _socket?)
그러나 진짜 질문은, 당신의 목표는 무엇입니까-당신은 일종의 보안을 시행하려고합니까? 또는 당신은 무엇을로드하는 것에 관심이 있습니까?
보안에 관심이있는 경우 exec / execfile을 통해 가져 오는 파일 이름은 중요하지 않습니다. rexec 를 사용해야 합니다. 다음을 제공합니다.
이 모듈에는 r_eval (), r_execfile (), r_exec () 및 r_import () 메소드를 지원하는 RExec 클래스가 포함되어 있습니다.이 메소드는 표준 Python 함수 eval (), execfile () 및 exec 및 import 문의 제한된 버전입니다. 이 제한된 환경에서 실행되는 코드는 안전하다고 간주되는 모듈과 기능에만 액세스 할 수 있습니다. RExec 추가 기능을 원하는대로 서브 클래 싱 할 수 있습니다.
그러나 이것이 학문적 추구에 더 가깝다면 여기에 좀 더 깊이 파고들 수있는 몇 가지 구피 접근법이 있습니다.
스크립트 예 :
./deep.py
print ' >> level 1'
execfile('deeper.py')
print ' << level 1'
./deeper.py
print '\t >> level 2'
exec("import sys; sys.path.append('/tmp'); import deepest")
print '\t << level 2'
/tmp/deepest.py
print '\t\t >> level 3'
print '\t\t\t I can see the earths core.'
print '\t\t << level 3'
./codespy.py
import sys, os
def overseer(frame, event, arg):
print "loaded(%s)" % os.path.abspath(frame.f_code.co_filename)
sys.settrace(overseer)
execfile("deep.py")
sys.exit(0)
산출
loaded(/Users/synthesizerpatel/deep.py)
>> level 1
loaded(/Users/synthesizerpatel/deeper.py)
>> level 2
loaded(/Users/synthesizerpatel/<string>)
loaded(/tmp/deepest.py)
>> level 3
I can see the earths core.
<< level 3
<< level 2
<< level 1
물론 이것은 리소스를 많이 사용하는 방법이므로 모든 코드를 추적해야합니다.별로 효율적이지 않습니다. 그러나 둥지에 깊숙이 들어가도 계속 작동하기 때문에 새로운 접근법이라고 생각합니다. 'eval'을 무시할 수 없습니다. 당신이 비록 수 execfile ()을 재정의합니다.
이 접근법은 '가져 오기'가 아닌 exec / exec 파일에만 적용됩니다. 더 높은 수준의 '모듈'로드 후크의 경우 sys.path_hooks (PyMOTW의 쓰기 제공) 를 사용할 수 있습니다 .
그게 내가 머리 꼭대기에서 벗어난 전부 야
여기에 지금까지 게시 된 모든 솔루션보다 여전히 나은 부분 솔루션이 있습니다.
import sys, os, os.path, inspect
#os.chdir("..")
if '__file__' not in locals():
__file__ = inspect.getframeinfo(inspect.currentframe())[0]
print os.path.dirname(os.path.abspath(__file__))
이제는 모든 호출이 작동하지만 누군가 chdir()
현재 디렉토리를 변경하는 데 사용 하면 실패합니다.
메모:
sys.argv[0]
작동하지 않을 것입니다-c
. 스크립트를 실행하면python -c "execfile('path-tester.py')"
- https://gist.github.com/1385555 에서 완전한 테스트를 게시했으며 이를 개선 할 수 있습니다.
대부분의 경우 작동합니다.
import os,sys
dirname=os.path.dirname(os.path.realpath(sys.argv[0]))
희망적으로 이것은 도움이됩니다 :-어디에서나 스크립트 / 모듈을 실행하면 스크립트 __file__
의 위치를 나타내는 모듈 변수 인 변수 에 액세스 할 수 있습니다 .
인터프리터를 사용하는 경우 반면에, 당신은 당신의 이름을 얻을 것이다 그 변수에 대한 액세스 권한이없는 NameError
그리고 os.getcwd()
당신이 다른 곳에서 파일을 실행하는 경우 당신에게 잘못된 디렉토리를 줄 것이다.
이 솔루션은 모든 경우에 원하는 것을 제공합니다.
from inspect import getsourcefile
from os.path import abspath
abspath(getsourcefile(lambda:0))
나는 그것을 철저히 테스트하지는 않았지만 내 문제를 해결했습니다.
단지 pwd
jupyter 노트북에서 : pwd
+ Shift + 입력
스파이더에서 : pwd
+ F9
참고 URL : https://stackoverflow.com/questions/3718657/how-to-properly-determine-current-script-directory
'Programing' 카테고리의 다른 글
힘내 HEAD 및 ORIG_HEAD (0) | 2020.04.19 |
---|---|
파일 이름 만 출력하는 diff (0) | 2020.04.19 |
vim의 캐릭터로 이동 (0) | 2020.04.19 |
목록에서 여러 데이터 프레임을 동시에 병합 (0) | 2020.04.19 |
여러 열을 참조하는 팬더 '적용'기능이 작동하지 않는 이유는 무엇입니까? (0) | 2020.04.19 |