两个 Python 的冷技巧(2)

                赖勇浩(http://laiyonghao.com)

先来看两句代码
record = cursor.execute('select * from tbl where id = 123456').fecth_all()[0]msg = struct.unpack('!I', buff)[0]
不知道大家怎么样看,我觉得有这两个 [0] 跟在后面,还是蛮丑的。其实可以利用 Python 赋值语句简写:
record, = cursor.execute('select * from tbl where id = 123456').fecth_all()msg, = struct.unpack('!I', buff)
注意 record 和 msg 后面的逗号,可以利用它让 Python 把这个赋值语句理解为多对多的赋值,但可省去后续的 [0] 了。
这种写法,据我以前的同事测试,比使用 [0] 要慢上 20% 左右。如果你很在意,建议使用之前的写法。另外,两种写法到底哪个更漂亮、更 Pythonic,也没有定论,但我喜欢后者。

用 ConfigPaser 是解释 ini 文件的利器,因为有这么好用库,所以我写的项目基本上都是用 ini 文件来配置的。ConfigParser 支持 DEFAULT 节,写在 DEFAULT 节中的配置项可以作为实参替换后续的引用,比如在我之前提过的“ 棋牌OnWeb”项目中,我们使用这样的配置文件:
[DEFAULT]base_path= /home/qipai-v1/src/server[policy]switch = onport = 18843path = %(base_path)s/hall/config/flashpolicy.xml
其中 policy.path 的值取出来的时候就变成了 /home/qipai-v1/src/server/hall/config/flashpolicy.xml,ConfigParser 会自动为你做好替换。
不过在这里要讲的不是这个技巧。如果你有留意到 ConfigParser.ConfigParser 类的那个 defaults 参数,那就是真正的我要讲的东西了。defaults 参数可以补充 ini 文件中 DEFAULT 文件的不足,能够运行时再决定变量的实参。比如在“ 棋牌OnWeb”项目中,所有的游戏进程都由一个叫 desk 的程序来服务的,它通过命令行传入的参数加载不同的插件实现不同的业务逻辑。比如 desk --game-name=doudizhu 可以运行一个斗地主的服务,desk --game-name=xiangqi 则是中国象棋。显然,为了方便通过日志分析错误,有必要把日志按游戏、房间和桌子分离开来,比如 xiangqi-1-10.log 表示象棋游戏的房间1桌子10的进程的日志,这时候如果使用内置的 logging 模块来记录日志,我们可以这样配置它:
[handler_game]class=handlers.TimedRotatingFileHandlerformatter=commargs=('log/%(game_name)s-%(rid)d-%(did)d.log', 'midnight', 1, 30)
在程序运行的时候,我们这样初始化 logging:
logging.config.fileConfig(conf, defaults = dict(game_name = game_name, rid = rid, did = did))
把 rid 和 did 通过命令行参数传过来即可。之所以能够这样做,完全因为 logging 也是使用 ConfigParser 来解释它的配置文件的呀,Python 吸引人之一的地方就是这样的:尽量重用,更加灵活。


           

猜你喜欢

转载自blog.csdn.net/qq_44925249/article/details/89788959