【问题标题】:How to Explicitly Specify File Directory in Python Script如何在 Python 脚本中显式指定文件目录
【发布时间】:2016-09-25 04:36:12
【问题描述】:

我试图通过在 Windows 上的运行对话框中键入文件名来直接运行我的 Python 脚本,但我无法有效地做到这一点。在阅读 Python 上的 Automate 书籍后,我创建了两个简单的脚本:一个是应该接受命令行参数的剪贴板,另一个是应该生成文本文件的脚本。我已经修改了环境变量以适应这些文件的路径,甚至还为它们添加了一个 shebang 行。我还尝试为每个文件创建一个批处理文件。然而,当我尝试运行它们中的任何一个时,尤其是使用运行对话框的不是 .pyw 文件的文本生成脚本时,它会非常简短地显示命令提示符,但实际上并没有生成文件。当我双击它时,脚本运行得很好,从命令行或 Python shell 运行它。我该如何解决这个问题?

如果可以使用,这是文本生成脚本的代码:

#! /usr/bin/python

import random

   # The quiz data. Keys are states and values are their capitals.

capitals = {'Alabama': 'Montgomery', 'Alaska': 'Juneau', 'Arizona': 'Phoenix',
'Arkansas': 'Little Rock', 'California': 'Sacramento', 'Colorado': 'Denver',
'Connecticut': 'Hartford', 'Delaware': 'Dover', 'Florida': 'Tallahassee',
'Georgia': 'Atlanta', 'Hawaii': 'Honolulu', 'Idaho': 'Boise', 'Illinois':
'Springfield', 'Indiana': 'Indianapolis', 'Iowa': 'Des Moines', 'Kansas':
'Topeka', 'Kentucky': 'Frankfort', 'Louisiana': 'Baton Rouge', 'Maine':
'Augusta', 'Maryland': 'Annapolis', 'Massachusetts': 'Boston', 'Michigan':
'Lansing', 'Minnesota': 'Saint Paul', 'Mississippi': 'Jackson', 'Missouri':
'Jefferson City', 'Montana': 'Helena', 'Nebraska': 'Lincoln', 'Nevada':
'Carson City', 'New Hampshire': 'Concord', 'New Jersey': 'Trenton',
'New Mexico': 'Santa Fe', 'New York': 'Albany', 'North Carolina': 'Raleigh',
'North Dakota': 'Bismarck', 'Ohio': 'Columbus', 'Oklahoma': 'Oklahoma City',
'Oregon': 'Salem', 'Pennsylvania': 'Harrisburg', 'Rhode Island': 'Providence',
'South Carolina': 'Columbia', 'South Dakota': 'Pierre', 'Tennessee':
'Nashville', 'Texas': 'Austin', 'Utah': 'Salt Lake City', 'Vermont':
'Montpelier', 'Virginia': 'Richmond', 'Washington': 'Olympia',
'West Virginia': 'Charleston', 'Wisconsin': 'Madison', 'Wyoming': 'Cheyenne'}

for quiz_num in range(35):
    quiz_file = open('quiz_questions %s.txt' % (quiz_num+1), 'w')
    quiz_answers = open('quiz_answers %s.txt' % (quiz_num+1), 'w')
quiz_file.write('Name:\nMatric no:\nDate: \n \n')
quiz_file.write(' '*40 + "Ajanlekoko Nursery and Primary School \n \n")

states = list(capitals.keys())
random.shuffle(states)

for qst_num in range (50):
    right_answer = capitals[states[qst_num]]
    wrong_answers = list(capitals.values())
    del wrong_answers[wrong_answers.index(right_answer)]
    wrong_answers = random.sample(wrong_answers, 3)
    options = wrong_answers + [right_answer]
    random.shuffle(options)

    quiz_file.write('%s. What is the capital of %s?' % (qst_num+1, states[qst_num]))

    for i in range(4):
        quiz_file.write('\n \n  %s. %s \n' % ('ABCD'[i], options[i]) )
    quiz_file.write('\n')

    quiz_answers.write('%s. %s \n' % (qst_num+1, 'ABCD'[options.index(right_answer)]) )

quiz_file.close()
quiz_answers.close()

【问题讨论】:

  • 如果您通过搜索 PATH 而不是使用绝对路径来运行脚本来查找脚本,则资源管理器可能会将工作目录设置为用户的配置文件目录或 %SystemRoot%\System32。我刚刚签入了 Windows 10,它是用户配置文件目录,但我不会依赖它。检查print(os.getcwd()); input()
  • 顺便说一句,您的问题中是否有缩进拼写错误,或者您的代码在range(35) 循环中是否存在错误?
  • 如果你想创建相对于脚本目录的文件,使用script_dir = os.path.abspath(os.path.dirname(__file__))。如果您希望它们与用户的个人资料相关,请使用home_dir = os.path.expanduser('~')
  • 谢谢,埃克森。我现在可以看到,不是脚本没有运行,而是文件是在您提到的用户配置文件目录下创建的。如何让它在脚本自己的目录中创建文件?我不明白我应该用 script_dir = os.path.abspath(os.path.dirname(file)) 做什么(请原谅我的天真;我是新手)。如果它是我面临的问题的解决方案,我应该把它放在哪里?再次感谢您提供帮助。 :)
  • 只使用完全限定路径而不是使用相对路径。例如:quiz_file_path = os.path.join(script_dir, 'quiz_questions %02d.txt' % (quiz_num + 1))。当您知道您对脚本目录具有写入权限时,将文件保存到脚本目录供个人使用是可以的,但通常非管理员用户可能缺乏写入权限,例如位于“Program Files”子目录中的脚本。您可以默认为当前目录,如果这是交互式命令行使用的首选,但提供命令行选项来显式设置输出目录。

标签: python windows file file-io path


【解决方案1】:

经过一番挖掘,我发现这是我问题的最简单答案:

import random
import os
import sys

os.chdir(os.path.dirname(sys.argv[0]))

这会在脚本运行时将工作目录更改为第一个命令行参数(脚本的名称)。

【讨论】:

  • script_dir = os.path.abspath(os.path.dirname(sys.argv[0])) 有效,并且即使您使用 py2exe 将脚本冻结为可执行文件,也将继续有效。但我不鼓励改变工作目录的习惯。一旦您开始使用多个编写器编写多线程代码,可能会写入不同的目录,它就不是线程安全的。此外,在 Windows 中,这意味着最终的绝对路径仅限于 MAX_PATH (260) 个字符;将其扩展为 32768 个字符的 \\?\ 前缀仅适用于完全限定的路径。
猜你喜欢
  • 1970-01-01
  • 2011-01-18
  • 2018-10-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多