ホームページ バックエンド開発 Python チュートリアル Python の魔法のメタプログラミングをマスターする: 自らを記述するコード

Python の魔法のメタプログラミングをマスターする: 自らを記述するコード

Dec 08, 2024 am 10:41 AM

Mastering Python

Python のメタプログラミング機能は本当に魅力的です。彼らは言語を私たちの意志に合わせて曲げて、コードを書くコードを作成することを可能にします。それは、Python をプログラマーそのものとして教えるようなものです!

コード生成から始めましょう。ここで、Python コードを文字列として作成し、実行します。シンプルに聞こえるかもしれませんが、信じられないほど強力です。基本的な例を次に示します:

code = f"def greet(name):\n    print(f'Hello, {{name}}!')"
exec(code)
greet("Alice")
ログイン後にコピー
ログイン後にコピー

これはオンザフライで関数を作成し、それを呼び出します。しかし、さらに先へ進むこともできます。実行時の条件に基づいて、クラス全体、モジュール、さらには複雑なアルゴリズムを生成できます。

優れたトリックの 1 つは、構成にコード生成を使用することです。構成ファイルをロードする代わりに、設定を定義する Python コードを生成できます。これは、従来の構成解析よりも高速かつ柔軟です。

それでは、抽象構文ツリー (AST) に移りましょう。ここからが本当に興味深いことになります。 AST は Python コードのツリー表現です。 Python ソースを AST に解析し、変更して、コンパイルして実行可能コードに戻すことができます。

関数を変更してログを追加する簡単な例を次に示します。

import ast

def add_logging(node):
    if isinstance(node, ast.FunctionDef):
        log_stmt = ast.Expr(ast.Call(
            func=ast.Attribute(
                value=ast.Name(id='print', ctx=ast.Load()),
                attr='__call__',
                ctx=ast.Load()
            ),
            args=[ast.Str(s=f"Calling {node.name}")],
            keywords=[]
        ))
        node.body.insert(0, log_stmt)
    return node

tree = ast.parse("def hello(): print('Hello, world!')")
modified_tree = ast.fix_missing_locations(ast.NodeTransformer().visit(tree))
exec(compile(modified_tree, '<string>', 'exec'))
hello()
ログイン後にコピー
ログイン後にコピー

これにより、すべての関数の先頭に print ステートメントが追加されます。これは単純な例ですが、AST 操作の威力を示しています。これは、コードの最適化、インストルメンテーションの追加、さらには新しい言語機能の実装など、あらゆる種類の変換に使用できます。

AST 操作の特に優れた使い方の 1 つは、ドメイン固有言語 (DSL) の作成です。カスタム構文を AST に解析し、通常の Python に変換して実行できます。これにより、Python の能力を最大限に活用しながら、特定の問題に合わせた言語を作成できます。

たとえば、単純な数学 DSL を作成できます。

import ast

class MathTransformer(ast.NodeTransformer):
    def visit_BinOp(self, node):
        if isinstance(node.op, ast.Add):
            return ast.Call(
                func=ast.Name(id='add', ctx=ast.Load()),
                args=[self.visit(node.left), self.visit(node.right)],
                keywords=[]
            )
        return node

def parse_math(expr):
    tree = ast.parse(expr)
    transformer = MathTransformer()
    modified_tree = transformer.visit(tree)
    return ast.fix_missing_locations(modified_tree)

def add(a, b):
    print(f"Adding {a} and {b}")
    return a + b

exec(compile(parse_math("result = 2 + 3 + 4"), '<string>', 'exec'))
print(result)
ログイン後にコピー
ログイン後にコピー

これにより、加算演算が関数呼び出しに変換され、基本的な数学演算にカスタム動作 (ログ記録など) を追加できるようになります。

もう 1 つの強力なテクニックはバイトコード操作です。 Python は、ソース コードを実行する前にバイトコードにコンパイルします。このバイトコードを操作することで、ソース コード レベルでは困難または不可能な最適化や変更を実現できます。

次に、関数が呼び出された回数をカウントするように変更する簡単な例を示します。

