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

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

作者: 清风Python | 来源:发表于2020-09-13 00:37 被阅读0次

    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中下划线的内容就到此为止,是否起到了稳固执行的效果呢?如果觉得有所收获,欢迎分享给你的小伙伴们一起进步啊!

    The End

    期待你关注我的公众号清风Python,如果你觉得不错,希望能动动手指转发给你身边的朋友们。
    我的github地址:https://github.com/BreezePython

    相关文章

      网友评论

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

        本文链接:https://www.haomeiwen.com/subject/avbgektx.html