iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Python面向对象中的封装详情
  • 875
分享到

Python面向对象中的封装详情

2024-04-02 19:04:59 875人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录一 封装的概念二 _ 和__ 对属性和方法的私有化1. 单下划线_2. 双下划线__3. 子类中访问父类的私有属性和私有方法三 访问及修改类的私有属性和私有方法1. 自定义公有方

一 封装的概念

封装其实在我们的生活中处处都是,如电视机,电脑,手机等物品。我们通常只能看到其外部的形状,以及使用他们提供的功能,并不能看到其内部复杂的硬件组成,这些都是封装好的,不能让我们看到,避免我们的一些“特殊”操作,使其不能正常工作。编程源于生活。在python中也有对对象的封装操作,使其对外只提供固定的访问模式,不能访问其内部的私有属性和私有方法。Python中的封装,一般指的是对类属性,类方法的封装,即类属性私有化和类方法私有化,具体如下面的小结所讲。

二 _ 和__ 对属性和方法的私有化

1. 单下划线_

当类中的属性和方法以_ 单下划线开头时,即说明这是类的保护变量和保护方法,按照编码约定,是不希望被外部访问的。但如果你要进行访问,也不会报错。

如下:

class A():
    #_ 声明是保护属性和保护方法
    _name = '张三'
    def __init__(self):
        self._age = 23

    def _method(self):
        print("我叫{},今年{}岁".fORMat(self._name, self._age))

if __name__ == '__main__':
    a = A()
    #打印类A的dir
    print(a.__dir__())
    #访问保护变量和保护方法
    print(a._name)
    a._method()

输出结果:

>>>
['_age', '__module__', '_name', '__init__', '_method', '__dict__', '__weakref__', '__doc__', '__repr__', '__hash__', '__str__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__']
张三
我叫张三,今年23岁

可以看出,以_单下划线开头的属性和方法其实在类外部是可以访问的,但是根据约定,当我们看见这样的属性和方法时,不应该在外部对其进行访问。

2. 双下划线__

上面以单下划线开头的属性和方法虽然是保护的,但是在外部还是可以访问的。而当你看到以双下划线__开头的属性和方法时,请记住它们是类的私有属性和私有方法,在类外以及子类中以常规访问类属性类方法的方法是无法访问的,也无法对其进行修改,如下

class B():
    #__ 声明是私有化的
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        #私有方法
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def fun(self):
        #公有方法
        print("this is a public method")
 
 if __name__ == '__main__':
    b = B()
    #打印类B的dir
    print(b.__dir__())
    #访问类B的私有属性和私有方法
    b.fun()
    print(b.__name, b.__age, b.__luange)
    b.__method()

输出结果:

>>>
['_B__age', '_B__luange', '__module__', '_B__name', '__init__', '_B__method', 'fun', '__dict__', '__weakref__', '__doc__', '__repr__', '__hash__', '__str__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__']
this is a public method
Traceback (most recent call last):
  File "C:/Users/admin/python-learning/python学习文件/Python基础/python类封装.py", line 56, in <module>
    print(b.__name, b.__age, b.__luange)
AttributeError: 'B' object has no attribute '__name'

从结果可以看出,访问类B的公有方法fun()是正常输出的,但是当我们访问私有属性name时就抛错:类B没有name属性。上面单下划线时,我们打印类A的dir,可以看到类A的name属性和method方法在dir里面是下面这样的

上面我们也打印了类B的私有属性和私有方法,如下:

可以看到私有属性和私有方法都变成了_B__属性和_B__方法的形式,所以我们在上面以__ name或者name的形式去访问是报错的,其实我们如果以 类名(). _ 类名__ 属性(实例属性)或者类名. _ 类名__ 属性(类属性)的形式去访问,还是会访问成功的。如下

class B():
    #__ 声明是私有化的
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        #私有方法
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def fun(self):
        #公有方法
        print("this is a public method")

if __name__ == '__main__':
    b = B()
    #打印类B的dir
    print(b.__dir__())
    #访问类B的私有属性和私有方法
    b.fun()
    print(B._B__name, b._B__age, b._B__luange)
    b._B__method()

