标签:开头 没有 class 属性 继承 相同 pre ted info
Object Oriented Programming,简称OOP,是一种程序设计思想。OOP把对象作为程序的基本单元,一个对象包含了数据和操作数据的函数。
面向过程的程序设计把计算机程序视为一系列的命令集合,即一组函数的顺序执行。为了简化程序设计,面向过程把函数继续切分为子函数,即把大块函数通过切割成小块函数来降低系统的复杂度。
而面向对象的程序设计把计算机程序视为一组对象的集合,而每个对象都可以接收其他对象发过来的消息,并处理这些消息,计算机程序的执行就是一系列消息在各个对象之间传递。
类是对象的模板。即类是对一组有相同数据和相同操作的对象的定义,一个类所包含的方法和数据描述一组对象的共同属性和行为。
类是在对象之上的抽象,对象则是类的具体化,是类的实例。类可有其子类,也可有其它类,形成类层次结构。
给对象发消息实际上就是调用对象对应的关联函数,称之为对象的方法(Method).
综上,面向对象的设计思想是抽象出Class,根据Class创建Instance。
面向对象的抽象程度又比函数要高,因为一个Class既包含数据,又包含操作数据的方法。
面向对象的三大特点:数据封装、继承、多态
类是抽象的模板,而实例是根据类创建出来的一个个具体的“对象”,每个对象都拥有相同的方法,但各自的数据可能不同。
(1) 关键字class
(2) 类名:通常是大写开头的单词,如Student
(3)(继承类):表示该类是从哪个类继承下来的,通常,如果没有合适的继承类,就是用object类,这是所有类最终都会继承的类。
1 #这是最简单的定义类
2 class Student(object):
3 pass
4
5 #创建一个实例
6 bart=Student()
7 print(bart)
运行结果:
<__main__.Student object at 0x00000000010CC2E8>,变量bart指向Student的一个实例,后面的0x00000000010CC2E8是内存地址,每个object的地址都不一样。
作用:绑定属性
1,__init__方法的第一个参数永远是self,表示创建的实例本身,在__init__方法内部,把各种属性绑定到self
2,有了__init__方法,在创建实例的时候,必须传入与__init__方法匹配的参数,但self不需要传,Python解释器自己会把实例变量传进去。
在类的内部定义访问数据的函数,这样,就把“数据”给封装起来了,这些封装数据的函数是和类本身关联起来的,称为类的方法。
实战
1 class Student(object):
2
3 def __init__(self, name, score):
4 self.name = name
5 self.score = score
6
7 def print_score(self):
8 print(‘%s: %s‘ % (self.name, self.score))
9
10 def get_grade(self):
11 if self.score >= 90:
12 return ‘A‘
13 elif self.score >= 60:
14 return ‘B‘
15 else:
16 return ‘C‘
17
18 bart = Student(‘Bart Simpson‘, 59)
19 lisa = Student(‘Lisa Simpson‘, 87)
20
21 print(‘bart.name =‘, bart.name)
22 print(‘bart.score =‘, bart.score)
23 bart.print_score()
24
25 print(‘grade of Bart:‘, bart.get_grade())
26 print(‘grade of Lisa:‘, lisa.get_grade())
运行结果:
特殊变量:以双下划线开头,并且以双下划线结尾的,是特殊变量,类似__xxx__,特殊变量是可以直接访问的,不是 Private变量,所以,在需要特殊变量的地方不能用__name__、__score__这样的变量名。
私有变量(Private):有2种形式,一个下划线开头,两个下划线开头
这样的实例变量,外部是可以访问的,但是,按照约定俗成的规定,当你看到这样的变量时,意思是:“虽然我可以被访问,但是,请把我视为私有变量,不要随意访问。”
双下划线开头的实例变量是不是一定不能从外部访问呢?
答:也不是,不能直接访问__name,是因为python解释器对外把__name变量改成了_Student__name,所以,仍然可以通过_Student__name来访问__name变量。
但是强烈建议不要这么做,因为不同版本的Python解释器可能会把__name改成不同的变量名。
总的来说,Python本身没有任何机制阻止你干坏事,一切全靠自觉。
举例:
bart=Student(‘bart simpon‘,98)
备注:空白处相当于新增变量,不会改变score的值 | 1个下划线 | 2个下划线 |
bart.score | ||
bart._score | 可以访问 | |
bart.__score | ||
bart._Student_score | ||
bart._Student__score | 可以访问 |
备注:1,空白处相当于新增变量,不会改变score的值
2,可以访问,说明我们可以修改私有变量的值
3,既然已经设置成私有变量了,应尽量不要随意访问
1个下划线的代码:
1 class Student(object):
2
3 def __init__(self, name, score):
4 self._name = name
5 self._score = score
6
7 def print_score(self):
8 print(‘Print语句返回值:%s: %s‘ % (self._name, self._score))
9
10
11 bart=Student(‘bart simpon‘,98)
12
13 print("这是1个下划线的私有变量:")
14 bart.score=45
15 print("bart.score:",bart.score)
16 print("bart.print_score():",bart.print_score())
17 print("")
18
19
20 bart.__score=45
21 print("bart.__score:",bart.__score)
22 print("bart.print_score():",bart.print_score())
23 print("")
24
25
26 bart._Student_score=45
27 print("bart._Student_score:",bart._Student_score)
28 print("bart.print_score():",bart.print_score())
29 print("")
30
31
32 bart._Student__score=45
33 print("bart._Student__score:",bart._Student__score)
34 print("bart.print_score():",bart.print_score())
35 print("")
36
37 bart._score=45
38 print("bart._score:",bart._score)
39 print("bart.print_score():",bart.print_score())
一个下划线: 只有bart._score能修改_score的值
2个下划线的代码:
1 class Student(object):
2
3 def __init__(self, name, score):
4 self.__name = name
5 self.__score = score
6
7 def print_score(self):
8 print(‘Print语句返回值:%s: %s‘ % (self.__name, self.__score))
9
10
11
12
13 bart=Student(‘bart simpon‘,98)
14 # print("gds",bart.print_score())
15 print("这是2个下划线的私有变量:")
16 bart.score=45
17 print("bart.score:",bart.score)
18 print("bart.print_score():",bart.print_score())
19 print("")
20
21
22 bart._score=45
23 print("bart._score:",bart._score)
24 print("bart.print_score():",bart.print_score())
25 print("")
26
27
28 bart.__score=45
29 print("bart.__score:",bart.__score)
30 print("bart.print_score():",bart.print_score())
31 print("")
32
33
34 bart._Student_score=45
35 print("bart._Student_score:",bart._Student_score)
36 print("bart.print_score():",bart.print_score())
37 print("")
38
39
40 bart._Student__score=45
41 print("bart._Student__score:",bart._Student__score)
42 print("bart.print_score():",bart.print_score())
2个下划线: 只有bart._Student__score能修改__score的值
如果要使内部属性不被外部访问,可以在属性名前加两个下划线__,变成私有变量,只有内部可以访问,外部不能访问。
这样确保了外部代码不能随意修改对象内部的状态,这样通过访问限制的保护,代码更加健壮。
如果外部代码要获取name和score怎么办?可以给Student类增加get_name和get_score这样的方法。
如果外部代码要修改score怎么办?可以给Student类增加get_score方法。
代码如下:
1 class Student(object):
2
3 def __init__(self, name, score):
4 self.__name = name
5 self.__score = score
6
7 def print_score(self):
8 print(‘%s: %s‘ % (self.__name, self.__score))
9
10 def get_name(self):
11 return self.__name
12
13 def get_score(self):
14 return self.__score
15
16 def set_score(self, score): #使用这种方法修改属性值,可以检查传入的参数值是否有问题
17 if 0 <= score <= 100:
18 self.__score = score
19 else:
20 raise ValueError(‘bad score‘)
21
22 bart = Student(‘Bart Simpson‘, 59)
23 print("bart._Student__score:",bart._Student__score) #这种就运行正确了
24
25 print("bart.get_score():",bart.get_score())
26
27 print(bart.score) #运行结果出错,提示‘Student‘ object has no attribute ‘score‘
28
29 # print(bart.__score) #运行结果出错,提示‘Student‘ object has no attribute ‘__score‘
运行结果:
答:前者是相当于新增加了一个变量,所以能运行出结果,而后者的print( bart.score )这种访问形式是不能直接访问私有变量,所以报错,而print( bart._Student__score )却访问成功了
参考网址:https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000
标签:开头 没有 class 属性 继承 相同 pre ted info
原文地址:https://www.cnblogs.com/bravesunforever/p/10742250.html