Python学习总结【连载】(十三)

Python学习总结【连载】(十三)

2018.06.28 Juttachen


概述

  • 42 属性私有化
  • 43 假的私有属性
  • 44 类字段和类方法
  • 45 静态方法
  • 46 类的继承
  • 47 类的继承重写

四十二、属性私有化(了解)

1.访问权限

属性的访问权限:
公开、保护(python中没有)、私有
公开: 指的是在类的外部可以直接使用(默认)
公开属性: 属性名是普通单词(字符串,就是不以两个下划线开头的属性)
私有: 只能在类的内部直接使用,但是可以间接使用
私有属性: 属性名以两个下划线开头(一定不以下划线结束)

    class Person:
        """人"""
        def __init__(self,name='',age=0):
            # name属性是公开的属性
            self.name = name
            self.__age =age

        def show(self):
            print(self.name)
            # 私有属性可以在类的内部使用
            print(self.__age)

    p1 = Person('小明')
    print(p1.name)
    # 私有属性不能在类的外部使用
    # print(p1.__age) # AttributeError:'Person' object has no attribute '__age'

    p1.show()

四十三、假的私有属性

实际开发中,声明类的属性,很少会使用真正的私有属性(前面加两个下划线的)。
实际对属性的要求:
1.可以通过对象.属性语法方便的给属性赋值或者拿到他的值(所以不能是私有的)
2.给属性赋值的时候,需要对付过来的值进行规范,不能直接就让值赋给属性(所以不能是公开的)
如何满足要求:就使用假的私有属性 + getter 和 setter

  1. 声明属性,属性名前加一个下划线(这是一个规范,不是语法:带一个下划线的属性,不要直接通过一个下划线对应的属性名去访问这个属性)
  2. 通过给属性添加getter和setter来限制赋值和获取值的过程
    a. 添加getter(是一种方法):-->目的:限制获取属性值的操作,-->getter的作用:返回属性的值
    格式:

@property
def 前面不带下划线的属性名(self):
其他任何操作
return self.带一个下划线的属性名

b.添加setter
格式:

@去掉下划线的属性名.setter
def 去掉下划线的属性名(self, 参数):
其他任何操作
self.带下划线的属性名 = 参数

  1. 在类的外部通过不带下划线的属性去获取属性的值,或者给属性赋值

     class Person:
         def __init__(self,name='',age=0):
             self.name = name 
             self.age = age
    
     p1 = Person()
     p1.name = '小明'
     p1.name = 100 # 公开化就可以随意改变值的数据类型
     p1.name = ['',1] # 甚至是对象都可以赋
    

例子

    class Person:
        def __init__(self,name='',age=0):
            self._name = name 
            self._age = age

        # 给_name属性添加getter
        @property
        def name(self):
            if len(self._name) == 0:
                return '无名氏'
            return self._name

        # 给_name属性添加setter
        @name.setter
        def name(self, name1):
            self._name = name1

        # 给_age属性添加getter
        @property
        def age(self):
            # 判断是否是int类型
            if isinstance(self._age,int) == False:
                return '属性错误'
            # 判断是不是为0
            if self._age == 0:
                return '还未诞生'
            return self._age

        @age.setter
        def age(self,age1):
            self._age = age1    
    p1 = Person()
    # print(p1._name) #是不会报错的,但不推荐使用
    print(p1.name) #有了getter再写p1.name就没有问题了
    p1.name = "路飞"
    print(p1.name)
    p1.age = 17
    print(p1.age)

注意:
1.getter一般要添加,setter可以不用写(实际运用中,一般只加getter)
2.如果要添加setter必须添加getter

实例二:getter 和 setter 的使用以及关系

    class Cat:
        """猫"""

        def __init__(self,name='',color=''):
            self._name = name
            self._color = color

    cat1 = Cat('小花','yellow')
    cat1._name = '喵喵'  #推荐
    print(cat1._name)  #不推荐

四十四、类字段和类方法

1.类字段就是类属性:通过类来获取

类字段是声明在类中,函数外面的变量

    class Person:
        '''人类'''
        # 这个person_num就是一个类字段
        person_num = 0
        # 类字段要通过类来使用
        print(Person.person_num)

2.类方法:通过类调用的方法

开发中怎么确定声明成对象方法还是类方法:看实现的功能是否需要对象属性来支持,如果需要就必须申明成对象方法
注意:类方法不管是类还是对象都可以调用,实际开发的时候类方法就必须用类调用。
步骤:
1.@classmethod来说明下面的函数是类方法
2.所有的类方法都有一个默认参数cls,这个参数不需要传参,系统会自动将调用方法的类传给它

     class Person:
         '''人类'''
        @classmethod
        def hurt_earth(cls):
            print(cls)
            print('人类破坏环境,伤害地球',Person.person_num,cls.person_num)

    Person.hurt_earth()
    print(Person)

练习: 写一个数学类,提供数据的加,减,乘,除的功能

    class Math:
            """实现数据的加"""
        @classmethod
        def add(cls,*num):
            sum1 = 0
            for item in num:
                sum1 += item
            return sum1

    sum1 = Math.add(1,2,34,5)
    print(sum1)

四十五、静态方法

