关于Python的前后、单双下划线作用,看完这篇文章,吊打面试官!

时间:2022-07-25
本文章向大家介绍关于Python的前后、单双下划线作用,看完这篇文章,吊打面试官!,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

重磅干货,第一时间送达

来源:清风python

python的各种下划线

在Python中,可能最常见的就是各种常量、变量、函数、方法前后添加的那些下划线了。有前面加的、后面加的,加一个的,加两个的,看到头晕。那么,你对这些知识都掌握了吗 ?让我们先来做一个自测吧。

题目:说明以下四个例子输出的结果分别是什么。

自测题

各位,请开始你的表演,来看看以上4段代码分别输出的结果是什么?OK,记住你的答案,等看完文章解开谜底后,再来看看的答对了没。

单前导下划线

单前导下划线(_xxx),作为Python的命名约定,表示仅供内部使用。但注意这个命名约定,在类中你使用单前导线声明的变量,**依然可以在外部直接访问。**那这种命名约定还有什么意义呢?有!当代码使用

from modlue import *

导入某个模块时,单前导线这种定义方式的属性,不会被导入。举例:

# demo1.py
Name = "清风"
_Age = 18

# demo2.py
from demo1 import *

print(print(Name,_Age))

#output:
NameError: name '_Age' is not defined  

正常的情况是如上结果,但是万事无绝对,面试官的阴人考点来了:

__all__ = ["Name", "_Age"]

demo1.py在开头声明如上,使用__all__单独声明了可导入内容时,可以正常导入。虽然使用起来矛盾,但是面试阴人必备有木有?

单末尾下划线

单末尾下划线(xxx_),按照PEP8规定,单末尾下划线也是一个约定 用来避免与python关键字产生命名冲突。

例如:我们使用Beautifulsoup进行网页解析,通过类方法定位时,会找某个标签它的存在class=‘xxx’的情况,此时css的class与Python中的类重名,需要在class后添加单下划线进行区分。类似场景还有很多,就不一一列举了。

双前导和双末尾下划线

日常开发中,最好避免在自己的程序中使用以双下划线(“dunders”)开头和结尾的名称,因为它是Python语言定义的一种特殊方法(魔法方法),我们熟知的__init__ 、__dict__ 、__getitem__等等。

但是,如果你非要使用这种写法去声明,那可真是无底坑...如果你声明的变量不是内置的魔法方法,Python会将它当做普通的变量来操作。如果和内置的方法重名,要么重写,要么因为功能冲突而引发报错,所以不作死就不会死,还是别这么玩了。

双前导下划线

这个为什么放在最后,因为压轴啊!双前导下划线,在面试中被考到的几率太大了,尤其是那种长相猥琐,心术不正的面试官,最爱问这个知识点,所以要牢记。

首先双前导下划线(__xxx)的命名,90%情况下是真切的私有变量、方法,剩下10%一会儿再说。下来说说双前导下划线的作用,既然为私有属性,那么仅在当前类中可用,外部、子类均无法调用和继承。知道这点写代码差不多够了,但还差一点,拿文章开头的最后一个例子来说

# Test4
class Root:
    def __func(self):
        print('root')

class Child(Root):
    def __init__(self):
        self.__func()

Child()

大家刚才的答案是什么,root?恭喜你,打错了,结果是:

AttributeError: 'Child' object has no attribute '_Child__func'

不该是子类没有的方法,继承父类么,明明父类有,为什么会报错。刚才我们说到了,双前导下划线是真切的私有变量、方法,无法被子类所继承。如果我们把双前导下划线,变成了单前导下划线(如Test3),那么结果是root。

不能继承的问题明白了,但这个_Child__func是什么鬼?这就要说为什么刚才我说双前导下划线90%的情况下是真切的私有变量了、让我们来看下面的例子:

class Demo:
    def __init__(self):
        self.__name = "清风Python"

    def __say_hello(self):
        print(f"你好:{self.__name}")
        
D = Demo()
# usually
print(D.__name)
D.__say_hello

# specially
print(D._Demo__name)
D._Demo__say_hello()

我们定义一个Demo类,其中存在双前导下划线的__name __say_hello,当我们使用通常的调用方式时,是无法执行的,但Python的私有属性声明时,其实就是将某个私有属性前添加单下划线+类名,所以如果其实不存在什么私有属性,我们可以通过_classname__privatefunc的方式实现强制调用。

那么,日常中我们能否尽量的避免这种方式呢?方法是有的,但是只能骗骗初学者,对有心人还是没用,比如:

class Demo:
    def __init__(self):
        self.__money = 100

    @property
    def money(self):
        return self.__money
    
    @money.setter
    def money(self, pwd):
        pass
    
D = Demo()
D.money = 1000000
print(D.money) # 依旧为100块

python的property装饰器,可以将方法声明为类的属性,当某人调用D.money得到自己余额为100块时,肯定想着我重新赋值余额秒变土豪,但真实的余额我们使用的是私有的self.__money。而通过property创建了一个money的属性,当用户对money赋值时,money.setter的方法是空的,你怎么赋值都是无用的(空的干嘛还要写,因为不写会报错啊...AttributeError: can't set attribute)。

这样看起来很完美啊,为什么说只能骗骗初学者?当你打印print(D.__dict__){'_Demo__money': 100}一目了然。

最后,文章开头的测试题答案你做对了么?结果是:

child、root、root、报错 你答对了么?

今天关于Python中下划线的内容就到此为止,是否起到了稳固执行的效果呢?如果觉得有所收获,欢迎分享给你的小伙伴们一起进步啊!

END

下载1:动手学深度学习