Python类和对象笔记

小白笔记
仅记录常规操作中较为不熟悉的操作类型

1、类可以理解为一种自定义的数据类型,它定义了这一类对象普遍具有的或需要的属性(变量)、方法(自定义函数)

例如:学生是一个自定义类(可理解为特殊数据类型),学号、姓名、班级信息、授课老师等是其普遍性的属性,成绩记录工具、期末评价工具等是其普遍有的方法(工具)
具体学生:张三、李四,就是这个类的实例对象

2、类用 def __ init __() 方法来定义 对象 的属性,所有对象的方法(类定义中的自定义函数)的第一个参数均为:self

class Student(object):
     def __init__(self, name, score):
         self.name = name 
         self.score = score
 
 student1 = Student()
 studen1.name = "Tony"

如果要内部属性不被外部访问,可以把属性的名称前加上两个下划线 __,在Python中,实例的变量名如果以 __ 开头,就变成了私有变量(private),只有内部可以访问,外部不能访问

私有方法声明:用两个下划线开头的方法名(函数名)def __XX
注意的是,在Python中,变量名类似 __ xxx __ 的,也就是以双下划线开头,并且以双下划线结尾的,是特殊变量,特殊变量是可以直接访问的,不是private变量

3、在方法中引用对象的属性,需要用:self.属性名 的形式

 class Student(object):
     ...
     def get_grade(self):
         if self.score >=90: 
             return  'A' 
         elif self.score >=60: 
             return 'B' 
         else: 
             return 'C'

4、整个类的 属性 定义,直接在类定义中定义一个变量并复制即可,无需用__ init __方法

 class Student(object): 
     school_name ='Beijing Primary School'

5、类定义中 __ slots __ 变量,用来限定类属性数量

 class Student(object): 
     __slots__ = ('name', 'age')    # 用tuple定义允许绑定的属性名称

如果定义一个 Student 实例,限制不允许向该实例添加除 name 、age 以外的属性了

 >>> s = Student()    # 创建新的实例 
 >>> s.name ='Michael'    # 绑定属性'name'
 >>> s.age =25    # 绑定属性'age' 
 >>> s.score =99    # 绑定属性'score' 
 Traceback (most recent call last):  
      File"<stdin>", line1,in<module> 
 AttributeError:'Student'object has no attribute'score'

使用 __ slots __ 定义的属性仅对当前类实例起作用,对继承的子类是不起作用,除非在子类中也定义 __ slots __ ,这样,子类实例允许定义的属性就是自身的 __ slots __ 加上父类的 __ slots __

6、装饰器@property 对某些类对象的属性来说,可能要一定的范围限定如Student类中的分数 score 属性就有范围限制,所以我们就需要在类中定义一些方法来对其进行设置、验证和输出

 class Student(object): 
     def __init__(self, name, score): 
         self.name = name 
         self.score = score     

     def get_score(self): 
         return self._score 

     def set_score(self, value): 
         if not is instance(value, int): 
             raise ValueError('score must be an integer!') 
         if value < 0 or value  > 100: 
             raise ValueError('score must between 0 ~ 100!')  
          self._score = value 

 student1 = Student()
 student1.get_score()
 student1.set_score(90)

那我们在对属性 score 进行读取、设置时,就无法像属性 name 那样直接读取或通过赋值实现,需要通过 student1.get_score()、student1.set_score(90) 来实现,整个过程比较麻烦。
为了便于属性设置,我们可以通过装饰器@property 将这麻烦的过程省略掉,实现将方法变为按属性调用的过程

 class Student(object): 
     ……
     @property
     def score(self):
         return self._score

     @score.setter
     def score(self, value):
         if not is instance(value, int):
             raise ValueError('score must be an integer!')
         if value < 0 or value  > 100:
             raise ValueError('score must between 0 ~ 100!') 
         self._score = value
 
 student1 = Student()
 student1.score = 90
 student1.score

这样我们就可以直接通过属性调用的方式对 score 进行设置或读取:student1.score = 90、student1.score
@property 装饰方法后(读取),@property本身又创建了另外两个装饰器@score.setter(设置) 和 @score.deleter(删除),这两个装饰器可以用也可以不用

7、@classmethod 装饰器,严格说是一种类型声明。它声明了该方法为类的方法,非对象方法,大部分用于实例化前的数据清洗、转换、判断等用途。
我们新建一个日期类,在给出年、月、日的数字信息时,按特定格式输出

 class Demo:
     def __init__(self,year,month,day):
         self.year= year
         self.month= month
         self.day= day

     def out_date(self):
         return "year:%d, month:%d, day:%d" % (self.year,self.month,self.day)

 demo1= Demo(2017,12,25)
 print(demo1.out_date())

 #  year:2017, month:12, day:25

