面向对象最重要的概念就是类(Class)和实例(Instance)
类是抽象的模板,而实例是根据类创建出来的一个个具体的“对象”
class 内部有属性和方法
**下面的object是继承的类
self可以当做类里面的实例的 形参 **
由于类可以起到模板的作用,因此,可以在创建实例的时候,把一些我们认为必须绑定的属性强制填写进去。通过定义一个特殊的__init__方法,在创建实例的时候,就把name,score等属性绑上去:
class Student(object):
def init(self, name, score):
self.name = name
self.score = score #在这里定义了类属性
上面对类属性的定义的一段话可以永远不变
注意到__init__方法的第一个参数永远是self,表示创建的实例本身,因此,在__init__方法内部,就可以把各种属性绑定到self,因为self就指向创建的实例本身。
有了__init__方法,在创建实例的时候,就不能传入空的参数了,必须传入与__init__方法匹配的参数,但self不需要传,Python解释器自己会把实例变量传进去
bart = Student(‘Bart Simpson’, 59)
bart.name
‘Bart Simpson’
bart.score
59
方法:面向对象编程的一个重要特点就是数据封装,可以直接在class内部定义打印函数 def print_score(std): print(’%s: %s’ % (std.name, std.score)) ,需要打印时直接print_score(bart)即可直接打印出来
诸如此类可以用def 增添很多新方法
notice:本质上函数名和变量都是对象,所以在有class之后,class内部的变量和函数返回值,均可以用 实例.变量 来引用,或者 实例.函数名()来引用函数返回值
和静态语言不同,Python允许对实例变量绑定任何数据,也就是说,对于两个实例变量,虽然它们都是同一个类的不同实例,但拥有的变量名称都可能不同
即:可以通过
bart = Student(‘Bart Simpson’, 59)
lisa = Student(‘Lisa Simpson’, 87)
bart.age = 8
bart.age
8
lisa.age
Traceback (most recent call last):
File “”, line 1, in
AttributeError: ‘Student’ object has no attribute ‘age’
给某一个实例添加新的属性,但是其他同 类 实例没有这个属性
访问限制
如果要让内部属性不被外部访问,可以把属性的名称前加上两个下划线__,在Python中,实例的变量名如果以__开头,就变成了一个私有变量(private),只有内部可以访问,外部不能访问,所以,我们把Student类改一改:
class Student(object):
def __init__(self, name, score):
self.__name = name
self.__score = score
def print_score(self):
print('%s: %s' % (self.__name, self.__score))
改完后,对于外部代码来说,没什么变动,但是已经无法从外部访问实例变量.__name和实例变量.__score了
这样就确保了外部代码不能随意修改对象内部的状态,这样通过访问限制的保护,代码更加健壮。
可通过给Student类加入get_name和get_score这样的方法从外部获取name和score:
class Student(object):
def get_name(self):
return self.__name
def get_score(self):
return self.__score
同理通过增加方法也可以在外部对其进行修改
你也许会问,原先那种直接通过bart.score = 99也可以修改啊,为什么要定义一个方法大费周折?因为在方法中,可以对参数做检查,避免传入无效的参数:
class Student(object):
def set_score(self, score):
if 0 <= score <= 100:
self.__score = score
else:
raise ValueError('bad score')