目次
1. 先頭の 1 つのアンダースコア: _var
3. 先頭の二重アンダースコア __var
これは名前マングリングとも呼ばれます。インタープリターは、クラスが拡張されたときに競合する可能性が低くなるように変数の名前を変更します。
4.双前导和双末尾下划线 _var_
5.单下划线 _
ホームページ バックエンド開発 Python チュートリアル Python におけるアンダースコアの意味と使い方は何ですか?

Python におけるアンダースコアの意味と使い方は何ですか?

Apr 20, 2023 pm 08:04 PM
python

1. 先頭の 1 つのアンダースコア: _var

変数名とメソッド名に関しては、1 つのアンダースコア接頭辞には従来の意味があります。これはプログラマーへの注意喚起です。つまり、Python コミュニティはその意味について合意していますが、プログラムの動作には影響しません。

アンダースコア接頭辞の意味は、単一のアンダースコアで始まる変数またはメソッドが内部使用のみであることを他のプログラマに知らせることです。この規則は PEP 8 で定義されています。

これは Python では必須ではありません。 Python には、Java のような「プライベート」変数と「パブリック」変数の明確な区別がありません。これは、誰かが「これは実際にはクラスのパブリック インターフェイスの一部であることを意図したものではありません。そのままにしておいてください。」という小さな下線の警告サインを設置するようなものです。

場合によっては、変数に最も適切な名前がすでにキーワードによって占有されていることがあります。したがって、class や def のような名前は、Python では変数名として使用できません。この場合、次のようにアンダースコアを追加することで名前の競合を解決できます。

>>> def make_object(name, class):
SyntaxError: "invalid syntax"

>>> def make_object(name, class_):
...    pass
ログイン後にコピー

つまり、末尾の単一のアンダースコア (サフィックス) は、Python キーワードとの名前の競合を避けるための規則です。 PEP 8 では、この規則について説明しています。

3. 先頭の二重アンダースコア __var

これまで説明したすべての命名パターンの意味は、合意された規則に基づいています。二重アンダースコアで始まる Python クラス属性 (変数やメソッドを含む) の場合、状況は少し異なります。

二重アンダースコア接頭辞により、Python インタープリターはサブクラスでの名前の競合を避けるために属性名を書き換えます。

これは名前マングリングとも呼ばれます。インタープリターは、クラスが拡張されたときに競合する可能性が低くなるように変数の名前を変更します。

#これは抽象的に聞こえるかもしれません。したがって、説明するために小さなコード例をまとめました:

class Test:
   def __init__(self):
       self.foo = 11
       self._bar = 23
       self.__baz = 23
ログイン後にコピー

組み込みの dir() 関数を使用して、このオブジェクトのプロパティを見てみましょう:

>>> t = Test()
>>> dir(t)
['_Test__baz', '__class__', '__delattr__', '__dict__', '__dir__',
'__doc__', '__eq__', '__format__', '__ge__', '__getattribute__',
'__gt__', '__hash__', '__init__', '__le__', '__lt__', '__module__',
'__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__',
'__setattr__', '__sizeof__', '__str__', '__subclasshook__',
'__weakref__', '_bar', 'foo']
ログイン後にコピー

上記は次のとおりです。オブジェクトのプロパティのリスト。このリストを見て、元の変数名

foo、_bar

__baz

を探してみましょう - いくつかの興味深い変更に気づくでしょう。

self.foo 変数は、プロパティ リストでは変更されずに foo として表示されます。 self._bar も同様に動作します。クラス上では

_bar として表示されます。前に述べたように、この場合、先頭のアンダースコアは単なる慣例です。プログラマー向けのヒントです。ただし、self.__baz の場合は、状況が少し異なります。このリストで __baz

を検索すると、この名前の変数は表示されません。

__バズに何が起こったのですか? よく見ると、このオブジェクトに _Test__baz

というプロパティがあることがわかります。これは、Python インタプリタによって行われる名前のマングリングです。これは、変数がサブクラスでオーバーライドされるのを防ぐために行われます。

Test クラスを拡張する別のクラスを作成し、コンストラクターに追加された既存のプロパティをオーバーライドしてみます。

class ExtendedTest(Test):
   def __init__(self):
       super().__init__()
       self.foo = 'overridden'
       self._bar = 'overridden'
       self.__baz = 'overridden'
ログイン後にコピー

さて、foo、

_bar という値が返されると思います。および __baz

は ExtendedTest クラスのこのインスタンスに表示されますか?見てみましょう:

>>> t2 = ExtendedTest()
>>> t2.foo
'overridden'
>>> t2._bar
'overridden'
>>> t2.__baz
AttributeError: "'ExtendedTest' object has no attribute '__baz'"
ログイン後にコピー