结果如下:

>>>
['_B__age', '_B__luange', '__module__', '_B__name', '__init__', '_B__method', 'fun', '__dict__', '__weakref__', '__doc__', '__repr__', '__hash__', '__str__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__']
this is a public method
张三 23 python
我叫张三,今年23岁,我喜欢python。

3. 子类中访问父类的私有属性和私有方法

子类无法访问父类的私有属性和私有方法:

class B():
    #__ 声明是私有化的
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        #私有方法
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def fun(self):
        #公有方法
        print("this is a public method")

class C(B):

    def __init__(self):
        super().__init__()

    def fun1(self):
        #访问父类B的私有属性和私有方法
        print(self.__name, self.__age, self.__luange)
        self.__method()

if __name__ == '__main__':
    c = C()
    c.fun1()

输出结果:

>>>
AttributeError: 'C' object has no attribute '_C__name'
AttributeError: 'C' object has no attribute '_C__method'

可以看出子类也是无法访问父类的私有属性和私有方法的。

当子类中的的属性和方法与父类的私有属性,私有方法同名时,不会覆盖父类的私有属性和私有方法。

class B():
    #__ 声明是私有化的
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        #私有方法
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def fun(self):
        #公有方法
        print("this is a public method")

class C(B):
    __name = '李四'
    def __init__(self):
        super().__init__()
        self.__age = 24
        self.__luange = 'c++'

    def fun1(self):
        #访问父类B的私有属性和私有方法
        print(self.__name, self.__age, self.__luange)
        self.__method()

    def __method(self):
        #类C的私有方法,与父类方法同名,但不重写父类方法
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))
        #调用父类的私有方法
        B()._B__method()

if __name__ == '__main__':
    c = C()
    #访问类C的私有方法
    c._C__method()

结果如下:

>>>
我叫李四,今年24岁,我喜欢C++。
我叫张三,今年23岁,我喜欢python。

可以看到,子类C并没有重写父类B的__method()方法。这是为什么呢?我们打印一下B和C的dir,如下:

>>>
['_B__age', '_B__luange', '_C__age', '_C__luange', 'fun1',
 '_C__method', '__doc__', '_B__name', '_B__method', '_C__name', ...]

可以看到,在类C的dir中,父类B的私有属性和私有方法是以 _B__属性(方法)存在的,二类C自己的私有属性和私有方法是以_C__属性(方法)存在的,即类的私有属性和私有方法会以_类名_属性(方法)的形式存在dir中,所以当子类的属性和方法与父类的私有属性和私有方法同名时,并不会覆盖重写。

三 访问及修改类的私有属性和私有方法

类通过对属性和方法的私有化,可以对其起到封装保护作用。但是,当外部需要访问和改变时怎么办呢?就像电视机,电脑也会对外提供固定的接口。
上面,虽然我们可以通过类名(). _ 类名__ 属性(实例属性)或者类名. _ 类名__ 属性(类属性)的形式去访问类的私有属性和私有方法,但是这是违反编程规范的,不支持这么做,就像不会拆开电视机对其操作一样。

正确对类的私有属性和私有方法进行访问修改的一般有两种发方法,如下:

1. 自定义公有方法

class D():
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def get_value(self):
        return self.__name, self.__age, self.__luange

    def get_method(self):
        self.__method()

    def set_value(self, name, age, luange):
        self.__name, self.__age, self.__luange = name, age, luange

if __name__ == '__main__':
    d = D()
    #通过get_value方法访问私有属性
    print(d.get_value())
    #通过get_method方法访问私有方法
    print('=' * 30)
    d.get_method()
    #通过set_value方法修改私有属性
    print('='*30)
    print(d.get_value())
    d.set_value('王二麻子', 25, 'linux')
    print(d.get_value())
    d.get_method()

输出结果:

>>>
('张三', 23, 'python')
==============================
我叫张三,今年23岁,我喜欢python。
==============================
('张三', 23, 'python')
('王二麻子', 25, 'Linux')
我叫王二麻子,今年25岁,我喜欢Linux。

