Pythonの作業ディレクトリを変更するにはどうすればよいですか?


696

cd 作業ディレクトリを変更するためのシェルコマンドです。

Pythonで現在の作業ディレクトリを変更するにはどうすればよいですか?


2
だから通訳でos.chdir(os.path.join(os.path.abspath(os.path.curdir),u'subfolder'))-または?
Mr_and_Mrs_D 2014

回答:


765

次のコマンドで作業ディレクトリを変更できます。

import os

os.chdir(path)

この方法を使用する際に従うべき2つのベストプラクティスがあります。

  1. 無効なパスで例外(WindowsError、OSError)をキャッチします。例外がスローされた場合は、再帰的な操作、特に破壊的な操作を実行しないでください。それらは新しいパスではなく古いパスで動作します。
  2. 完了したら、古いディレクトリに戻ります。これは、ブライアンM.ハントが彼の回答で行ったように、コンテキストマネージャーでchdir呼び出しをラップすることにより、例外セーフな方法で実行できます。

サブプロセスの現在の作業ディレクトリを変更しても、親プロセスの現在の作業ディレクトリは変更されません。これはPythonインタープリターにも当てはまります。を使用os.chdir()して、呼び出しプロセスのCWDを変更することはできません。


3
cdunn2001の軽量なデコレータベースの回答は、最新のPythonにとって理想的なアプローチです。上記の答えはその理由を示しています。自分が何をしているのかわかっていると思わない限り、コンテキストマネージャーの外部で呼び出さos.chdir()ないでください。(あなたはおそらくそうしません。
セシルカレー

6
これはインタラクティブシェルで最も簡単な方法だと思います。Windowsでは、次のようにスラッシュを使用する必要があることに注意してくださいos.chdir("C:/path/to/location")
Josiah '25

注意すべきことの1つは、Pythonプログラムを実行可能ファイルにしてcronで実行すると、ホームディレクトリで起動することです。したがって、完全修飾パスを使用するのが最善です。これは間違いなく機能しますが、Pythonから呼び出す可能性のあるスクリプトでは、完全修飾パスを使用します。これは、これがPythonプログラム自体の外部で適用される保証がないためです。
SDsolar 2017

310

次に、作業ディレクトリを変更するためのコンテキストマネージャの例を示します。他の場所で参照されている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.

FROMを変更したディレクトリを知る必要がある場合return selfは、の最後に追加するだけです__enter__。そのようにしてwith cd('foo') as cm:、以前のディレクトリにアクセスしてアクセスできますcm.savedPath
Sam F

古いディレクトリ(「savedPath」に保存されているディレクトリ)に戻ることができない場合があることに注意してください。たとえば、特権レベルの高いプロセスが特権レベルの低いプロセスを実行する場合、2番目のプロセスが独自の機能でその作業ディレクトリに入ることができない場合でも、2番目のプロセスは最初のプロセスの作業ディレクトリを継承します。
カイペツケ


117

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'.

3
また、この潜在的な失敗に注意してください(を忘れるtry/finally)。
cdunn2001

5
輝き!受け入れられた回答からの紹介的なコメントがこの回答に挿入された場合、これは計り知れないほど理想的です。それでも、この答えは簡潔で、Pythonで安全な実装なので、私が与えなければならないすべての賛成投票が保証されます。
セシルカレー

3
なぜyieldありませんかreturn?これはジェネレータであるはずですか?
EKons 2016

収益と収益の関連性についてコメントしてください!
NicoBerrogorry 2017

1
@NicoBerrogorry、それはジェネレーターです。contextlib.contextmanagerのドキュメントを参照してください。これはPythonで非常に役立つパターンであり、学ぶ価値があります。
cdunn2001

25

あなたは、Pythonの比較的新しいバージョンを使用している場合、あなたはまたのような、コンテキストマネージャを使用することができ、この1

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)

1
良い一般的な考え。ここで、他の依存関係のないActivestateレシピ
cfi 2012

