24. 企业级开发基础5:面向对象特征(封装)

面向对象程序设计最主要的有三个特征:封装、继承、多态

本节内容主要讲解面向对象的第一个特征:封装

1 封装的意义

在我们程序开发过程中,定义好类型之后就可以通过类型来创建对象
如:我们定义一个中华人民共和国公民的类型

# 创建一个人的类型
class Person(object):
    def __init__(name, age):
        self.name = name
        self.age = age

此时如果我们创建好对象之后,对对象的数据进行如下修改,大家是否认为合适呢?

# 创建一个人的对象
xiaoMing = Person("小明", 18)
# 修改属性
xiaoMing.age = 1000

我们会发现,上面的代码在运行时是正确的,也就是可以修改age属性为1000
此时我们需要明确一个概念:代码运行正确,但是不代表符合业务逻辑,这样的代码我们一般会说代码处理不合法!

2. 实现封装的过程

对于上面这样的问题,我们应该怎么处理呢
常规的方案就是:

  1. 定义一种这样的属性,属性只有在当前类的内部可以访问
  2. 类的外部不能访问这个属性,只能通过类提供的方法来进行属性的取值和赋值
  3. 在取值或者赋值的方法中,就可以添加一定的限制处理的代码了

python中,提供了这样的一种特殊的变量,变量名称使用两个下划线开头,这样的变量智能在类的内部访问,类的外部是访问不了的,我们称之为私有属性

# 定义类型
class Person(object):
    def __init__(self, name, age):
        self.__name = name;
        self.age = age
# 创建对象
xiaoMing = Person("小明明", 18)
# 访问属性
print(xiaoMing.age)
~ 18,可以访问,age只是一个普通的成员属性
print(xiaoMing.__name)
~ 出现错误,AttributeError: 'Person' object has no attribute '__name'

这样我们就限制了变量的访问范围。
但是变量定义出来就是为了被访问和操作的,如果上述代码中的__name一旦限制了不让访问,就木有存在的价值了。所以,我们通过自定义的方法给__name属性添加访问控制

# 创建一个人的类型
class Person(object):  
    # 对象的初始化方法
    def __init__(self, name):
       # 初始化私有成员变量__name
        self.__name = name
    # 定义获取__name的函数
    def get_name(self):
        return self.__name
    # 定义设置__name的函数
    def set_name(self, name):
        self.__name = name
# 根据类型创建对象
xiaoMing = Person("小明");
# 访问数据
xiaoMing.set_name("小明明");
print(xiaoMing.get_name()) 
# 执行结果:小明明

OK,通过以上对属性进行私有化(属性名称双下划线开头),给属性提供set/get的访问方法完成封装过程,此时就可以对本文开头的年龄设置问题进行一定的逻辑限制了

# 定义一个人的类型
class Person(object):
    # 初始化变量
    def __init__(self, name, age):
        self.__name = name
        if(age >= 0 and age <= 100):
            self.__age = age
        else:
            age = 18
    # 定义访问属性的get方法
    def get_name(self):
        return self.__name
    def get_age(self):
        return self.__age
    # 定义访问属性的get方法
    def set_name(self, name):
        self.__name = name;
    def set_age(self, age):
        if(age >= 0 and age <= 100):
            self.__age = age
        else:
            print("您设置的年龄不合法,还原默认值")
# 创建对象
p = Person("张小凡", 19)
p.set_age(1000)
print(p.get_age())
# 执行结果
~ 您设置的年龄不合法,还原默认值
~ 19

以上执行的结果,才是我们想要的结果

什么是封装
封装,就是将对象敏感的数据封装在类的内部,不让外界直接访问,但是提供了让外界可以间接访问的set/get方法,我们可以在set/get方法中添加数据的访问限制逻辑,完善我们的代码,提高程序的健壮性

3. 封装的高级使用方式

我们从上面的代码中已经看到了,可以通过函数操作的形式来进行属性的处理
但是某些情况下,函数操作的形式并不是特别美妙,我们突发奇想~想再提供了set/get访问方法的情况下,对属性的操作还能像以前那样直接赋值或者取值进行操作

# 封装以后通过函数操作的方式
p.set_name("tom")
print(p.get_name())
# 封装以前通过属性直接操作的方式
p.name = "tom"
print(p.name)

将类中的set/get方法操作的形式,转换成属性直接操作的形式,python中是可以的

首先:给get方法上添加@property注解,(关于注解的东东,之前的函数装饰器章节中已经有使用,可以参考一下操作原理),就可以让get方法当成属性进行直接取值操作了
其次,@property同时它会自动生成一个@get方法名称.setter注解,将@get方法名称.setter注解写在set方法上,就可以让set方法进行直接赋值操作了,代码如下:

class Person(object):
    def __init__(self, name):
        self.__name = name;
    @property
    def get_name(self):
        return self.__name
    @get_name.setter
    def set_name(self, name):
        self.__name = name
# 创建对象
p = Person("tom")
print(p.get_name)
p.set_name = "jerry"
print(p.get_name)
# 执行结果
~ tom
~ jerry

上述代码我们可以看出来,set/get方法已经可以当成普通的属性取值赋值的方式进行快捷的操作了,我们继续改造一下上述代码,让set/get更加符合属性取值赋值的方式

class Person(object):
    def __init__(self, name):
        self.__name = name;
    @property
    def name(self):
        return self.__name
    @name.setter
    def name(self, name):
        self.__name = name
# 创建对象
p = Person("tom")
print(p.name)
p.name= "jerry"
print(p.name)
# 执行结果
~ tom
~ jerry

此时,你还能在不看原来类型定义中的get/set,区分出来name是否是Person类型的属性还是方法呢?

封装的注解方式,在一定程度上,能隐藏我们方法在底层的实现,让调用者的操作变得简单。但是同时也降低了代码的可读性,后续的操作中,我们还是遵循原来封装的操作方案将类的属性私有化,提供set/get方法进行属性的操作。


大牧莫邪.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,830评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,992评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,875评论 0 331
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,837评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,734评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,091评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,550评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,217评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,368评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,298评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,350评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,027评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,623评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,706评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,940评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,349评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,936评论 2 341

推荐阅读更多精彩内容