在Python中,类也是以class开头定义的。我们定义一个动物类,它有名字和年龄,在java变量有实例变量和局部变量,方法内的变量是局部变量,类里面的变量是实例变量。那么在Python中的类及其属性这些是如何定义使用的呢?
class Animal():
pass
类的属性
我们知道类都是有属性的,在java中一般是在类中就定义了属性,而在python中是可以直接就给实例添加属性的。
>>> a = Animal()
>>> a.name = 'dog'
>>> print a.name
dog
>>> a
<__main__.Animal instance at 0x0000000002C5E708>
>>>
但是这个属性仅作用于这个实例变量,当我们再次新建一个实例的时候,就没有这个属性了。
>>> b = Animal()
>>> print b.name
Traceback (most recent call last):
File "<pyshell#16>", line 1, in <module>
print b.name
AttributeError: Animal instance has no attribute 'name'
>>>
那么如何创建一个类属性呢?可以在内部直接定义一个变量。
>>> class Student():
name = 'Student'
pass
>>> a = Student()
>>> print a.name
Student
>>>
实例化对象后可更改这个属性值。
>>> a.name = 'teacher'
>>> print a.name
teacher
>>>
类的私有属性
在java中我们知道可以通过private来控制访问限制,可以修饰变量,也可以修饰方法,使之只能在类的内部使用,那么在python中是如何实现的呢?通过在变量前添加‘__’来定义私有变量。
>>> class Student():
def __init__(self,name):
self.__name = name
>>> s = Student('Bob')
>>> s.name
Traceback (most recent call last):
File "<pyshell#69>", line 1, in <module>
s.name
AttributeError: Student instance has no attribute 'name'
>>> s.__name
Traceback (most recent call last):
File "<pyshell#70>", line 1, in <module>
s.__name
既然私有变量无法直接访问到,那么我们如何使用呢?当时是setter,getter方法,在类的内部为私有变量添加setter和getter方法我们就可以访问和修改它们了。这里就不做描述了。
限制类的属性
在python中既然是可以随意给实例添加属性的,那如果我们想限制只能添加给定的属性呢?需要用到__slots__
>>> class Animal():
__slots__ = ('name','age')
>>> a = Animal()
>>> a.name = 'dog'
>>> a.age = 24
>>> a.name
'dog'
>>> a.age
24
>>> a.color = red
Traceback (most recent call last):
File "<pyshell#9>", line 1, in <module>
a.color = red
NameError: name 'red' is not defined
>>>
构造方法
在java中有默认的构造方法,也可以根据我们的需要创建构造方法,实例化对象时根据需要创建符合我们需求的对象。在Python中也是有构造方法的。使用__init__,传入self及需要的变量。
>>> class Animal():
def __init__(self,name,age):
self.name = name
self.age = age
>>> a = Animal()
Traceback (most recent call last):
File "<pyshell#34>", line 1, in <module>
a = Animal()
TypeError: __init__() takes exactly 3 arguments (1 given)
>>> a = Animal('dog',23)
值得注意的是,当我们写了类的构造方法的时候,实例化对象的时候必须使用构造方法来实例化对象,否则就会报错误。
类方法
在类中不光有属性还有方法,类方法就是在类中直接定义方法。
>>> class Animal():
def __init__(self,name,age):
self.name = name
self.age = age
def printName(self):
print(self.name)
>>> a = Animal('dog',22)
>>> a.printName
<bound method Animal.printName of <__main__.Animal instance at 0x0000000002C64488>>
>>> a.printName()
dog
>>>
面向对象的三大特性,封装,继承,多态,封装不用多说了,在很多地方都有体现。我们来看下python中的继承和多态。
>>> class Animal():
def __init__(self,name,age):
self.name = name
self.age = age
def printName(self,sex):
print(self.name,sex)
>>> a = Animal('dog',22)
>>> a.printName('aa')
('dog', 'aa')
>>> class Dog(Animal):
pass
>>> b = Dog()
Traceback (most recent call last):
File "<pyshell#54>", line 1, in <module>
b = Dog()
TypeError: __init__() takes exactly 3 arguments (1 given)
\
>>> b = Dog('2哈',12)
SyntaxError: invalid syntax
>>> b = Dog('hashiqi',12)
>>> b
<__main__.Dog instance at 0x0000000002C64848>
>>>
我们定义一个Dog类继承于Animal,当我们实例话的时候发现它的构造方法也是继承于父类,需要传入名字和年龄,但是我们想让它拥有自己的构造方法怎么办呢?重写__init__方法。
>>> class Dog(Animal):
def __init__(self,name,color):
self.name = name
self.color = color
>>> b = Dog('2ha','red')
>>>
这样就体现了面向对象的继承和多态。