可以看到,我们通过自定义的的get_value(),get_method()以及set_value()方法就实现了对私有属性和私有方法的访问和修改。

2. property

property一般有两个作用,如下:

  • 作为装饰器,@property 将类的方法转为只读的类属性
  • property 重新实现一个属性的 getter 和 setter 方法

来看看下面这个E类,如下:

class E():
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    def get_value(self):
        return self.__name

    def set_value(self, name):
        self.__name = name

    getValue = property(get_value, set_value)

    @property
    def get_method(self):
        self.__method()

if __name__ == '__main__':
    e = E()
    #访问
    print(e.getValue)
    e.get_method
    #修改
    e.getValue = '王二'
    print(e.getValue)

结果:

>>>
张三
我叫张三,今年23岁,我喜欢python。
王二

可以看到,我们将get_valueset_value方法传入property后,类方法就转换成类属性,并赋值给getValue变量。此时e.getValue就是只读,即get_value方法,e.value = ‘王二’ 就是修改,即get_value方法。同一,通过@propert,将get_method方法,变成了属性。

下面property 重新实现一个属性的 getter 和 setter 方法,不同于上面的写法,较上面常用。

class E():
    __name = '张三'

    def __init__(self):
        self.__age = 23
        self.__luange = 'python'

    def __method(self):
        print("我叫{},今年{}岁,我喜欢{}。".format(self.__name, self.__age, self.__luange))

    @property
    def name(self):
        return self.__name
    @name.setter
    def name(self, name):
        self.__name = name

if __name__ == '__main__':
    e = E()
    #访问
    print(e.name)
    #修改
    print("修改前:", e.name)
    e.name = '隔壁老王'
    print("修改后:", e.name)

输出结果:

>>>
张三
修改前: 张三
修改后: 隔壁老王

上面是首先把name方法送给propert装饰器进行装饰,然后调用装饰后的setter方法,即可实现对私有属性进行修改。

到此这篇关于Python面向对象中的封装详情的文章就介绍到这了,更多相关Python封装内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Python面向对象中的封装详情

