Example: having selected 10000 edges, convert the selection to faces.
PyMel: | 1.7 |
maya.cmds: | 7.5 |
MEL: | 7.4 |
OpenMaya: | 0.2 |
OpenMaya is quickest by far, but PyMel is still 4.5x faster then MEL or maya.cmds. What will slow it down however, is if we add a small change - flattening the selection list (i.e.: "ls -sl -fl"):
Timings with selection list flattening:
PyMel: | 7.9 |
maya.cmds: | 8.0 |
MEL: | 7.7 |
OpenMaya: | 0.2 (there is no distinction in the API, same code) |
Here it's roughly the same speed as MEL and maya.cmds. What seems to slow it down is creating instances of PyMel classes as we will see in the next example.
Example: custom edge ring select:
PyMel: | 2.4 |
maya.cmds: | 1.2 |
OpenMaya: | 0.03 |
PyMel stumbles here, maya.cmds is quicker but OpenMaya turns out nearly 40x faster still. However it's interesting to note that a lot of the PyMel code actually runs 2 or 3 times faster then the maya.cmds code. What slows it down is one part of it, which creates PyMel objects. To be clear, by creating PyMel objects, I mean something like: edge = pymel.core.general.MeshEdge( someMesh, someEdgeIndex ).
The overall conclusion I can draw is that PyMel is quick. A lot of it builds on top of OpenMaya, which is the quickest of all the scripting APIs. Maya.cmds is really just a thin wrapper on top of MEL, and there is no significant performance difference between the two. As long as you can avoid creating a lot of PyMel objects in your code, you should find it outperforming both maya.cmds and MEL.
We know OpenMaya is very quick, but lets compare it to it's non interpreted, compiled flavour: the C++ API. I have a simple plugin deformer implemented in both C++ and Python. It basically takes a point on a curve, calculates a normal to the curve at that point and then bends geometry along that curve. The timings are:
Python
Finding the point along the curve: | 0.027 |
Calculating a normal: | 0.15 |
Deforming passed points: | 0.060 |
C++
Finding the point along the curve: | 0.017 |
Calculating a normal: | 0.0023 |
Deforming passed points: | 0.000062 |
"Finding the point along the curve" is a single function call and most of the work gets done by Maya. There is only a small difference between Python and C++. "Deforming passed points" however, is a bunch of adding and multiplying. No API function gets called. Here C++ outperforms Python by nearly a thousand-fold. The more I work on this plugin the more dramatic the overall difference becomes. For deformers at least, C++ is massively quicker then Python.
These tests are only very limited, I admit, but the medal standings are: gold for C++, silver for OpenMaya, and bronze for PyMel. MEL and maya.cmds are tied for a close fourth.
I was wondering for the non API calls (such as 'deforming passed point' op) it should be possible to optimize python further by using fast math module such as numpy (using numpy array) or Cython (cross compile python code to C), or even PyOpenCL
ReplyDeleteThis comment has been removed by the author.
ReplyDeleteI was wondering exacltly what Shih proposed (Cython and Numpy for perfomance increases) when I got here.
ReplyDeleteHave any of you tried it already? :)