基于python的opcode优化和模块按需加载机制研究(学习与个人思路)(原创)

基于python的opcode优化和模块按需加载机制研究(学习与思考)

姓名:XXX

学校信息:XXX

主用编程语言:python3.5

个人技术博客:http://www.cnblogs.com/Mufasa/

文档转换为PDF有些图片无法完全显示,请移步我的博客查看

完成时间:2019.03.06

 

本项目希望您能完成以下任务:

  - 优化python字节码解析代码,从底层提升python脚本运行效率;底层、编译器、虚拟机

  - 基本思路可以统计游戏常用opcode指令,进行类似opcode合并,opcode排序;

  - 另外,可以研究下指令预测相关资料,比如indirect threading,寻找更优的机制;(自然语言处理里面的东西好像可以用,类似语言翻译)

  - 为了缩短应用的启动时间,需要在应用启动时,把模块进行按需加载(或者延迟加载lazy import)优化启动项

  - 目前的不同实现主要是针对 Python 标准库进行处理,对第三方扩展库,尤其是游戏引擎相关的扩展支持不好,甚至无法支持;软件适配&通用化

  - 此课题不仅有一定的学术研究意义,更在手游等App中有很好的实用价值意义价值

  - 希望在自适应学习的基础上,能够做到按需加载自适应

 

一,Python 字节码

  Python 源代码文件.py结尾,字节码文件.pyc结尾;其中字节码文件在一个叫__pycache__ 的子目录中,它可以避免每次运行 Python 时去重新解析源代码。

 

 

1  python代码运行过程

 

Python执行的四步操作:

  1lexing: 词法分析,就是把一个句子分解成 token。大致来说,就是用str.split()可以实现的功能。            

  2parsing:解析,就是把这些 token 组装成一个逻辑结构。

  3compiling:编译,把这个逻辑结构转化成一个或者多个code object (代码对象)

  4interpreting:解释,执行每个code object 代表的代码。

  其中前三步可以归类为“代码编译”,最后一步单独成类。Python 程序的执行过程就是,它先把代码编译成 bytecode (字节码)指令,交给虚拟机,逐条执行 bytecode 指令。

 

分清function objectcode object ,以及 bytecode

  ①function object:定义一个函数之后,它就成了一个function object (函数对象)。只要不使用函数调用符号——也就是小括号——这个函数就不会执行。但是它已经被编译了,可以通过这个function object __code__ 属性找到它的 code object

  ②code objectcode object 的类型是‘code

  ③bytecodebytecode code object 的一个属性的值。这个属性名为 co_code,它的类型是‘bytes’,长度是8。例:b'|\x00\x00d\x01\x00\x14S'

 

实例1

1 >>> def double(a):
2     return a*2  # 并不知道为什么贴在这里缩进会是这样
3 
4 >>> import dis
5 >>> dis.dis(double)
6   2           0 LOAD_FAST                 0 (a)
7               2 LOAD_CONST               1 (2)
8               4 BINARY_MULTIPLY
9               6 RETURN_VALUE

12源代码中的行号2的数字 0 3 6 7 bytecode 的偏移量3很好理解,都是opcode

 

  因为可以节省编译时间,这里有一篇非常详细的文章,作者在遗传编程领域工作,发现他们Python 程序的总运算时间中,50%都被编译过程吃掉。于是作者深入到 bytecode 层次进行了小小改动,大幅削减了编译时间,把总的运算时间降至不足原先的一半。有改进的潜力

 

理论上的优化方向

1,从字节码bytecode上下手

2  python字节码优化猜想1-代码级优化

 

  优点:有一些固定的套路可以使用并且实施起来比较简单,例子:累加可以直接将很多分步直接在同一次处理中进行,节省步数

  缺点:优化后的效率,不能达到量级变化

 

2,从串行转并行入手(多线程、多进程、多核心) 

3  python字节码优化猜想2-处理方式优化

 

  优点:需要从python解释器底层进行重新布局

  缺点:优化的效率可以成倍数提升,并且效率与相关硬件有一定关系,参考nvidiapascal架构的并行计算卡

 

  前景:现在的手机芯片、电脑芯片大都是多核心、多进程的,这个可以一试。

 

  我自己之前去实习的公司中船重工709所凌久电子,设计过一台拥有256C66X核心的DSP处理机,这台机器这个就是实时、并发计算的,耗电快赶上空调,但是性能真的很强很强!!!(我在简历里面写过)

 

  (注:我只是最近两天看了一下相关的文档资料,我现在不确定python解释器是否已经在内部集成并行处理的功能)

