Python:告别if...else...做真正面向对象的编程
欢迎转载,转载请注明原文地址:http://blog.csdn.net/majianfei1023/article/details/51417564
最近在给服务器做一套指令系统,大家都知道一个服务器的指令有很多,像(加金币,加道具,加经验...)
代码就像这样:
cmd = REQUEST.get("cmd", "") if cmd == "add_gold": pass elif cmd == "add_item": pass elif cmd == "add_exp": pass ... ... else: send("cmd unknow") exit(9)
然而一个普通的游戏,指令起码也得上百,于是乎我们就拖着鼠标从头拉到尾看看if else有没有我们需要的指令。而且,如果多人维护同一个系统,比如A要加一个指令,B也要加一个指令,这个时候就会有SVN冲突的隐患。
更重要的是,python作为一门面向对象的语言,难道就只能用if else处理这种问题吗,这完全是面向过程的c语言编程方式。既然作为面向对象的语言,我们就用面向对象的方法来解决问题。
什么是面向对象:
把一组数据结构和处理它们的方法组成对象(object),把相同行为的对象归纳为类(class),通过类的封装(encapsulation)隐藏内部细节,通过继承(inheritance)实现类的特化(specialization)/泛化(generalization),通过多态(polymorphism)实现基于对象类型的动态分派(dynamic dispatch)。也就是说,我们给每个指令构建一个对象,然后每接收一个指令的时候就直接通过这个对象去处理。每加一个指令,我们创建一个文件,构造一个对象,这样代码看起来会简洁很多,而且也不会有冲突的隐患。
问题是,怎么避免if else呢。因为有这么多指令,不用if else真的有办法解决吗?
答案是有的,我们把构造的对象自动注册到一个数据结构中,如果查到cmd在这个数据结构中,那么就直接执行这个对象的特定方法就好了,想想c++的多态就明白了。
那么现在的问题就是怎么自动注册到一个数据结构中。
我们扫描同一个目录下的所有文件,找到某一个指定基类的所有子类,然后获取一个列表。
就像下面这样:
def _get_module_files(module_dir): module_name_set = set() try: files = os.listdir(module_dir) except: print "error in generate_module_list for directory:", module_dir return () for fileName in files: list = fileName.split(".") if len(list) == 2: module_name = list[0] extension = list[1] if extension in ("py", "pyc"): module_name_set.add(module_name) module_name_set.discard("__init__") return module_name_set def _get_class_list(module, base_class): """得到模块里面所有属于指定类子类的类""" m = __import__(module, fromlist=[""]) cls_name = module cls = getattr(m, cls_name, None) if issubclass(cls, base_class): return cls return None def scan_classes(module_dir, base_class): """主要功能就是扫描一个目录 , 并把下面所有python模块中属于指定类子类的类找出来""" class_dict = {} module_set = _get_module_files(module_dir) for module in module_set: clist = _get_class_list(module, base_class) if clist is None: continue class_dict[clist.__name__] = clist return class_dict
我们的目录结构如下。
cmds:
--__int__.py
--add_gold.py
--add_item.py
--add_exp.py
--base_cmd.py
我们就可以写出以下代码:
from cmds.base_cmd import base_cmd member_path = os.path.split(os.path.realpath(__file__))[0] + cmds" class_dict = scan_classes(member_path, base_cmd) #然后我们每个类继承自base_cmd,写一个通用的接口deal_cmd. cmd = REQUEST.get("cmd", "") if cmd in class_dict: a = class_dict[cmd]() a.deal_cmd(REQUEST.__dict__["dict"])
ok,大功告成,以后每加一个指令,只需要在cmds目录增加一个**.py,继承自base_cmd,我们重写deal_cmd()方法就可以了。
对比之前与现在的两种写法,之前的写法感觉真的是弱爆了。
声明:该文观点仅代表作者本人,入门客AI创业平台信息发布平台仅提供信息存储空间服务,如有疑问请联系rumenke@qq.com。
- 上一篇:没有了
- 下一篇: 二级指针的作用详解