


Codeobjekt-Datenstruktur
typedef struct { PyObject_HEAD int co_argcount; /* #arguments, except *args */ int co_kwonlyargcount; /* #keyword only arguments */ int co_nlocals; /* #local variables */ int co_stacksize; /* #entries needed for evaluation stack */ int co_flags; /* CO_..., see below */ PyObject *co_code; /* instruction opcodes */ PyObject *co_consts; /* list (constants used) */ PyObject *co_names; /* list of strings (names used) */ PyObject *co_varnames; /* tuple of strings (local variable names) */ PyObject *co_freevars; /* tuple of strings (free variable names) */ PyObject *co_cellvars; /* tuple of strings (cell variable names) */ /* The rest aren't used in either hash or comparisons, except for co_name (used in both) and co_firstlineno (used only in comparisons). This is done to preserve the name and line number for tracebacks and debuggers; otherwise, constant de-duplication would collapse identical functions/lambdas defined on different lines. */ unsigned char *co_cell2arg; /* Maps cell vars which are arguments. */ PyObject *co_filename; /* unicode (where it was loaded from) */ PyObject *co_name; /* unicode (name, for reference) */ int co_firstlineno; /* first source line number */ PyObject *co_lnotab; /* string (encoding addr<->lineno mapping) See Objects/lnotab_notes.txt for details. */ void *co_zombieframe; /* for optimization only (see frameobject.c) */ PyObject *co_weakreflist; /* to support weakrefs to code objects */ } PyCodeObject;
Das Folgende ist die Rolle jedes Felds im Codeobjekt:
Zunächst müssen Sie das Konzept des Codeblocks verstehen , das als kleine Einheit als Ganzes ausgeführt wird. Zu den gängigen Codeblöcken in Python gehören: Funktionskörper, Klassendefinition und ein Modul.
argcount, dies stellt die Anzahl der Parameter in einem Codeblock dar. Dieser Parameter ist nur für Funktionskörpercodeblöcke nützlich, da die obige Funktion pycdemo.py beispielsweise ein Modul und keine Funktion sein kann , also dieser Parameter Der entsprechende Wert ist 0.
co_code, der spezifische Inhalt dieses Objekts ist eine Bytesequenz, die echten Python-Bytecode speichert. Er wird hauptsächlich für die Ausführung virtueller Python-Maschinen verwendet und wird in diesem Artikel nicht im Detail analysiert.
co_consts, dieses Feld ist ein Listenfeld, das hauptsächlich einige Zeichenfolgenkonstanten und numerische Konstanten enthält, z. B. „__main__“ und 100 oben.
co_filename, die Bedeutung dieses Feldes ist der Dateiname der entsprechenden Quelldatei.
co_firstlineno, die Bedeutung dieses Feldes ist die Anzahl der Zeilen, die in der ersten Codezeile in der Python-Quelldatei erscheinen. Dieses Feld ist beim Debuggen sehr wichtig.
co_flags, die Hauptbedeutung dieses Feldes besteht darin, den Typ dieses Codeobjekts zu identifizieren. 0x0080 zeigt an, dass es sich bei diesem Block um eine Coroutine handelt, 0x0010 zeigt an, dass dieses Codeobjekt verschachtelt ist und so weiter.
co_lnotab, die Bedeutung dieses Feldes wird hauptsächlich zur Berechnung der Anzahl der Quellcodezeilen verwendet, die jeder Bytecode-Anweisung entsprechen.
co_varnames, die Hauptbedeutung dieses Feldes besteht darin, einen Namen darzustellen, der lokal in einem Codeobjekt definiert ist.
co_names, das Gegenteil von co_varnames, stellt Namen dar, die nicht lokal definiert, sondern im Codeobjekt verwendet werden.
co_nlocals, dieses Feld gibt die Anzahl der lokal in einem Codeobjekt verwendeten Variablen an.
co_stackszie, da die virtuelle Python-Maschine ein Stapelcomputer ist, gibt der Wert dieses Parameters den für diesen Stapel erforderlichen Maximalwert an.
co_cellvars, co_freevars, diese beiden Felder beziehen sich hauptsächlich auf verschachtelte Funktionen und Funktionsabschlüsse. Wir werden dieses Feld in den folgenden Artikeln ausführlich erläutern.
Detaillierte Analyse von Codeobjekten
Jetzt verwenden wir einige praktische Beispiele, um bestimmte Codeobjekte zu analysieren.
import dis import binascii import types d = 10 def test_co01(c): a = 1 b = 2 return a + b + c + d
Im vorherigen Artikel haben wir erwähnt, dass eine Funktion ein Codeobjektobjekt enthält. Das Ausgabeergebnis des Codeobjektobjekts von test_co01 (siehe co01 für den vollständigen Code) lautet wie folgt:
code argcount 1 nlocals 3 stacksize 2 flags 0043 0x43 code b'6401007d01006402007d02007c01007c0200177c0000177400001753' 9 0 LOAD_CONST 1 (1) 3 STORE_FAST 1 (a) 10 6 LOAD_CONST 2 (2) 9 STORE_FAST 2 (b) 11 12 LOAD_FAST 1 (a) 15 LOAD_FAST 2 (b) 18 BINARY_ADD 19 LOAD_FAST 0 (c) 22 BINARY_ADD 23 LOAD_GLOBAL 0 (d) 26 BINARY_ADD 27 RETURN_VALUE consts None 1 2 names ('d',) varnames ('c', 'a', 'b') freevars () cellvars () filename '/tmp/pycharm_project_396/co01.py' name 'test_co01' firstlineno 8 lnotab b'000106010601'
Der Wert des Feldes argcount ist gleich 1, was darauf hinweist, dass die Funktion einen Parameter hat und dass diese Funktion test_co01 einen Parameter c hat, der einander entspricht.
Der Wert des Felds nlocals ist gleich 3, was darauf hinweist, dass in der Funktion test_co01 insgesamt drei lokale Funktionsvariablen a, b, c implementiert sind.
Die Feldnamen entsprechen co_names im Code. Gemäß der vorherigen Definition wird die globale Variable d in der Funktion test_co01 verwendet, sie ist jedoch nicht in der Funktion definiert.
Feldvariablennamen, dies stellt die in lokalen Definitionen verwendeten Variablen dar. In der Funktion test_co01 gibt es drei Hauptvariablen a, b, c.
Das Feld Dateiname ist die Adresse der Python-Datei.
Field firstlineno gibt an, dass die erste Zeile der Funktion in Zeile 8 des entsprechenden Python-Codes erscheint.
Detaillierte Analyse des Flag-Felds
Wir verwenden speziell den Quellcode von Python3.5 für die Analyse. Die spezifische Implementierung der virtuellen cpython-Maschine lautet wie folgt (Include/code.h):
/* Masks for co_flags above */ #define CO_OPTIMIZED 0x0001 #define CO_NEWLOCALS 0x0002 #define CO_VARARGS 0x0004 #define CO_VARKEYWORDS 0x0008 #define CO_NESTED 0x0010 #define CO_GENERATOR 0x0020 /* The CO_NOFREE flag is set if there are no free or cell variables. This information is redundant, but it allows a single flag test to determine whether there is any extra work to be done when the call frame it setup. */ #define CO_NOFREE 0x0040 /* The CO_COROUTINE flag is set for coroutine functions (defined with ``async def`` keywords) */ #define CO_COROUTINE 0x0080 #define CO_ITERABLE_COROUTINE 0x0100
Wenn das Flag-Feld und Jede der oben genannten Makrodefinitionen führt die &-Operation aus. Wenn das Ergebnis größer als 0 ist, bedeutet dies, dass die entsprechenden Bedingungen erfüllt sind.
Die obige Makrodefinition hat folgende Bedeutung:
CO_OPTIMIZED Dieses Feld zeigt an, dass das Codeobjekt optimiert ist und lokal von der Funktion definierte Variablen verwendet.
CO_NEWLOCALS, die Bedeutung dieses Feldes besteht darin, dass beim Ausführen des Codes dieses Codeobjekts ein Diktatobjekt für das f_locals-Objekt im Stapelrahmen erstellt wird.
CO_VARARGS gibt an, ob dieses Codeobjekt Positionsparameter enthält.
CO_VARKEYWORDS gibt an, ob dieses Codeobjekt Schlüsselwortparameter enthält.
CO_NESTED, was darauf hinweist, dass dieses Codeobjekt eine verschachtelte Funktion ist.
CO_GENERATOR, was anzeigt, dass dieses Codeobjekt ein Generator ist.
CO_COROUTINE, was darauf hinweist, dass dieses Codeobjekt eine Coroutine-Funktion ist.
CO_ITERABLE_COROUTINE, was anzeigt, dass das Codeobjekt eine iterierbare Coroutine-Funktion ist.
CO_NOFREE, das bedeutet, dass es keine Freevars und Cellvars gibt, das heißt, es gibt keinen Funktionsabschluss.
Jetzt analysieren wir die Flags der vorherigen Funktion test_co01. Der entsprechende Wert ist gleich 0x43, was bedeutet, dass diese Funktion drei Merkmale erfüllt: CO_NEWLOCALS, CO_OPTIMIZED und CO_NOFREE.
freevars & cellvars
我们使用下面的函数来对这两个字段进行分析:
def test_co02(): a = 1 b = 2 def g(): return a + b return a + b + g()
上面的函数的信息如下所示(完整代码见co02):
code argcount 0 nlocals 1 stacksize 3 flags 0003 0x3 code b'640100890000640200890100870000870100660200640300640400860000' b'7d0000880000880100177c00008300001753' 15 0 LOAD_CONST 1 (1) 3 STORE_DEREF 0 (a) 16 6 LOAD_CONST 2 (2) 9 STORE_DEREF 1 (b) 18 12 LOAD_CLOSURE 0 (a) 15 LOAD_CLOSURE 1 (b) 18 BUILD_TUPLE 2 21 LOAD_CONST 3 (<code object g at 0x7f133ff496f0, file "/tmp/pycharm_project_396/co01.py", line 18>) 24 LOAD_CONST 4 ('test_co02.<locals>.g') 27 MAKE_CLOSURE 0 30 STORE_FAST 0 (g) 20 33 LOAD_DEREF 0 (a) 36 LOAD_DEREF 1 (b) 39 BINARY_ADD 40 LOAD_FAST 0 (g) 43 CALL_FUNCTION 0 (0 positional, 0 keyword pair) 46 BINARY_ADD 47 RETURN_VALUE consts None 1 2 code argcount 0 nlocals 0 stacksize 2 flags 0013 0x13 code b'8800008801001753' 19 0 LOAD_DEREF 0 (a) 3 LOAD_DEREF 1 (b) 6 BINARY_ADD 7 RETURN_VALUE consts None names () varnames () freevars ('a', 'b') cellvars () filename '/tmp/pycharm_project_396/co01.py' name 'g' firstlineno 18 lnotab b'0001' 'test_co02.<locals>.g' names () varnames ('g',) freevars () cellvars ('a', 'b') filename '/tmp/pycharm_project_396/co01.py' name 'test_co02' firstlineno 14 lnotab b'0001060106021502'
从上面的输出我们可以看到的是,函数 test_co02 的 cellvars 为 ('a', 'b'),函数 g 的 freevars 为 ('a', 'b'),cellvars 表示在其他函数当中会使用本地定义的变量,freevars 表示本地会使用其他函数定义的变量。
再来分析一下函数 test_co02 的 flags,他的 flags 等于 0x3 因为有闭包的存在因此 flags 不会存在 CO_NOFREE,也就是少了值 0x0040 。
stacksize
这个字段存储的是在函数在被虚拟机执行的时候所需要的最大的栈空间的大小,这也是一种优化手段,因为在知道所需要的最大的栈空间,所以可以在函数执行的时候直接分配指定大小的空间不需要在函数执行的时候再去重新扩容。
def test_stack(): a = 1 b = 2 return a + b
上面的代码相关字节码等信息如下所示:
code argcount 0 nlocals 2 stacksize 2 flags 0043 0x43 code b'6401007d00006402007d01007c00007c01001753' # 字节码指令 # 字节码指令参数 # 参数对应的值 24 0 LOAD_CONST 1 (1) 3 STORE_FAST 0 (a) 25 6 LOAD_CONST 2 (2) 9 STORE_FAST 1 (b) 26 12 LOAD_FAST 0 (a) 15 LOAD_FAST 1 (b) 18 BINARY_ADD 19 RETURN_VALUE consts None # 下标等于 0 的常量 1 # 下标等于 1 的常量 2 # 下标等于 2 的常量 names () varnames ('a', 'b') freevars () cellvars ()
我们现在来模拟一下执行过程,在模拟之前我们首先来了解一下上面几条字节码的作用:
LOAD_CONST,将常量表当中的下标等于 i 个对象加载到栈当中,对应上面的代码 LOAD_CONST 的参数 i = 1。因此加载测常量等于 1 。因此现在栈空间如下所示:
STORE_FAST,将栈顶元素弹出并且保存到 co_varnames 对应的下标当中,根据上面的字节码参数等于 0 ,因此将 1 保存到 co_varnames[0] 对应的对象当中。
LOAD_CONST,将下标等于 2 的常量加载进入栈中。
STORE_FAST,将栈顶元素弹出,并且保存到 varnames 下标为 1 的对象。
LOAD_FAST,是取出 co_varnames 对应下标的数据,并且将其压入栈中。我们直接连续执行两个 LOAD_FAST 之后栈空间的布局如下:
BINARY_ADD,这个字节码指令是将栈空间的两个栈顶元素弹出,然后将两个数据进行相加操作,然后将相加得到的结果重新压入栈中。
RETURN_VALUE,将栈顶元素弹出并且作为返回值返回。
从上面的整个执行过程来看整个栈空间使用的最大的空间长度为 2 ,因此 stacksize = 2 。
Das obige ist der detaillierte Inhalt vonWelche Rolle spielt das Codeobjekt in der virtuellen Python-Maschine?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Die Flexibilität von Python spiegelt sich in Multi-Paradigm-Unterstützung und dynamischen Typsystemen wider, während eine einfache Syntax und eine reichhaltige Standardbibliothek stammt. 1. Flexibilität: Unterstützt objektorientierte, funktionale und prozedurale Programmierung und dynamische Typsysteme verbessern die Entwicklungseffizienz. 2. Benutzerfreundlichkeit: Die Grammatik liegt nahe an der natürlichen Sprache, die Standardbibliothek deckt eine breite Palette von Funktionen ab und vereinfacht den Entwicklungsprozess.