参考:

Python实现多核心并行计算

浅谈多核CPU、多线程与并行计算

 

 

3,python解释器层级进行优化

4  python字节码优化猜想3-解释器层级

 

  之前的两个都是不触及python最底层的东西,这里是从最底层进行优化的思考。

 

  如上图4可知,我们现在常使用的CPython解释器是通过C语言进行二级运行的,这就和android虚拟机一样,一台机器上运行另一个环境,当我们想要改变什么的时候还需要通过中介来通知做出改变,这个就和两个人隔着墙通过手机来通话,但是这样不如我们面对面沟通的明了!!!

 

  优点:可以省去中间的很多步骤,直接对计算机硬件进行操作,效率提升至C语言那般畅快

  缺点:开发难度大,计算机越接近底层开发难度越大,这需要一个团队来进行。

 

参考:Python解释器

 

总体参考链接:

理解 Python 的执行方式,与字节码 bytecode 玩耍 (上)

理解 Python 的执行方式,与字节码 bytecode 玩耍 (下)

Fun with Python bytecode

二,opcode指令

  根据上文中的bytecode以及其附属的给人类理解查看opcodeopcode又称为操作码,是将python源代码进行编译之后的结果,python虚拟机无法直接执行human-readable的源代码,因此python编译器第一步先将源代码进行编译,以此得到opcode。例如在执行python程序时一般会先生成一个pyc文件,pyc文件就是编译后的结果,其中含有opcode序列。Opcodebytecode是有一定相关性的两种不同表述。(这里不做累赘表述)

  python的目标不是一个性能高效的语言,出于脚本动态类型的原因虚拟机做了大量计算来判断一个变量的当前类型,并且整个python虚拟机是基于栈逻辑的,频繁的压栈出栈操作也需要大量计算。

  缺点即为可能的改进方向!

 

参考链接:

深入理解pythonOpcode备忘录

理解Pythonopcode及优化

操作码定义opcode.h

PythonCodeObjectParser

Peephole optimization

三,指令预测

  这个研究比较少,但是感觉应该和NLP有一定的关系(可以进行一定的迁移应用)

 

参考链接:

PaddlePaddle】自然语言处理:句词预测

Wikipedia-Threaded code

高性能虚拟机解释器:DTC vs ITCIndirect-Threaded Code

Dynamically Disabling Way-prediction to Reduce Instruction Replay

The research of indirect transfer prediction technology based on information feedback

 

四,模块按需加载

Python import原理:

 

5 import运行大致原理

 

 

  使用import module_name语句就可以将这个文件作为模块导入。系统在导入模块时,要做以下三件事: 
    1.为源代码文件中定义的对象创建一个名字空间,通过这个名字空间可以访问到模块中定义的函数及变量。

    2.在新创建的名字空间里执行源代码文件

    3.创建一个名为源代码文件的对象,该对象引用模块的名字空间,这样就可以通过这个对象访问模块中的函数及变量

 

普通加载方式:

  文件抬头就对所有所需的库进行加载,这样的缺点是耗时(尤其是对快应用、启动速度有要求的程序很敏感)

 

以前的两种惰性/延迟加载方法:

  ①本地子功能区加载而非程序启动时的全局加载。直到你的程序运行需要这个库的时候才进行加载;缺点:易重复载入库文件、容易遗忘库载入的范围。

 

  ②惰性加载。需要模块的时候触发 ModuleNotFoundError  提前发现这个模块,而延迟的只是后续补加载过程;缺点:显式优于隐式、如果一个模块希望立即加载,那么在延迟加载时,它可能会严重崩溃。(Mercurial实际上开发了一个模块黑名单,以避免延迟加载来解决这个问题,但是他们必须确保对其进行更新,因此这也不是一个完美的解决方案。)

 

最新py3.7中的方法:

  在Python 3.7中,模块现在可以在其上定义__getattr__(),允许编写一个函数,在模块上的属性不可用时导入模块。这样做的缺点是使它成为一个惰性导入而不是一个加载,因此很晚才发现是否会引发ModuleNotFoundError。但是它是显式的,并且仍然是为您的模块全局定义的,因此更容易控制。

 