本文链接: https://www.lsjlt.com/news/141569.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Python面向对象中的封装详情
    目录一 封装的概念二 _ 和__ 对属性和方法的私有化1. 单下划线_2. 双下划线__3. 子类中访问父类的私有属性和私有方法三 访问及修改类的私有属性和私有方法1. 自定义公有方...
    99+
    2024-04-02
  • python 面向对象之class和封装
    # 封装 # Python并没有真正的私有化支持,但可用下划线得到伪私有 访问私有变量:实例._类名__变量名 访问私有方法:实例._类名__方法名() class Wife02(...
    99+
    2024-04-02
  • Python面向对象编程之类的封装
    目录1、封装的理解2、私有类属性、公开类属性、私有实例属性和公开实例属性2.1 公开类属性2.2 私有类属性2.3 公开实例属性2.4 私有实例属性2.5 私有属性不一定真的私有3、...
    99+
    2024-04-02
  • Python面向对象中封装的概念是什么
    这篇文章将为大家详细讲解有关Python面向对象中封装的概念是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一 封装的概念封装其实在我们的生活中处处都是,如电视机,电脑,手机等物品。我们通常只能看到其...
    99+
    2023-06-29
  • Python面向对象的程序设计详情
    1. 设计一个圆类,包括圆心位置、半径、颜色等属性。编写构造方法, 计算周长和面积。 import math class Circle:     def __init__(self,...
    99+
    2024-04-02
  • 详解C语言面向对象编程中的封装
    目录前言一、面向对象基本概念什么是对象?对象与类面向对象的编程方式二、C语言实现面向对象面向对象的三大特征面向对象之封装简介代码实现–基础版代码实现-进阶版总结前言 面向...
    99+
    2024-04-02
  • Java面向对象如何封装
    这篇文章主要讲解了“Java面向对象如何封装”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java面向对象如何封装”吧!1.认识封装简单的一句话就是套壳屏蔽细节。比如说一部手机,里面有电池,...
    99+
    2023-06-30
  • 详解JavaScript面向对象实战之封装拖拽对象
    目录概述1、如何让一个DOM元素动起来2、如何获取当前浏览器支持的transform兼容写法3、如何获取元素的初始位置5、我们需要用到哪些事件?6、拖拽的原理7、我又来推荐思维导图辅...
    99+
    2024-04-02
  • AJPFX:关于面向对象的封装
    回顾        面向对象 -- 注重的是结果,强调的是具备功能的对象。        面向过程 -- 强调的是函数,注重的实现的过程。  &nbs...
    99+
    2023-06-02
  • Java 面向对象 之 封装方法
    转载于 : http://www.verejava.com/id=16992728331734 public class Encapsulation {public static&...
    99+
    2023-06-02
  • 面向对象封装、继承、多态
    一、面向对象封装   01. 封装 封装 是面向对象编程的一大特点 面向对象编程的 第一步 —— 将 属性 和 方法 封装 到一个抽象的 类 中 外界 使用 类 创建 对象,然后 让对象调用方法 对象方法的细节 都被 封装...
    99+
    2023-01-31
    面向对象 多态
  • Java全面分析面向对象之封装
    目录什么是封装呢封装的好处意义getter方法和setter方法toString方法面向对象封装之包自定义包什么是封装呢 封装就是一种将数据和操作数据的方法进行有机结合,一种函数抽象...
    99+
    2024-04-02
  • Java面向对象的封装你了解吗
    目录面向对象思想之封装什么是封装呢?封装的好处意义getter方法和setter方法toString方法总结:面向对象思想之封装 或许大家都听说过java是纯面向对象语言,面向对象思...
    99+
    2024-04-02
  • 详细理解JAVA面向对象的封装,继承,多态,抽象
    目录类和对象的使用(面向对象思想落地的实现):子类对象实例化的全过程1.从结果上看:(继承性)2.从过程上来看:1.封装性2.继承性继承性的好处:3.多态性虚拟方法调用4.抽象性1....
    99+
    2024-04-02
  • PHP面向对象之封装,继承与多态详解
    在普通的编程中,没有涉及架构或者良好的设计,绝大多数都是使用的面向过程的方式。 当编程逐步深入后,就需要合理使用面向对象的知识来设计程序,而不是简单地脑海里有了思路就去写代码来实现,...
    99+
    2024-04-02
  • PHP面向对象编程怎么对对象进行封装
    这篇文章主要讲解了“PHP面向对象编程怎么对对象进行封装”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“PHP面向对象编程怎么对对象进行封装”吧!我们今天为大家介绍的是关于PHP面向对象编程的...
    99+
    2023-06-17
  • Python面向对象的三大特性封装、继承、多态
    Python是一门面向对象的语言。面向对象都有三大特性:封装、继承、多态。 下面分别来说说这三大特性: 1、封装 隐藏对象的属性和实现细节,仅对外提供公共访问方式。在python中用...
    99+
    2024-04-02
  • PHP中封装性与面向对象编程的关系
    在面向对象编程中,封装性是一个非常重要的概念。它允许开发者将数据和方法包装在一个类中,并通过对外提供接口来访问和操作数据,同时隐藏内部的实现细节。在PHP中,封装性与面向对象编程密不可分。封装性的主要目的是实现信息隐藏。通过将数据和方法封装...
    99+
    2023-10-21
    PHP 面向对象编程 封装性
  • 详析Python面向对象中的继承
    目录一单继承1.继承的基本语法格式如下2.查看类继承情况3.继承中的属性和方法4.初始化函数__init__()和super二多层继承三多重继承一 单继承 类继承作为python的三...
    99+
    2024-04-02
  • Python面向对象三大特征 封装、继承、多态
    目录1、封装1.1 私有属性和私有方法2、继承2.1 方法的重写2.2 在子类方法中调用父类方法2.3 多继承2.4 新式类和经典类 3、多态1、封装 封装: 根据 职责 将 属性 ...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作