Arg解析:将文件名解析为字符串(python)

时间:2016-09-06 13:00:22

标签: python string filenames argparse

我想将文件的名称解析为我的脚本,而不是直接将文件转换为对象。

以下是示例代码test.py

import argparse
import os.path
def is_valid_file(parser, arg):
      if not os.path.exists(arg):
           parser.error("The file %s does not exist! Use the --help flag for input options." % arg)
      else:
           return open(arg, 'r')

parser = argparse.ArgumentParser(description='test') 
parser.add_argument("-test", dest="testfile", required=True,
                    help="test", type=lambda x: is_valid_file(parser, x))
args = parser.parse_args()    
print args.testfile

testfile是一个.txt文件,其中包含:1,2,3,4

原则上希望print args.testfile将调用的testfile名称作为字符串返回:

$ python test.py -test test.txt
>> "test.txt"

为了实现这一点,我需要阻止argparser将test.txt转换为对象。我怎么能这样做?

非常感谢!

2 个答案:

答案 0 :(得分:1)

您可以按如下方式修改函数,以便在检查字符串后返回该字符串:

def is_valid_file(parser, arg):
      if not os.path.exists(arg):
           parser.error("The file %s does not exist! Use the --help flag for input options." % arg)
      else:
           return arg

还有一种更直接的方法:

parser.add_argument("-test", dest="testfile", required=True,
                    help="test", type=file)  # file exists in python 2.x only

parser.add_argument("-test", dest="testfile", required=True,
                    help="test", type=lambda f: open(f))  # python 3.x

args = parser.parse_args()    
print(args.testfile.name)  # name of the file from the file handle

实际上args.testfile是文件句柄,由argparser打开(如果未找到则为异常)。你可以直接阅读。

答案 1 :(得分:1)

FileType类型工厂完成了代码所做的大部分工作,并且消息机制略有不同:

In [16]: parser=argparse.ArgumentParser()
In [17]: parser.add_argument('-f',type=argparse.FileType('r'))

In [18]: args=parser.parse_args(['-f','test.txt'])
In [19]: args
Out[19]: Namespace(f=<_io.TextIOWrapper name='test.txt' mode='r' encoding='UTF-8'>)
In [20]: args.f.read()
Out[20]: '  0.000000,  3.333333,  6.666667, 10.000000, 13.333333, 16.666667, 20.000000, 23.333333, 26.666667, 30.000000\n'
In [21]: args.f.close()

对于有效名称,它会打开文件,您可以使用并关闭该文件。但是你不能在with上下文中使用它。

如果该文件不存在,则退出并使用cant open消息。

In [22]: args=parser.parse_args(['-f','test11.txt'])
usage: ipython3 [-h] [-f F]
ipython3: error: argument -f: can't open 'test11.txt': [Errno 2] No such file or directory: 'test11.txt'

FileType __call__使用argparse.ArgumentTypeError

处理错误
   except OSError as e:
        message = _("can't open '%s': %s")
        raise ArgumentTypeError(message % (string, e))

使用此错误机制,省略open我建议:

def valid_file(astring):
    if not os.path.exists(astring):
        msg = "The file %s does not exist! Use the --help flag for input options." % astring
        raise argparse.ArgumentTypeError(msg)
    else:
        return astring

可以用作:

In [32]: parser=argparse.ArgumentParser()
In [33]: parser.add_argument('-f',type=valid_file)

In [34]: args=parser.parse_args(['-f','test11.txt'])
usage: ipython3 [-h] [-f F]
ipython3: error: argument -f: The file test11.txt does not exist! Use the --help flag for input options.
An exception has occurred, use %tb to see the full traceback.

SystemExit: 2

In [35]: args=parser.parse_args(['-f','test.txt'])
In [36]: args
Out[36]: Namespace(f='test.txt')
In [37]: with open(args.f) as f:print(f.read())
  0.000000,  3.333333,  6.666667, 10.000000, 13.333333, 16.666667, 20.000000, 23.333333, 26.666667, 30.000000

http://bugs.python.org/issue13824担心FileType打开文件但不关闭文件。我提出了一个FileContext,模仿FileType,但不是打开文件,而是返回一个可以用作的对象:

with arg.file() as f:
    f.read()

它可以进行文件存在或创建测试,而无需实际打开或创建文件。这是一个更复杂的解决方案。