改进方向:发现导入错误被推迟,如何提前获知这个可能出现的导入错误防止程序抛出异常并终止。

 

  缺点:当你用的时候才开始加载,这个会锁住主线程进行库加载动作,如果是带有画面的操作,那么就会有视觉延迟(假设这个加载是第一次运行,且很耗时)

  改进:能不能在主线程旁边开一条线程提前进行预加载!!!虽然这个多线程还是使用的同一条时间线,但是这种方法将延迟的时间进行了平分

例如:

如果使用原始的加载需要消耗0.5s的时间,这0.5s=500ms>>16.6ms我们人眼是可以察觉的

但是,使用多线程加载,我们可以把这个延迟的时间分配在各个过程(假如这其中有20个 过程,有时人类自身的操作比较慢也可以在这个过程中见缝插针进行预加载)中,0.5/20=0.001s=1ms<<16.6ms这个时间人眼是感觉不错来的。

 1 import importlib
 2 
 3 # 这个是实现lazy_import的功能函数
 4 def lazy_import(importer_name, to_import):
 5     module = importlib.import_module(importer_name) # 直接加载调用的后一级函数
 6 
 7     import_mapping = {} # 字典 键名:有可能为缩写名  值名:为原始可查找库名,例如:import_mapping['np'] = 'numpy'
 8     for name in to_import:
 9         importing, _, binding = name.partition(' as ')
10         if not binding:
11             _, _, binding = importing.rpartition('.')
12         import_mapping[binding] = importing
13 
14     def __getattr__(name):
15         if name not in import_mapping:  # 如果这个库没在import_mapping中,就抛出异常错误,并且中断
16             message = f'module {importer_name!r} has no attribute {name!r}'
17             raise AttributeError(message)
18         importing = import_mapping[name]
19         imported = importlib.import_module(importing,module.__spec__.parent)
20         # print('name=',name,'module=',module,'module.__spec__=',module.__spec__,'module.__spec__.parent=',module.__spec__.parent)
21         setattr(module, name, imported) # sub, np, numpy
22         return imported
23 
24     return module, __getattr__  #返回一个库和一个方法

详情见网址:lazy_import源码解析(原创) 我自己的博客

 

现在的思路:

 

6 按需预加载

 

7 按需预加载运行逻辑

 

粗糙的实现代码1preload.py

优点:

  将preload在需要的函数之前运行(这个是多线程的加载方式,不会锁定主线程的相关计算,同时在计算机IO空闲的时候加载,见缝插针进行。提高程序运行效率),在后面需要相关函数时就直接调用这个功能即可

 

缺点:

  ①预加载的代码提前多少,这个我现在还没有办法说清,要具体看机器的计算时间;②不太人性化,需要人为或者程序转换原始.py程序。

总地来说也是一种尝试!后面还可以试试其他的多种方法解决。

 

 1 import threading
 2 from importlib import import_module
 3 # 可以返回值的功能函数
 4 class MyThread(threading.Thread):
 5     def __init__(self, func, args, name=''):
 6         threading.Thread.__init__(self)
 7         self.name = name
 8         self.func = func
 9         self.args = args
10         self.result = self.func(*self.args)
11 
12     # 返回一个函数
13     def get_result(self):
14         try:
15             return self.result
16         except Exception:
17             return None
18 
19 def module_before(module_name):
20     t = MyThread(import_module, [module_name], import_module.__name__)
21     t.start()
22     return t
23 
24 if __name__ == '__main__':
25     numpy_mid = module_before('numpy')
26     numpy = numpy_mid.get_result()
27     print(numpy.array([1, 2, 3, 4]))

 

代码段2:测试_preload.py

 1 import preload as pld
 2 
 3 # 这里面的预加载是放在需要这个函数的前面的相关代码块前,
 4 modules = ['numpy', 'sys', 'os']
 5 module = {}
 6 for i in modules:
 7     module[i] = pld.module_before(i)
 8 
 9 np = module['numpy'].get_result()
10 print(np.array([1, 2, 3, 4]))

参考链接:

__getattr__使用方法

setattr() 函数

An approach to lazy importing in Python 3.7

动态加载lazy_import(利用__import__)

python之import机制

动态导入对象,importlib.import_module()使用

关于Python的import机制原理

 

备注:这个里面的思路只是现在的一些方向,欢迎大家讨论指教。

posted on 2019-03-06 22:46  周健康  阅读(1111)  评论(0编辑  收藏  举报

导航