답변:
open
그리고 start
이 일을 맥 OS / X 및 Windows 명령 인터프리터 가지 각각이다.
Python에서 호출하려면 subprocess
module 또는 os.system()
.
사용할 패키지에 대한 고려 사항은 다음과 같습니다.
을 통해 전화를 걸 수 os.system
있지만 작동하지만 ...
이스케이프 : os.system
경로 이름에 공백이나 다른 쉘 메타 문자가없는 파일 이름 (예 :) 에서만 작동합니다. 그렇지 않으면 A:\abc\def\a.txt
이스케이프해야합니다. 이 shlex.quote
유닉스 시스템,하지만 윈도우 아무것도 정말 표준입니다. python, windows : shlex로 명령 줄 구문 분석을 참조하십시오 .
os.system("open " + shlex.quote(filename))
os.system("start " + filename)
적절하게 말하는 것도 filename
이스케이프해야하는 곳 .subprocess
모듈을 통해 호출 할 수도 있지만 ...
Python 2.7 이상에서는 다음을 사용하십시오.
subprocess.check_call(['open', filename])
Python 3.5 이상에서는 약간 더 복잡하지만 다소 더 다양한 기능을 동등하게 사용할 수 있습니다.
subprocess.run(['open', filename], check=True)
Python 2.4와 완전히 호환되어야하는 경우 subprocess.call()
자체 오류 검사를 사용 하고 구현할 수 있습니다.
try:
retcode = subprocess.call("open " + filename, shell=True)
if retcode < 0:
print >>sys.stderr, "Child was terminated by signal", -retcode
else:
print >>sys.stderr, "Child returned", retcode
except OSError, e:
print >>sys.stderr, "Execution failed:", e
자, 사용의 장점은 무엇 subprocess
입니까?
'filename ; rm -rf /'
"문제를, 그리고 경우 파일 이름이 손상 될 수 있으며, 사용하는 것은 subprocess.call
우리에게 약간의 추가적인 보호 기능을 제공합니다.retcode
. 두 경우 모두 여전히 의존 하고 있습니다. 그러나 오류가 발생한 경우 명시 적으로 예외를 발생시키는 동작은 오류가 있는지 확인하는 데 확실히 도움이됩니다 (일부 시나리오에서는 추적이 단순히 오류를 무시하는 것보다 더 도움이되지 않을 수 있음).이의 제기 "하지만 subprocess
선호됩니다." 그러나 os.system()
는 더 이상 사용되지 않으며 어떤 의미에서는이 특정 작업을위한 가장 간단한 도구입니다. 결론 : os.system()
따라서 사용도 정답입니다.
표시 단점은 윈도우이다 start
명령은 필요 에 전달할 수 shell=True
있는 사용의 이점의 대부분을 부정 subprocess
.
filename
형식이 어디인지에 따라 이것은 os.system ()이 안전하지 않고 나쁜 이유에 대한 완벽한 예입니다. 하위 프로세스가 더 좋습니다.
os.system()
는 쉘을 사용한다는 것입니다 (그리고 여기서 쉘 이스케이프를 수행하지 않으므로 쉘 메타 문자를 포함하는 완벽하게 유효한 파일 이름에 대해 나쁜 일이 발생합니다). subprocess.call()
선호 되는 이유는를 사용하여 셸을 우회 할 수있는 옵션 이 있기 때문 subprocess.call(["open", filename])
입니다. 이것은 모든 유효한 파일 이름에 대해 작동하며 신뢰할 수없는 파일 이름에 대해서도 쉘 주입 취약점을 유발하지 않습니다.
subprocess
Python 2.4 이상에서 사용 가능한 모듈을 사용하십시오. os.system()
따라서 쉘 이스케이프를 처리 할 필요가 없습니다.
import subprocess, os, platform
if platform.system() == 'Darwin': # macOS
subprocess.call(('open', filepath))
elif platform.system() == 'Windows': # Windows
os.startfile(filepath)
else: # linux variants
subprocess.call(('xdg-open', filepath))
이중 괄호는 subprocess.call()
첫 번째 인수로 시퀀스를 원 하기 때문에 여기에서 튜플을 사용합니다. Gnome이있는 Linux 시스템 gnome-open
에는 동일한 작업을 수행하지만 xdg-open
Free Desktop Foundation 표준이며 Linux 데스크톱 환경에서 작동 하는 명령 도 있습니다.
xdg-open
- linux.die.net
xdg-open test.py
했고 그것은 나를 위해 firefox 다운로드 대화 상자를 열었습니다. 뭐가 문제 야? 나는 manjaro linux에 있습니다.
xdg-open
구성이 혼란스러운 것처럼 들리지만 실제로는 의견에서 문제를 해결할 수있는 문제가 아닙니다. 아마도 unix.stackexchange.com/questions/36380/…을
나는 선호한다:
os.startfile(path, 'open')
이 모듈은 폴더와 파일에 공백이있는 파일 이름을 지원합니다.
A:\abc\folder with spaces\file with-spaces.txt
( python docs ) 'open'은 추가 할 필요가 없습니다 (기본값). 문서에서는 Windows 탐색기에서 파일 아이콘을 두 번 클릭하는 것과 같음을 구체적으로 언급합니다.
이 솔루션은 Windows 전용입니다.
startfile
하지 않고 함수가 존재하지 않습니다. 즉, 사용자는 누락 된 함수에 대해 혼란스러운 오류 메시지를 받게됩니다. 이를 피하기 위해 플랫폼을 확인하는 것이 좋습니다.
완전성을 위해 (문제가 아님) xdg-open 은 Linux에서 동일한 작업을 수행합니다.
import os
import subprocess
def click_on_file(filename):
'''Open document with default application in Python.'''
try:
os.startfile(filename)
except AttributeError:
subprocess.call(['open', filename])
휴리스틱 방법을 사용해야하는 경우를 고려할 수 있습니다 webbrowser
.
표준 라이브러리이며 이름에도 불구하고 파일을 열려고 시도합니다.
일부 플랫폼에서는이 기능을 사용하여 파일 이름을 열려고하면 작동하고 운영 체제 관련 프로그램이 시작될 수 있습니다. 그러나 이것은 지원되거나 이식 가능하지 않습니다. ( 참조 )
이 코드를 시도했지만 Windows 7 및 Ubuntu Natty에서 제대로 작동했습니다.
import webbrowser
webbrowser.open("path_to_file")
이 코드는 Internet Explorer 8을 사용하는 Windows XP Professional에서도 잘 작동합니다.
open location
유효한 URL로 경로를 제공하는 경우가 해결해야이.
import webbrowser webbrowser.open("file:///Users/nameGoesHere/Desktop/folder/file.py")
계속 진행하려면 subprocess.call()
Windows에서 다음과 같이 표시되어야합니다.
import subprocess
subprocess.call(('cmd', '/C', 'start', '', FILE_NAME))
다음을 사용할 수 없습니다.
subprocess.call(('start', FILE_NAME))
때문에 start
실행 파일이 아니라 cmd.exe
프로그램 의 명령 입니다 . 이것은 작동합니다 :
subprocess.call(('cmd', '/C', 'start', FILE_NAME))
하지만 FILE_NAME에 공백이없는 경우에만 가능합니다.
while subprocess.call
방법en 는 매개 변수를 적절하게 인용start
명령에는 다소 이상한 구문이 있습니다.
start notes.txt
다음 이외의 작업을 수행합니다.
start "notes.txt"
첫 번째 인용 문자열은 창의 제목을 설정해야합니다. 공간에서 작동하도록하려면 다음을 수행해야합니다.
start "" "my notes.txt"
그게 위에있는 코드가하는 일입니다.
시작은 긴 경로 이름과 공백을 지원하지 않습니다. 8.3 호환 경로로 변환해야합니다.
import subprocess
import win32api
filename = "C:\\Documents and Settings\\user\\Desktop\file.avi"
filename_short = win32api.GetShortPathName(filename)
subprocess.Popen('start ' + filename_short, shell=True )
API 호출을 사용하려면 파일이 있어야합니다.
start "Title" "C:\long path to\file.avi"
나는 꽤 늦었지만 여기에 Windows API를 사용하는 솔루션이 있습니다. 이렇게하면 항상 연결된 응용 프로그램이 열립니다.
import ctypes
shell32 = ctypes.windll.shell32
file = 'somedocument.doc'
shell32.ShellExecuteA(0,"open",file,0,0,5)
많은 마법 상수. 첫 번째 0은 현재 프로그램의 hwnd입니다. 0 일 수 있습니다. 다른 두 개의 0은 선택적 매개 변수 (매개 변수 및 디렉토리)입니다. 5 == SW_SHOW, 앱 실행 방법을 지정합니다. 자세한 정보 는 ShellExecute API 문서 를 읽어보십시오 .
os.startfile(file)
됩니까?
Mac OS X에서 파일을 열 앱을 지정하려면 다음을 사용하십시오.
os.system("open -a [app name] [file name]")
Windows 8.1에서 아래는 subprocess.call
경로에 공백이 있는 다른 주어진 방법으로 작동했습니다.
subprocess.call('cmd /c start "" "any file path with spaces"')
이전에 this 및 other의 답변을 활용하여 여러 플랫폼에서 작동하는 인라인 코드가 있습니다.
import sys, os, subprocess
subprocess.call(('cmd /c start "" "'+ filepath +'"') if os.name is 'nt' else ('open' if sys.platform.startswith('darwin') else 'xdg-open', filepath))