ちょっと待って、

t2.__ baz の値を表示しようとすると AttributeError が発生するのはなぜですか?名前変更が再びトリガーされます。このオブジェクトには __baz 属性さえないことがわかります:

>>> dir(t2)
['_ExtendedTest__baz', '_Test__baz', '__class__', '__delattr__',
'__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__',
'__getattribute__', '__gt__', '__hash__', '__init__', '__le__',
'__lt__', '__module__', '__ne__', '__new__', '__reduce__',
'__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__',
'__subclasshook__', '__weakref__', '_bar', 'foo', 'get_vars']
ログイン後にコピー

ご覧のとおり、偶発的な事故を防ぐために

__baz_ExtendedTest__baz になります。変更:

>>> t2._ExtendedTest__baz
'overridden'
ログイン後にコピー

ただし、元の

_Test__baz はまだ存在します:

>>> t2._Test__baz
42
ログイン後にコピー
二重アンダースコア名の変更は、プログラマにとって完全に透過的です。次の例はこれを確認します:
class ManglingTest:
   def __init__(self):
       self.__mangled = 'hello'

   def get_mangled(self):
       return self.__mangled

>>> ManglingTest().get_mangled()
'hello'
>>> ManglingTest().__mangled
AttributeError: "'ManglingTest' object has no attribute '__mangled'"
ログイン後にコピー

名前の装飾はメソッド名にも適用されますか?はい、それも当てはまります。名前の装飾は、クラスのコンテキストにおける 2 つのアンダースコア文字

("dunders") で始まるすべての名前に影響します。

class MangledMethod:
   def __method(self):
       return 42

   def call_it(self):
       return self.__method()

>>> MangledMethod().__method()
AttributeError: "'MangledMethod' object has no attribute '__method'"
>>> MangledMethod().call_it()
42
ログイン後にコピー

これは、おそらく驚くべき別の名前の使用法です。 変更された例:

_MangledGlobal__mangled = 23

class MangledGlobal:
   def test(self):
       return __mangled

>>> MangledGlobal().test()
23
ログイン後にコピー

この例では、

_MangledGlobal__mangled という名前のグローバル変数を宣言します。次に、MangledGlobal というクラスのコンテキストで変数にアクセスします。名前が変更されたため、クラスの test() メソッド内で _MangledGlobal__mangled

グローバル変数を

__mangled

として参照できます。

名前 __mangled は 2 つのアンダースコア文字で始まるため、Python インタープリターは名前 __mangled_MangledGlobal__mangled

に自動的に展開します。これは、名前の装飾がクラス属性に特に関連付けられていないことを示します。これは、クラス コンテキストで使用される 2 つのアンダースコア文字で始まる任意の名前に対して機能します。

吸収すべきことがたくさんあります。 正直に言うと、これらの例と説明は頭から出てきませんでした。これを思いつくまでには、ある程度の調査と処理が必要でした。私は常に Python を使用しており、長年にわたって使用していますが、このようなルールや特殊なケースが常に思い浮かぶわけではありません。

プログラマーにとって最も重要なスキルは、「パターン認識」と情報をどこで探せばよいかを知ることである場合があります。この時点で少し圧倒されていると感じても、心配しないでください。時間をかけて、この記事の例をいくつか試してください。

让这些概念完全沉浸下来,以便你能够理解名称修饰的总体思路,以及我向您展示的一些其他的行为。如果有一天你和它们不期而遇,你会知道在文档中按什么来查。

4.双前导和双末尾下划线 _var_

也许令人惊讶的是,如果一个名字同时以双下划线开始和结束,则不会应用名称修饰。 由双下划线前缀和后缀包围的变量不会被Python解释器修改:

class PrefixPostfixTest:
   def __init__(self):
       self.__bam__ = 42

>>> PrefixPostfixTest().__bam__
42
ログイン後にコピー

但是,Python保留了有双前导和双末尾下划线的名称,用于特殊用途。 这样的例子有,__init__对象构造函数,或__call__ — 它使得一个对象可以被调用。

这些dunder方法通常被称为神奇方法 - 但Python社区中的许多人(包括我自己)都不喜欢这种方法。

最好避免在自己的程序中使用以双下划线(“dunders”)开头和结尾的名称,以避免与将来Python语言的变化产生冲突。

5.单下划线 _

按照习惯,有时候单个独立下划线是用作一个名字,来表示某个变量是临时的或无关紧要的。

例如,在下面的循环中,我们不需要访问正在运行的索引,我们可以使用“_”来表示它只是一个临时值:

>>> for _ in range(32):
...    print('Hello, World.')
ログイン後にコピー

