Как изменить каталог (cd) в Python?

cd как в команде shell для изменения рабочего каталога.

Как изменить текущий рабочий каталог в Python?

13 ответов


Вы можете изменить рабочий каталог с:

import os

os.chdir(path)

при использовании этого метода необходимо следовать двум рекомендациям:

  1. поймать исключение (WindowsError, OSError) по недопустимому пути. Если возникает исключение, не выполняйте никаких рекурсивных операций, особенно деструктивных. Они будут действовать по старому пути, а не по новому.
  2. вернуться в свой старый каталог, когда вы закончите. Это можно сделать в исключени-безопасном образе мимо упаковка вашего вызова chdir в контекстном менеджере, как это сделал Брайан М. Хант в ответ.

изменение текущего рабочего каталога в подпроцесс не изменяет текущий рабочий каталог родительского процесса. Это относится и к интерпретатору Python. Вы не можете использовать os.chdir() для изменения CWD вызывающего процесса.


вот пример контекстного менеджера для изменения рабочего каталога. Это проще, чем версия ActiveState упоминается в другом месте,но это делает работу.

Контекст-Менеджера: cd

import os

class cd:
    """Context manager for changing the current working directory"""
    def __init__(self, newPath):
        self.newPath = os.path.expanduser(newPath)

    def __enter__(self):
        self.savedPath = os.getcwd()
        os.chdir(self.newPath)

    def __exit__(self, etype, value, traceback):
        os.chdir(self.savedPath)

или более краткий эквивалент (ниже), используя ContextManager.

пример

import subprocess # just to call an arbitrary command e.g. 'ls'

# enter the directory like this:
with cd("~/Library"):
   # we are in ~/Library
   subprocess.call("ls")

# outside the context manager we are back wherever we started.

Я хотел бы использовать os.chdir такой:

os.chdir("/path/to/change/to")

кстати, если вам нужно выяснить свой текущий путь, используйте os.getcwd().

больше здесь


cd() легко писать с помощью генератора и оформителя.

from contextlib import contextmanager
import os

@contextmanager
def cd(newdir):
    prevdir = os.getcwd()
    os.chdir(os.path.expanduser(newdir))
    try:
        yield
    finally:
        os.chdir(prevdir)

затем каталог возвращается даже после исключения:

os.chdir('/home')

with cd('/tmp'):
    # ...
    raise Exception("There's no place like home.")
# Directory is now back to '/home'.

Если вы используете относительно новую версию Python, вы также можете использовать контекстный менеджер, например этот:

from __future__ import with_statement
from grizzled.os import working_directory

with working_directory(path_to_directory):
    # code in here occurs within the directory

# code here is in the original directory

обновление

Если вы предпочитаете свернуть свой собственный:

import os
from contextlib import contextmanager

@contextmanager
def working_directory(directory):
    owd = os.getcwd()
    try:
        os.chdir(directory)
        yield directory
    finally:
        os.chdir(owd)

os.chdir() Это правильный путь.


Как уже указывали другие, все решения выше изменяют только рабочий каталог текущего процесса. Это теряется при выходе обратно в оболочку Unix. Если отчаянный вы can измените каталог родительской оболочки в Unix с помощью этого ужасного взлома:

def quote_against_shell_expansion(s):
    import pipes
    return pipes.quote(s)

def put_text_back_into_terminal_input_buffer(text):
    # use of this means that it only works in an interactive session
    # (and if the user types while it runs they could insert characters between the characters in 'text'!)
    import fcntl, termios
    for c in text:
        fcntl.ioctl(1, termios.TIOCSTI, c)

def change_parent_process_directory(dest):
    # the horror
    put_text_back_into_terminal_input_buffer("cd "+quote_against_shell_expansion(dest)+"\n")

os.chdir() - Это обновления версии cd.


далее в направлении, указанном Брайаном и основанном на ш (1.0.8+)

from sh import cd, ls

cd('/tmp')
print ls()

Если вы хотите выполнить что-то вроде "cd.."вариант, просто введите:

os.chdir ("..")

Это то же самое, что и в Windows cmd: cd.. Конечно!--3-->импорт ОС необходимо (e.G введите его как 1-ю строку кода)