import types

def count_calls(func):
    code = func.__code__
    constants = list(code.co_consts)
    constants.append(0)  # Add a new constant for our counter
    counter_index = len(constants) - 1

    # Create new bytecode
    new_code = bytes([
        101, counter_index,  # LOAD_CONST counter
        100, 1,              # LOAD_CONST 1
        23,                  # BINARY_ADD
        125, counter_index,  # STORE_FAST counter
    ]) + code.co_code

    # Create a new code object with our modified bytecode
    new_code_obj = types.CodeType(
        code.co_argcount, code.co_kwonlyargcount, code.co_nlocals,
        code.co_stacksize + 1, code.co_flags, new_code, tuple(constants),
        code.co_names, code.co_varnames, code.co_filename, code.co_name,
        code.co_firstlineno, code.co_lnotab
    )

    return types.FunctionType(new_code_obj, func.__globals__, func.__name__, func.__defaults__, func.__closure__)

@count_calls
def hello():
    print("Hello, world!")

hello()
hello()
print(hello.__code__.co_consts[-1])  # Print the call count
ログイン後にコピー

これにより、関数のバイトコードが変更され、呼び出されるたびにカウンターがインクリメントされます。これは少し低レベルですが、非常に強力な最適化と変更が可能になります。

メタプログラミングが真価を発揮する領域の 1 つは、適応アルゴリズムの作成です。自身のパフォーマンスを分析し、より効率的になるように自身を書き換えるコードを書くことができます。たとえば、さまざまなアルゴリズムを試し、現在のデータに対して最も速いアルゴリズムを選択する並べ替え関数を作成できます。

code = f"def greet(name):\n    print(f'Hello, {{name}}!')"
exec(code)
greet("Alice")
ログイン後にコピー
ログイン後にコピー

このソーターは、表示されているデータに対して最速のアルゴリズムを使用するように自動的に適応します。

メタプログラミングは、テストやデバッグにも非常に役立ちます。これを使用して、テスト ケースを自動的に生成したり、オブジェクトをモックしたり、コードにインストルメンテーションを追加したりできます。

関数のテスト ケースを自動的に生成する簡単な例を次に示します。

import ast

def add_logging(node):
    if isinstance(node, ast.FunctionDef):
        log_stmt = ast.Expr(ast.Call(
            func=ast.Attribute(
                value=ast.Name(id='print', ctx=ast.Load()),
                attr='__call__',
                ctx=ast.Load()
            ),
            args=[ast.Str(s=f"Calling {node.name}")],
            keywords=[]
        ))
        node.body.insert(0, log_stmt)
    return node

tree = ast.parse("def hello(): print('Hello, world!')")
modified_tree = ast.fix_missing_locations(ast.NodeTransformer().visit(tree))
exec(compile(modified_tree, '<string>', 'exec'))
hello()
ログイン後にコピー
ログイン後にコピー

これにより、追加関数のランダムなテスト ケースが生成されます。これを拡張して関数の AST を分析し、より対象を絞ったテスト ケースを生成できます。