你也可以在拆分(unpacking)表达式中将单个下划线用作“不关心的”变量,以忽略特定的值。 同样,这个含义只是“依照约定”,并不会在Python解释器中触发特殊的行为。 单个下划线仅仅是一个有效的变量名称,会有这个用途而已。

在下面的代码示例中,我将汽车元组拆分为单独的变量,但我只对颜色和里程值感兴趣。 但是,为了使拆分表达式成功运行,我需要将包含在元组中的所有值分配给变量。 在这种情况下,“_”作为占位符变量可以派上用场:

>>> car = ('red', 'auto', 12, 3812.4)
>>> color, _, _, mileage = car

>>> color
'red'
>>> mileage
3812.4
>>> _
12
ログイン後にコピー

除了用作临时变量之外,“_”是大多数Python REPL中的一个特殊变量,它表示由解释器评估的最近一个表达式的结果。

这样就很方便了,比如你可以在一个解释器会话中访问先前计算的结果,或者,你是在动态构建多个对象并与它们交互,无需事先给这些对象分配名字:

>>> 20 + 3
23
>>> _
23
>>> print(_)
23

>>> list()
[]
>>> _.append(1)
>>> _.append(2)
>>> _.append(3)
>>> _
[1, 2, 3]
ログイン後にコピー

以上が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衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

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

SublimeText3 中国語版

SublimeText3 中国語版

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

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

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

ランプアーキテクチャの下でnode.jsまたはPythonサービスを効率的に統合する方法は? ランプアーキテクチャの下でnode.jsまたはPythonサービスを効率的に統合する方法は? Apr 01, 2025 pm 02:48 PM

多くのウェブサイト開発者は、ランプアーキテクチャの下でnode.jsまたはPythonサービスを統合する問題に直面しています:既存のランプ(Linux Apache MySQL PHP)アーキテクチャWebサイトのニーズ...

Scapy Crawlerを使用するときにパイプラインの永続的なストレージファイルを書き込めない理由は何ですか? Scapy Crawlerを使用するときにパイプラインの永続的なストレージファイルを書き込めない理由は何ですか? Apr 01, 2025 pm 04:03 PM

Scapy Crawlerを使用する場合、パイプラインの永続的なストレージファイルを書くことができない理由は?ディスカッションデータクローラーにScapy Crawlerを使用することを学ぶとき、あなたはしばしば...

PythonプロセスプールがTCPリクエストを同時に処理し、クライアントが立ち往生する理由は何ですか? PythonプロセスプールがTCPリクエストを同時に処理し、クライアントが立ち往生する理由は何ですか? Apr 01, 2025 pm 04:09 PM

Python Process Poolは、クライアントが立ち往生する原因となる同時TCP要求を処理します。ネットワークプログラミングにPythonを使用する場合、同時のTCP要求を効率的に処理することが重要です。 ...

Python functools.partialオブジェクトによって内部的にカプセル化された元の関数を表示する方法は? Python functools.partialオブジェクトによって内部的にカプセル化された元の関数を表示する方法は? Apr 01, 2025 pm 04:15 PM

python functools.partialオブジェクトのpython functools.partialを使用してPythonを使用する視聴方法を深く探索します。

LinuxターミナルでPythonバージョンを表示するときに発生する権限の問題を解決する方法は? LinuxターミナルでPythonバージョンを表示するときに発生する権限の問題を解決する方法は? Apr 01, 2025 pm 05:09 PM

LinuxターミナルでPythonバージョンを表示する際の許可の問題の解決策PythonターミナルでPythonバージョンを表示しようとするとき、Pythonを入力してください...

Pythonクロスプラットフォームデスクトップアプリケーション開発:どのGUIライブラリが最適ですか? Pythonクロスプラットフォームデスクトップアプリケーション開発:どのGUIライブラリが最適ですか? Apr 01, 2025 pm 05:24 PM

Pythonクロスプラットフォームデスクトップアプリケーション開発ライブラリの選択多くのPython開発者は、WindowsシステムとLinuxシステムの両方で実行できるデスクトップアプリケーションを開発したいと考えています...

Python hourglassグラフ図面:可変未定義エラーを避ける方法は? Python hourglassグラフ図面:可変未定義エラーを避ける方法は? Apr 01, 2025 pm 06:27 PM

Python:Hourglassグラフィック図面と入力検証この記事では、Python NoviceがHourglass Graphic Drawingプログラムで遭遇する可変定義の問題を解決します。コード...

Pythonで大規模な製品データセットを効率的にカウントしてソートするにはどうすればよいですか? Pythonで大規模な製品データセットを効率的にカウントしてソートするにはどうすればよいですか? Apr 01, 2025 pm 08:03 PM

データの変換と統計:大規模なデータセットの効率的な処理この記事では、製品情報を含むデータリストを別の含有しているものに変換する方法を詳細に紹介します...

See all articles