Python类


Python使用中面向对象的语言,支持继承、多态;

定义一个Person类:

>>> class Person:

... def sayHello(self):

... print('hello')

...

>>> Person.sayHello(None)

hello

>>> Person().sayHello()

hello

可以修改Person的类方法

>>> def hack_sayHello(obj):

... print('...hello')

...

>>>

>>> Person.sayHello = hack_sayHello

>>> Person.sayHello(None)

...hello

>>> Person().sayHello()

...hello

>>> sayHello = Person().sayHello

>>> sayHello()

...hello

Person().sayHello也是一个函数,可以赋值给变量,并可以直接调用;

>>> Person.sayHello is Person().sayHello

False

>>> Person.sayHello == Person().sayHello

False

Person.sayHello与Person().sayhello并不是同一个对象,直觉上,Person().sayHello关联(绑定)了一个Person实例,而Person.sayHello是一个类方法;

self参数事实上正是方法和函数的区别:方法将它们的第一个参数绑定到所属的实例上,因此这个参数可以不必提供;

>>> class Person:

... name = 'unkown'

... def sayHello(self):

... print('i\'m ' + name)

...

>>>

>>> Person.sayHello(None)

Traceback (most recent call last):

File "", line 1, in

File "", line 4, in sayHello

NameError: name 'name' is not defined

>>> p = Person()

>>> p.name = 'wyj'

>>> p.sayHello()

Traceback (most recent call last):

File "", line 1, in

File "", line 4, in sayHello

NameError: name 'name' is not defined

可见,Python在解析变量时,默认从local scope/global scope中查找;

>>> class Person:

... name = 'unkown'

... def sayHello(self):

... print('i\'m ' + self.name)

...

>>>

>>> Person.sayHello(None)

Traceback (most recent call last):

File "", line 1, in

File "", line 4, in sayHello

AttributeError: 'NoneType' object has no attribute 'name'

>>> p = Person()

>>> p.name = 'wyj'

>>> p.sayHello()

i'm wyj

访问成员都要通过self,假如以包含name属性的对象调用Person.sayHello(obj),是否可以呢?

>>> class Cat:

... name = 'huanhuan'

...

>>> Person.sayHello(Cat())

i'm huanhuan

可以,Python并不限制必须用相同类的实例对象作为参数调用类方法(貌似Python的类机制类似Javascript);

访问控制

\

<喎?http://www.Bkjia.com/kf/ware/vc/" target="_blank" class="keylink">vcD4KPHA+UHl0aG9usqKyu9axvdPWp7PWy73T0Le9t8POyqOstvjKx9Kqv7+zzNDy1LHX1Ly6sNHO1aGjPC9wPgo8cD4gIDwvcD4KPHA+sru5/aOsv8nS1NTayvTQ1MP7s8bHsLzTyc88c3Ryb25nPsurz8K7rs/fPC9zdHJvbmc+tvi4+Mbky73T0LfDzsrE3MGmo6i21M3isru/ybz7o6mjuzwvcD4KPHA+Pj4+IGNsYXNzIFBlcnNvbjo8L3A+CjxwPi4uLiBkZWYgX19wcml2YXRlX21ldGhvZChzZWxmKTo8L3A+CjxwPi4uLiBwcmludCg="private')

... def test(self):

... self.__private_method()

...

>>> Person().test()

private

>>> Person().__private_method()

Traceback (most recent call last):

File "", line 1, in

AttributeError: 'Person' object has no attribute '__private_method'

实际上,以上下划线打头的方法都有一个_ClassName__methodName的方法

>>> Person._Person__private_method

调用

>>> Person._Person__private_method(None)

private

总之,Python并不能阻止从类外进行方法调用;

类属性以及对象属性

\

首先,可以为类添加属性,新对象将得到属性的一份拷贝

>>> Person.age = 3

>>> Person().age

3

>>> Person.age = 4

>>> Person().age

4

>>> p = Person()

>>> Person.age = 31

>>> p.age

31

对类属性的修改,反映到了先前生成的对象的属性上,这说明类属性和对象的属性共享一个值;

>>> p.age = 34

>>> p.age

34

>>> Person.age

31

>>> Person.age = 99

>>> p.age

34

而一旦对对象的属性的修改,对象属性就拥有了自己的值,并不会反映到类属性上,而对类属性的修改,也不再反映到该对象的属性上;

这种行为与Javascript类似

评论关闭