изменение текущего каталога процесса сценария тривиально. Я думаю, что вопрос в том, как изменить текущий каталог командного окна, из которого вызывается скрипт python, что очень сложно. Сценарий Bat в Windows или сценарий Bash в оболочке Bash могут сделать это с помощью обычной команды cd, потому что сама оболочка является интерпретатором. В Windows и Linux Python-это программа, и никакая программа не может напрямую изменить среду своего родителя. Однако сочетание простого скрипта оболочки с скриптом Python, выполняющим большую часть сложных вещей, может достичь желаемого результата. Например, чтобы сделать расширенную команду cd с историей обхода для backward/forward/select revisit, я написал относительно сложный скрипт Python, вызванный простым скриптом bat. Список обхода хранится в файле с целевым каталогом в первой строке. Когда скрипт python возвращается, скрипт bat считывает первую строку файла и делает его аргументом cd. Полный сценарий bat (минус комментарии для краткости):

if _%1 == _. goto cdDone
if _%1 == _? goto help
if /i _%1 NEQ _-H goto doCd
:help
echo d.bat and dSup.py 2016.03.05. Extended chdir.
echo -C = clear traversal list.
echo -B or nothing = backward (to previous dir).
echo -F or - = forward (to next dir).
echo -R = remove current from list and return to previous.
echo -S = select from list.
echo -H, -h, ? = help.
echo . = make window title current directory.
echo Anything else = target directory.
goto done

:doCd
%~dp0dSup.py %1
for /F %%d in ( %~dp0dSupList ) do (
    cd %%d
    if errorlevel 1 ( %~dp0dSup.py -R )
    goto cdDone
)
:cdDone
title %CD%
:done

скрипт python, dSup.py есть:

import sys, os, msvcrt

def indexNoCase ( slist, s ) :
    for idx in range( len( slist )) :
        if slist[idx].upper() == s.upper() :
            return idx
    raise ValueError

# .........main process ...................
if len( sys.argv ) < 2 :
    cmd = 1 # No argument defaults to -B, the most common operation
elif sys.argv[1][0] == '-':
    if len(sys.argv[1]) == 1 :
        cmd = 2 # '-' alone defaults to -F, second most common operation.
    else :
        cmd = 'CBFRS'.find( sys.argv[1][1:2].upper())
else :
    cmd = -1
    dir = os.path.abspath( sys.argv[1] ) + '\n'

# cmd is -1 = path, 0 = C, 1 = B, 2 = F, 3 = R, 4 = S

fo = open( os.path.dirname( sys.argv[0] ) + '\dSupList', mode = 'a+t' )
fo.seek( 0 )
dlist = fo.readlines( -1 )
if len( dlist ) == 0 :
    dlist.append( os.getcwd() + '\n' ) # Prime new directory list with current.

if cmd == 1 : # B: move backward, i.e. to previous
    target = dlist.pop(0)
    dlist.append( target )
elif cmd == 2 : # F: move forward, i.e. to next
    target = dlist.pop( len( dlist ) - 1 )
    dlist.insert( 0, target )
elif cmd == 3 : # R: remove current from list. This forces cd to previous, a
                # desireable side-effect
    dlist.pop( 0 )
elif cmd == 4 : # S: select from list
# The current directory (dlist[0]) is included essentially as ESC.
    for idx in range( len( dlist )) :
        print( '(' + str( idx ) + ')', dlist[ idx ][:-1])
    while True :
        inp = msvcrt.getche()
        if inp.isdigit() :
            inp = int( inp )
            if inp < len( dlist ) :
                print( '' ) # Print the newline we didn't get from getche.
                break
        print( ' is out of range' )
# Select 0 means the current directory and the list is not changed. Otherwise
# the selected directory is moved to the top of the list. This can be done by
# either rotating the whole list until the selection is at the head or pop it
# and insert it to 0. It isn't obvious which would be better for the user but
# since pop-insert is simpler, it is used.
    if inp > 0 :
        dlist.insert( 0, dlist.pop( inp ))

elif cmd == -1 : # -1: dir is the requested new directory.
# If it is already in the list then remove it before inserting it at the head.
# This takes care of both the common case of it having been recently visited
# and the less common case of user mistakenly requesting current, in which
# case it is already at the head. Deleting and putting it back is a trivial
# inefficiency.
    try:
        dlist.pop( indexNoCase( dlist, dir ))
    except ValueError :
        pass
    dlist = dlist[:9] # Control list length by removing older dirs (should be
                      # no more than one).
    dlist.insert( 0, dir ) 

fo.truncate( 0 )
if cmd != 0 : # C: clear the list
    fo.writelines( dlist )

fo.close()
exit(0)

#import package
import os

#change directory
os.chdir('my_path')

#get location 
os.getcwd()

кроме того, хорошо проверить все другие полезные команды в пакете ОС здесь https://docs.python.org/3/library/os.html


и для удобного интерактивного использования, оболочкой IPython имеет все общие команды оболочки, встроенные.