4
依存関係は悪いです。Pythonの組み込みcontextlib.contextmanagerデコレーターは優れています。cdunn2001デコレータベースの回答を参照してください。これは、今では理想的な回答です。
セシルカレー

14

他の人がすでに指摘したように、上記のすべてのソリューションは現在のプロセスの作業ディレクトリを変更するだけです。これは、Unixシェルに戻ったときに失われます。絶望的な場合、この恐ろしいハックで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")

4
めちゃくちゃ、壊れやすいハックは必須の投票を取得します。特に「実行中にユーザーが入力した場合...」の警告では特に、これを行うべきではありません。それでも、親のCWDを変更すること一種のことですが、実際には実行可能ではないことを理解するために、私の反逆の首ひげを刺激します。賛成票!みんなに賛成投票!
セシルカレー


11

os.chdir()はのPythonicバージョンですcd


8
import os

abs_path = 'C://a/b/c'
rel_path = './folder'

os.chdir(abs_path)
os.chdir(rel_path)

両方をos.chdir(abs_path)またはos.chdir(rel_path)で使用できます。相対パスを使用するためにos.getcwd()を呼び出す必要はありません。


うまくいきます。一つは..前とディレクトリを変更した後、両方のカレントディレクトリを確認するために)(os.getcwd使用することができます
vinsinraw


3

「cd ..」オプションなどを実行する場合は、次のように入力します。

os.chdir( "..")

Windowsのcmdと同じです:cd ..もちろん、osのインポートが必要です(たとえば、コードの1行目として入力してください)。


0

spyderとlove GUIを使用している場合は、画面の右上隅にあるフォルダーボタンをクリックするだけで、現在のディレクトリとして必要なフォルダー/ディレクトリに移動できます。その後、spyder IDEのウィンドウのファイルエクスプローラータブに移動すると、そこに存在するすべてのファイル/フォルダーを確認できます。現在の作業ディレクトリを確認するには、spyder IDEのコンソールに移動し、次のように入力します

pwd

以前に選択したのと同じパスが印刷されます。


-1

スクリプトプロセスの現在のディレクトリを変更するのは簡単です。問題は、Pythonスクリプトが呼び出されるコマンドウィンドウの現在のディレクトリを変更する方法です。これは非常に難しいと思います。WindowsのBatスクリプトまたはBashシェルのBashスクリプトは、シェル自体がインタープリターであるため、通常のcdコマンドでこれを実行できます。WindowsとLinuxの両方でPythonはプログラムであり、プログラムが親の環境を直接変更することはできません。ただし、簡単なシェルスクリプトと、ほとんどの難しい処理を行うPythonスクリプトを組み合わせることで、目的の結果を得ることができます。たとえば、後方/前方/選択の再訪のためのトラバーサル履歴を持つ拡張cdコマンドを作成するために、単純なbatスクリプトによって呼び出される比較的複雑なPythonスクリプトを記述しました。走査リストはファイルに保存され、最初の行にターゲットディレクトリを指定します。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)

これは良い答えですが、OPは親プロセスのCWDを変更することではないという答えを選択しました。これにより、質問の意味に関する混乱が解消されます。
ティンマン

ティン・マンへ-私が提案を投稿する前にその答えが選ばれた。幅広い回答が混乱していたのではないかと思います。特定のプロセス(つまりpythonスクリプト)内のcdは非常に単純なので、なぜだれかがそれを要求するのかわかりません。
David McCracken

1
実際、その答えは何年も前に選択されました。それが適切でなければ、それ以来何度も呼び出されたでしょう。
ティンマン

混乱が残っていると思います。最近では、「Linuxの「cd」コマンドをpythonでシミュレートし、プログラムの終了後もディレクトリの変更を維持する[重複]」という質問は、ここで回答されたため却下されましたが、実際、この質問は選択された回答では扱われていません。私の提案はWindowsに関するものですが、問題はLinuxでも同じです。
David McCracken
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.