静态函数:在类中声明,由类来调用的方法
实例一:静态方法

    class Math:
        """数学类"""

        # multiply就是一个静态方法
        @staticmethod
        def multiply(num1,num2):
            return num1 *num2

    # 静态方法需要使用类来调用
    print(Math.multiply(10,4))

实例二:静态方法和类方法的区别

静态方法和类方法都是通过类来调用

区别:

  1. 类方法都有一个默认参数cls指向调用方法的类,但是静态方法没有

  2. 类型不一样,静态方法的类型是function,类方法的类型是method

     class Download:
         """下载类"""
    
         @staticmethod
         def download_image(image_file):
             print('下载%s下的图片'%(image_file))
    
    
         @classmethod
         def download_movie(cls,movie_file):
             print('下载电影:%s'%(movie_file))
    

    类中普通的函数也是通过类来调用(就是一个静态方法)

         def download(num):
             print('aaa',num)
    
    
     Download.download_image('aa/abc.png')
     print(Download.download_image)
    
     Download.download_movie('aa/123.mp4')
     print(Download.download_movie)
    
     print(Download.__dict__)
     print(Download.download(5))
     print(Download.download)
    

四十六、类的继承

继承: 继承就是让子类去拥有父类的东西
子类: 继承者
父类(超类): 被继承者

什么时候用继承:
在写一个类的时候,发现这个类中的部分属性和方法另一个类都有,这个时候就不用去声明这些属性和方法,直接从类中继承就好了

实例一:怎么继承
格式:
class 子类(父类):
子类的内容

    class Person:
        """人"""
        def __init__(self,name='aaa',age=0):
            self.name = name
            self.age = age


    class Student(Person):
        pass

    stu1 = Student()
    print(stu1.name)

实例二:可以继承哪些东西?
1.公开的属性可以继承
2.私有的属性不可以继承
3.公开的对象方法可以继承,私有的不行
4.类的字段可以继承
5.类的方法和静态也可以继承

总结:除了私有的其他的都可以继承,slots是不能继承的

    class Animal:
        # 对象的属性
        __slots__ = ('name','__age')
        def __init__(self,name='bb'):
            self.name = name
            self.__age = 18

        def run(self):
            print(self.__age)

        # 对象方法
        def eat(self):
            print('吃东西')

        # 私有的对象方法,只能类的内部才能调用
        def __shout(self):
            print("叫")

        # 类字段
        count = 100
        # 类方法
        @classmethod
        def class_func(cls):
            print('class_func')
        # 静态方法
        @staticmethod
        def static_func():
            print('static_func')


    class Dog(Animal):

          def eat_bone(self):
            print('吃骨头')

    dog1 = Dog()
    dog1.eat()
    print(Dog.count)
    Dog.class_func()
    Dog.static_func()

    dog1.color = 'red'
    print(dog1.color)

四十七、类的继承重写

声明一个类,如果不声明其父类,那么这个类默认继承自object这个类。
object类是python中所有类直接或者间接的父类。

    class Aniaml(object):

        def __init__(self,name):
              self.name = name

        def shout(self):
              print('%s嗷嗷叫'%(self.name))

        def be_beat(self):
              print(self.name,'在跑')
              print('跑的时候嗷嗷叫')

示例一:什么是重写

重新实现从父类继承下来的方法

    # noinspection PyCompatibility
    class Dog(Aniaml):

        # 重写父类的shout方法。然后通过Dog对象调用shout执行的是子类的方法。
        def shout(self):
              print('%s汪汪叫'%(self.name))

        # 重写父类的be_beat,保留了父类实现的功能,并且添加了新的功能
          def be_beat(self):
            # super()指的就是父类的对象
            super().be_beat()
            print('然后奋起反抗,汪汪叫')

    dog1 = Dog('大黄')
    dog1.shout()
    dog1.be_beat()

示例二:init示例的继承和重写

    class Person:
        """人"""
        def __init__(self,name='',age=0):
            self.name = name
            self.age = age


    # noinspection PyCompatibility
    class Student(Person):
        """学生类"""

重写init方法要注意:
1.如果需要继承父类的对象属性,就需要通过super().init去保留父类的对象属性。然后再添加新的属性。

def __init__(self,name='bb',age=0,study_id=''):
    super().__init__(name,age)
    self.study_id = study_id
stu1 = Student()
print(stu1.name)
stu2 = Student(study_id='001')

练习:
写一个正方形类,拥有方法:求面积,求周长 拥有的属性是边长
写一个长方形类,拥有方法:求面积,求周长 拥有的属性是长和宽

class Rectangle:
"""长方形类"""
def __init__(self,a=5,b=10):
    self.a = a
    self.b = b

def area(self):
    return self.a * self.b

def perimeter(self):
    return 2 * (self.a + self.b)

class Quare(Rectangle):
'''正方形类'''
def __init__(self):
    super().__init__()
    self.a = self.b

class Rect:
"""长方形类"""

def __init__(self, length=5, width=10):
    self.length = length
    self.width = width

def area(self):
    return self.length * self.width

def perimeter(self):
    return 2 * (self.length + self.width)


class Square(Rect):
'''正方形类'''

def __init__(self,length = 0,width = 0,side = 0):
    super().__init__(length, width)
    self._side = 0

@property
def side(self):
    return self._side

@side.setter
def side(self,side):
    self.length = side
    self.width = side
    self._side = side

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

推荐阅读更多精彩内容