但"2017-5-6"形式的日期,就无法利用这个类来输出特定格式的日期,这就需要我们在遇到这种字符格式时,现对它进行一次处理,这时我们就需要要一个方法来实现这个处理
而这个方法并不少这一类对象普遍需要的方法,且与对象属性有冲突,所以用类方法的形式来实现

 class Demo:
     def __init__(self,year,month,day):
         self.year= year
         self.month= month
         self.day= day

     def out_date(self):
         return "year:%d, month:%d, day:%d" % (self.year,self.month,self.day)
 
     @classmethod
     def pre_out(cls,date_string):
         year, month, day= map(int,date_string.split("-") 
         return cls(year, month, day)    # 返回一个初始化的类,对类进行实例化
 
 demo2= Demo.pre_out('2017-5-6')    # 不需要实例化类就可以被类本身调用
 print(demo2.out_date())    # 实例化调用
 
 # year:2017, month:5, day:6

8、@staticmethod 装饰器,它严格上来说是一个普通的函数,不强制要求传递参数(不代表没有参数),与类和实例都没有所谓的绑定关系,所以没有要求第一个参数必须是cls或self
它其实是对类的一些逻辑关联的处理,比如对输入的日期进行合法性判断(day<=31,mouth<=12),这种判断可以在类定义外解决, 但这样同样会扩散类内部的代码,造成维护困难。所以我们将它放在类定义中

 class Demo:
     def __init__(self,year,month,day):
         self.year= year
         self.month= month
         self.day= day
 
     def out_date(self):
         return "year:%d, month:%d, day:%d" % (self.year,self.month,self.day)

     @staticmethod
     def is_date_valid(date_as_string1):      # 注意参数,跟类没有严格的绑定关系
         year, month, day= map(int,date_as_string1.split('-'))
         return day<= 31 and month<= 12 and year<= 3999

     @classmethod
     def pre_out(cls,date_string):
         if cls.is_date_valid(date_string):
             year, month, day= map(int,date_string.split("-"))
             return cls(year, month, day)
         else:
             return cls(1900,1,1)
 
 date= '2017-13-6'
 demo2= Demo.pre_out(date) 
 print(demo2.out_date())
 
 #  year:1900, month:1, day:1

比如,上文中的Student类中,我们想判断学生是否放寒暑假,就可以使用静态方法,它只是对日期做这个类相关的逻辑判断,并没有对类或实例属性做修改等

 class Student():
 
     def __init__(self,name,score):
         self.__name= name
         self.__score= score
 
     @staticmethod
     def is_holiday(day):
         num_month= day.month
             if num_monthin [7,8]:
                 print('Students are on holiday!')
             else:
                 print('Students are attending school.')
 
 from datetime import date
 
 day= date.today()
 Student.is_holiday(day)  # 不需要实例化类就可以被类本身调用
 
 #    Students are attending school.

参考链接:
https://www.jianshu.com/p/eea864224e2d
https://blog.csdn.net/dyh4201/article/details/78336529
https://blog.csdn.net/ljt735029684/article/details/80714274
https://www.zhihu.com/question/20021164/answer/18224953
https://www.cnblogs.com/wangyongsong/p/6750454.html
https://www.jianshu.com/p/b92be62d7ad7
https://blog.csdn.net/caroline_wendy/article/details/23383995

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

推荐阅读更多精彩内容

  • 要点: 函数式编程:注意不是“函数编程”,多了一个“式” 模块:如何使用模块 面向对象编程:面向对象的概念、属性、...
    victorsungo阅读 1,495评论 0 6
  • 定义类并创建实例 在Python中,类通过 class 关键字定义。以 Person 为例,定义一个Person类...
    绩重KF阅读 3,948评论 0 13
  • 写在前面的话 代码中的# > 表示的是输出结果 输入 使用input()函数 用法 注意input函数输出的均是字...
    FlyingLittlePG阅读 2,753评论 0 8
  • 1. 使用__slots__ 正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实...
    时间之友阅读 292评论 0 1
  • 每每在日常生活中实践智慧的时候,我总是会觉得我是背一把大剑的勇士,特别是坏种子爆发的时候感觉特别像,我就像在开辟一...
    雅子12345阅读 199评论 0 0