前言
今天在写代码的时候遇到了一个关于parser的一些小坑,记录在此备用。
我们知道在python中可以用argprase来传递一些参数给代码执行,来看下面的例子,假设现在有一个test文件夹,下面有3个python文件,分别用a.py;b.py;c.py来表示,目录树如下。
1 2 3 4
| test ├── a.py ├── b.py ├── c.py
|
每一个的初始代码为一个简单的print函数。
1 2 3 4 5 6 7
| def out_a(): print("I am a.py")
if __name__ == '__main__': out_a()
|
1 2 3 4 5 6 7
| def out_b(): print("I am b.py")
if __name__ == '__main__': out_b()
|
1 2 3 4 5 6 7
| def out_c(): print("I am c.py")
if __name__ == '__main__': out_c()
|
现在在a.py中引入模块argprase,并定义一些简单的参数,代码如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| import argparse parser = argparse.ArgumentParser() parser.add_argument('--first_parameter', default='first') parser.add_argument('--second_parameter', default='second') parser.add_argument('--third_flag', action='store_true') args = parser.parse_args()
def out_a(): print("I am a.py")
if __name__ == '__main__': out_a()
|
这里面简单说一下第3个参数,这也是我今天想记录文章的原因,这个参数是argparse里面提供的开关布尔选项,actions记录的是一个动作,意味着在调用这个函数的时候,如果在命令行添加这个参数,则该参数为True,如果不添加这个参数,则该参数为False,归纳起来为如下的两个图。
这个是没有指定第3个参数的情况

这个是指定第3个参数的情况

对于这种开关布尔选项更为详细的介绍,可以参考知乎问题:Argparse中action的可选参数store_true,store_false到底是什么意思?
到目前为止没有出现问题,接下来,我希望b.py也使用参数,并且还希望使用a.py里面的函数,因此我对b.py进行如下修改。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| from a import out_a import argparse parser = argparse.ArgumentParser() parser.add_argument('--fourth_parameter', default='fourth') parser.add_argument('--fifth_parameter', default='fifth') parser.add_argument('--sixth_flag', action='store_true') args = parser.parse_args()
def out_b(): print("I am b.py")
if __name__ == '__main__': out_b() out_a()
|
然后同样的,我们分别用两种方式来测试b.py,效果如下。
这个是不使用参数的情况

这个是使用参数的情况

可以看到报错了,当时我看到这里的时候想了很久,排除了拼写错误的情况以后,观察这里面的输出,发现看到的是a.py当中的3个参数,而不是b.py当中设置的参数,于是我将a.py和b.py的参数表打印出来,看到这样子的结果。
输出两个python文件的参数表

可以发现尽管我使用的是from a import out_a,但依然引入了a.py当中的参数表,并且后引入的b.py的参数表没有办法覆盖掉。下面在c.py中同样引入3个参数,然后引入b.py的方法,代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| from b import out_b import argparse parser = argparse.ArgumentParser() parser.add_argument('--seventh_parameter', default='seventh') parser.add_argument('--eighth_parameter', default='eighth') parser.add_argument('--ninth_flag', action='store_true') args = parser.parse_args() print(args)
def out_c(): print("I am c.py")
if __name__ == '__main__': out_c() out_b()
|
效果如下
不使用任何参数调用c.py

看到有3个参数列表输出就知道c.py的参数也是无效的了,验证一下。
使用参数调用c.py

解决方案:
其实只需要将所有的参数表放到同一个文件里面就可以了,比如utils.py,由于这里是同一个文件夹下的3个文件,在import调用的时候就只需要初始化一次所有参数就可以使用了,有点类似于C语言当中的全局变量,因为这个东西排查了一个下午,也是有点恼火了。