メタプログラミングの最も強力な側面の 1 つは、定型コードを削減できることです。コードを記述するコードを記述して、反復的なタスクを自動化し、コードベースを DRY (Don'trepeat Yourself) に保つことができます。

たとえば、データ クラスの作成を自動化できます。

import ast

class MathTransformer(ast.NodeTransformer):
    def visit_BinOp(self, node):
        if isinstance(node.op, ast.Add):
            return ast.Call(
                func=ast.Name(id='add', ctx=ast.Load()),
                args=[self.visit(node.left), self.visit(node.right)],
                keywords=[]
            )
        return node

def parse_math(expr):
    tree = ast.parse(expr)
    transformer = MathTransformer()
    modified_tree = transformer.visit(tree)
    return ast.fix_missing_locations(modified_tree)

def add(a, b):
    print(f"Adding {a} and {b}")
    return a + b

exec(compile(parse_math("result = 2 + 3 + 4"), '<string>', 'exec'))
print(result)
ログイン後にコピー
ログイン後にコピー

これにより、指定されたフィールドと型ヒントを使用して新しいクラスが作成されます。これを拡張して、メソッド、プロパティ、または他のクラス機能を追加できます。

メタプログラミングは、単にコードを記述するコードを記述することではありません。それは、より柔軟で、適応性があり、強力なソフトウェアを作成することです。これにより、さまざまなユースケースに適応できるフレームワークを作成したり、特定のシナリオに最適化されたコードを生成したり、複雑なタスクを簡素化するドメイン固有の言語を作成したりすることができます。

しかし、大きな力には大きな責任が伴います。メタプログラミングは、慎重に使用しないと、コードの理解やデバッグが困難になる可能性があります。メタプログラミング コードを徹底的に文書化し、慎重に使用することが重要です。

結論として、Python でのメタプログラミングは可能性の世界を開きます。パフォーマンスの最適化、ボイラープレートの削減、DSL の作成、適応アルゴリズムの構築のいずれの場合でも、コード生成や AST 操作などのメタプログラミング手法は、Python ツールキットの強力なツールです。これにより、通常を超えたコードを記述して、それ自体を分析、変更、改善できるソフトウェアを作成できるようになります。これらのテクニックを探求すると、Python コードをこれまでよりも柔軟、効率的、強力にする新しい方法が見つかるでしょう。


私たちの作品

私たちの作品をぜひチェックしてください:

インベスターセントラル | スマートな暮らし | エポックとエコー | 不可解な謎 | ヒンドゥーヴァ | エリート開発者 | JS スクール


私たちは中程度です

Tech Koala Insights | エポックズ&エコーズワールド | インベスター・セントラル・メディア | 不可解な謎 中 | 科学とエポックミディアム | 現代ヒンドゥーヴァ

以上がPython の魔法のメタプログラミングをマスターする: 自らを記述するコードの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

中間の読書にどこでもfiddlerを使用するときにブラウザによって検出されないようにするにはどうすればよいですか? 中間の読書にどこでもfiddlerを使用するときにブラウザによって検出されないようにするにはどうすればよいですか? Apr 02, 2025 am 07:15 AM

fiddlereveryversings for the-middleの測定値を使用するときに検出されないようにする方法

プロジェクトの基本と問題駆動型の方法で10時間以内にコンピューター初心者プログラミングの基本を教える方法は? プロジェクトの基本と問題駆動型の方法で10時間以内にコンピューター初心者プログラミングの基本を教える方法は? Apr 02, 2025 am 07:18 AM

10時間以内にコンピューター初心者プログラミングの基本を教える方法は?コンピューター初心者にプログラミングの知識を教えるのに10時間しかない場合、何を教えることを選びますか...

Investing.comの反クローラーメカニズムをバイパスするニュースデータを取得する方法は? Investing.comの反クローラーメカニズムをバイパスするニュースデータを取得する方法は? Apr 02, 2025 am 07:03 AM

Investing.comの反クラウリング戦略を理解する多くの人々は、Investing.com(https://cn.investing.com/news/latest-news)からのニュースデータをクロールしようとします。

Python 3.6のロードピクルスファイルエラーmodulenotfounderror:ピクルスファイル「__builtin__」をロードした場合はどうすればよいですか? Python 3.6のロードピクルスファイルエラーmodulenotfounderror:ピクルスファイル「__builtin__」をロードした場合はどうすればよいですか? Apr 02, 2025 am 06:27 AM

Python 3.6のピクルスファイルの読み込みエラー:modulenotfounderror:nomodulenamed ...

Scapy Crawlerを使用するときにパイプラインファイルを書き込めない理由は何ですか? Scapy Crawlerを使用するときにパイプラインファイルを書き込めない理由は何ですか? Apr 02, 2025 am 06:45 AM

SCAPYクローラーを使用するときにパイプラインファイルを作成できない理由についての議論は、SCAPYクローラーを学習して永続的なデータストレージに使用するときに、パイプラインファイルに遭遇する可能性があります...

See all articles