Python argparse dict arg

时间:2015-05-01 11:39:34

标签: python dictionary argparse

我想从命令行收到dict(str -> str)参数。 argparse.ArgumentParser提供它吗?还是其他任何图书馆?

对于命令行:

program.py --dict d --key key1 --value val1 --key key2 --value val2

我希望以下字典:

d = {"key1": "val1", "key2": "val2"}

8 个答案:

答案 0 :(得分:22)

这是使用自定义操作的另一种解决方案,如果您想以逗号分隔指定dict密钥对 -

import argparse
import sys
parser = argparse.ArgumentParser(description='parse key pairs into a dictionary')

class StoreDictKeyPair(argparse.Action):
     def __call__(self, parser, namespace, values, option_string=None):
         my_dict = {}
         for kv in values.split(","):
             k,v = kv.split("=")
             my_dict[k] = v
         setattr(namespace, self.dest, my_dict)

parser.add_argument("--key_pairs", dest="my_dict", action=StoreDictKeyPair, metavar="KEY1=VAL1,KEY2=VAL2...")

args = parser.parse_args(sys.argv[1:])
print args

运行:

python parse_kv.py --key_pairs 1=2,a=bbb,c=4 --key_pairs test=7,foo=bar

输出:

Namespace(my_dict={'1': '2', 'a': 'bbb', 'c': '4', 'test': '7', 'foo': 'bar'})

如果您想在字符串中使用 nargs 而不是逗号分隔值:

class StoreDictKeyPair(argparse.Action):
     def __init__(self, option_strings, dest, nargs=None, **kwargs):
         self._nargs = nargs
         super(StoreDictKeyPair, self).__init__(option_strings, dest, nargs=nargs, **kwargs)
     def __call__(self, parser, namespace, values, option_string=None):
         my_dict = {}
         print "values: {}".format(values)
         for kv in values:
             k,v = kv.split("=")
             my_dict[k] = v
         setattr(namespace, self.dest, my_dict)

parser.add_argument("--key_pairs", dest="my_dict", action=StoreDictKeyPair, nargs="+", metavar="KEY=VAL")

args = parser.parse_args(sys.argv[1:])
print args

运行

python arg_test4.py --key_pairs 1=2 a=bbb c=4 test=7 foo=bar

输出:

values: ['1=2', 'a=bbb', 'c=4', 'test=7', 'foo=bar']
Namespace(my_dict={'1': '2', 'a': 'bbb', 'c': '4', 'test': '7', 'foo': 'bar'})

答案 1 :(得分:16)

我会用这样的东西:

p = argparse.ArgumentParser()
p.add_argument("--keyvalue", action='append',
               type=lambda kv: kv.split("="), dest='keyvalues')

args = p.parse_args("--keyvalue foo=6 --keyvalue bar=baz".split())
d = dict(args.keyvalues)

你可以创建一个自定义动作,它会附加"追加"解析后的键值对直接进入字典,而不是简单地累积(key, value)元组列表。 (我看到的是skyline75489的作用;我的答案与使用自定义类型的单个--keyvalue选项而不是单独的--key--value选项来指定对时不同。)

答案 2 :(得分:3)

Python以数组argv的形式接收参数。您可以使用它在程序本身中创建字典。

import sys
my_dict = {}
for arg in sys.argv[1:]:
    key, val=arg.split(':')[0], arg.split(':')[1]
    my_dict[key]=val

print my_dict

对于命令行:

python program.py key1:val1 key2:val2 key3:val3

输出:

my_dict = {'key3': 'val3', 'key2': 'val2', 'key1': 'val1'}
  

注意:args将以字符串形式出现,因此您必须将它们转换为存储数值。

我希望它有所帮助。

答案 3 :(得分:2)

另一种简单的方法:

parser = argparse.ArgumentParser()
parser.add_argument('--key1')
parser.add_argument('--key2')
args = parser.parse_args()
my_dict = args.__dict__

答案 4 :(得分:1)

Python单行argparse字典参数argparse_dictionary.py

# $ python argparse_dictionary.py --arg_dict=1=11,2=22;3=33 --arg_dict=a=,b,c=cc,=dd,=ee=,
# Namespace(arg_dict={'1': '11', '2': '22', '3': '33', 'a': '', 'c': 'cc', '': 'dd'})

import argparse

arg_parser = argparse.ArgumentParser()
arg_parser.add_argument('--arg_dict', action=type(b'', (argparse.Action,), dict(__call__=lambda self, parser, namespace, values, option_string: getattr(namespace, self.dest).update(dict([v.split('=') for v in values.replace(';', ',').split(',') if len(v.split('=')) == 2])))), default={}, metavar='KEY1=VAL1,KEY2=VAL2;KEY3=VAL3...')
print(arg_parser.parse_args())

答案 5 :(得分:1)

如果您只是想将argparse输入转换为字典,那么Python 3.6中有一个简单的解决方案。一个例子如下:

import argparse 

parser = argparse.ArgumentParser()
parser.add_argument('-i', '--input', help='the path to the input file')
parser.add_argument('-o', '--output', help='the path to the output file')
args = parser.parse_args()

arguments = dict(args._get_kwargs())

for k, v in arguments.items():
    print(k, v) 

给出诸如python3 script_name.py --input 'input.txt' --output 'output.txt'之类的命令行输入,代码将输出到终端:

input input.txt
output output.txt

答案 6 :(得分:1)

使用vars

function runAccessibilityTest(fileName, elem) {
  const fileDelimiter = ",";
  fileName = fileName.split(".")[0] + "_" + getDateTime() + ".csv";
  browser.execute(axeSource);
  const options = { runOnly: { type: "tag", values: ["wcag2aa"] } };
  let results = browser.executeAsync(function (options, done) {
    // run axe on our site
    axe.run(
      {
        include: [[elem]],
      },
      function (err, results) {
        if (err) done(err);
        done(results);
      }
    );
  }, options);
}

答案 7 :(得分:-1)

对于像argparsedocoptclick这样的当前库,它们都不支持使用dict args。我能想到的最好的解决方案是自定义argparse.Action以支持它自己:

import argparse

class MyAction(argparse.Action):
    def __init__(self, option_strings, dest, nargs=None, **kwargs):
        super(MyAction, self).__init__(option_strings, dest, nargs, **kwargs)

    def __call__(self, parser, namespace, values, option_string=None):
        print '%r %r %r' % (namespace, values, option_string)
        value_dict = {}
        values.reverse()
        while len(values) > 0:
            v = eval(values.pop()).lstrip('--') # This is like crazy hack, I know.
            k = eval(values.pop())
            value_dict[k] = v

        setattr(namespace, self.dest, value_dict)

parser = argparse.ArgumentParser()
parser.add_argument('-d', action=MyAction, nargs='*')
args = parser.parse_args('-d "--key" "key1" "--value" "val1" "--key" "key2" "--value" "val2"'.split())

print(args)