Python ist für seine Einfachheit und Kraft sehr beliebt, geeignet für alle Anforderungen von Anfängern bis hin zu fortgeschrittenen Entwicklern. Seine Vielseitigkeit spiegelt sich in: 1) leicht zu erlernen und benutzten, einfachen Syntax; 2) Reiche Bibliotheken und Frameworks wie Numpy, Pandas usw.; 3) plattformübergreifende Unterstützung, die auf einer Vielzahl von Betriebssystemen betrieben werden kann; 4) Geeignet für Skript- und Automatisierungsaufgaben zur Verbesserung der Arbeitseffizienz.

Ja, lernen Sie Python in zwei Stunden am Tag. 1. Entwickeln Sie einen angemessenen Studienplan, 2. Wählen Sie die richtigen Lernressourcen aus, 3. Konsolidieren Sie das durch die Praxis erlernte Wissen. Diese Schritte können Ihnen helfen, Python in kurzer Zeit zu meistern.

Python eignet sich für eine schnelle Entwicklung und Datenverarbeitung, während C für hohe Leistung und zugrunde liegende Kontrolle geeignet ist. 1) Python ist einfach zu bedienen, mit prägnanter Syntax, und eignet sich für Datenwissenschaft und Webentwicklung. 2) C hat eine hohe Leistung und eine genaue Kontrolle und wird häufig bei der Programmierung von Spielen und Systemen verwendet.

Die Zeit, die zum Erlernen von Python erforderlich ist, variiert von Person zu Person, hauptsächlich von früheren Programmiererfahrungen, Lernmotivation, Lernressourcen und -methoden und Lernrhythmus. Setzen Sie realistische Lernziele und lernen Sie durch praktische Projekte am besten.

Python zeichnet sich in Automatisierung, Skript und Aufgabenverwaltung aus. 1) Automatisierung: Die Sicherungssicherung wird durch Standardbibliotheken wie OS und Shutil realisiert. 2) Skriptschreiben: Verwenden Sie die PSUTIL -Bibliothek, um die Systemressourcen zu überwachen. 3) Aufgabenverwaltung: Verwenden Sie die Zeitplanbibliothek, um Aufgaben zu planen. Die Benutzerfreundlichkeit von Python und die Unterstützung der reichhaltigen Bibliothek machen es zum bevorzugten Werkzeug in diesen Bereichen.

Um die Effizienz des Lernens von Python in einer begrenzten Zeit zu maximieren, können Sie Pythons DateTime-, Zeit- und Zeitplanmodule verwenden. 1. Das DateTime -Modul wird verwendet, um die Lernzeit aufzuzeichnen und zu planen. 2. Das Zeitmodul hilft, die Studie zu setzen und Zeit zu ruhen. 3. Das Zeitplanmodul arrangiert automatisch wöchentliche Lernaufgaben.

Python zeichnet sich in Gaming und GUI -Entwicklung aus. 1) Spielentwicklung verwendet Pygame, die Zeichnungen, Audio- und andere Funktionen bereitstellt, die für die Erstellung von 2D -Spielen geeignet sind. 2) Die GUI -Entwicklung kann Tkinter oder Pyqt auswählen. Tkinter ist einfach und einfach zu bedienen. PYQT hat reichhaltige Funktionen und ist für die berufliche Entwicklung geeignet.


Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

EditPlus chinesische Crack-Version
Geringe Größe, Syntaxhervorhebung, unterstützt keine Code-Eingabeaufforderungsfunktion

WebStorm-Mac-Version
Nützliche JavaScript-Entwicklungstools

Sicherer Prüfungsbrowser
Safe Exam Browser ist eine sichere Browserumgebung für die sichere Teilnahme an Online-Prüfungen. Diese Software verwandelt jeden Computer in einen sicheren Arbeitsplatz. Es kontrolliert den Zugriff auf alle Dienstprogramme und verhindert, dass Schüler nicht autorisierte Ressourcen nutzen.

SublimeText3 Englische Version
Empfohlen: Win-Version, unterstützt Code-Eingabeaufforderungen!

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung