一,面向对象结构与成员
1,1 面向对象结构分析:
那么每个大区域又可以分为多个小部分:
1 class A: 2 3 company_name = '老男孩教育' # 静态变量(静态字段) 4 __iphone = '1353333xxxx' # 私有静态变量(私有静态字段) 5 6 7 def __init__(self,name,age): #普通方法(构造方法) 8 9 self.name = name #对象属性(普通字段)10 self.__age = age # 私有对象属性(私有普通字段)11 12 def func1(self): # 普通方法13 pass14 15 def __func(self): #私有方法16 print(666)17 18 19 @classmethod # 类方法20 def class_func(cls):21 """ 定义类方法,至少有一个cls参数 """22 print('类方法')23 24 @staticmethod #静态方法25 def static_func():26 """ 定义静态方法 ,无默认参数"""27 print('静态方法')28 29 @property # 属性30 def prop(self):31 pass32 33 类的结构细分
类有这么多的成员,那么我们先从那些地方研究呢? 可以从私有与公有部分,方法的详细分类两个方向去研究.
1,2面向对象的私有与公有
对于每一个类的成员而言都有两种形式:
- 公有成员,在任何地方都能访问
- 私有成员,只有在类的内部才能方法
私有成员和公有成员的访问限制不同:
静态字段(静态变量)
- 公有静态字段:类可以访问;类内部可以访问;派生类中可以访问
- 私有静态字段:仅类内部可以访问;
1 class C: 2 3 name = "公有静态字段" 4 5 def func(self): 6 print C.name 7 8 class D(C): 9 10 def show(self):11 print C.name12 13 14 C.name # 类访问15 16 obj = C()17 obj.func() # 类内部可以访问18 19 obj_son = D()20 obj_son.show() # 派生类中可以访问21 22 公有静态字段
1 class C: 2 3 __name = "私有静态字段" 4 5 def func(self): 6 print C.__name 7 8 class D(C): 9 10 def show(self):11 print C.__name12 13 14 C.__name # 不可在外部访问15 16 obj = C()17 obj.__name # 不可在外部访问18 obj.func() # 类内部可以访问 19 20 obj_son = D()21 obj_son.show() #不可在派生类中可以访问 22 23 私有静态字段
普通字段(对象属性)
- 公有普通字段:对象可以访问;类内部可以访问;派生类中可以访问
- 私有普通字段:仅类内部可以访问;
1 class C: 2 3 def __init__(self): 4 self.foo = "公有字段" 5 6 def func(self): 7 print self.foo # 类内部访问 8 9 class D(C):10 11 def show(self):12 print self.foo # 派生类中访问13 14 obj = C()15 16 obj.foo # 通过对象访问17 obj.func() # 类内部访问18 19 obj_son = D();20 obj_son.show() # 派生类中访问21 22 公有普通字段
1 class C: 2 3 def __init__(self): 4 self.__foo = "私有字段" 5 6 def func(self): 7 print self.foo # 类内部访问 8 9 class D(C):10 11 def show(self):12 print self.foo # 派生类中访问13 14 obj = C()15 16 obj.__foo # 通过对象访问 ==> 错误17 obj.func() # 类内部访问 ==> 正确18 19 obj_son = D();20 obj_son.show() # 派生类中访问 ==> 错误21 22 私有普通字段
方法:
- 公有方法:对象可以访问;类内部可以访问;派生类中可以访问
- 私有方法:仅类内部可以访问;
1 class C: 2 3 def __init__(self): 4 pass 5 6 def add(self): 7 print('in C') 8 9 class D(C):10 11 def show(self):12 print('in D')13 14 def func(self):15 self.show()16 obj = D()17 obj.show() # 通过对象访问 18 obj.func() # 类内部访问 19 obj.add() # 派生类中访问 20 21 公有方法
1 class C: 2 3 def __init__(self): 4 pass 5 6 def __add(self): 7 print('in C') 8 9 class D(C):10 11 def __show(self):12 print('in D')13 14 def func(self):15 self.__show()16 obj = D()17 obj.__show() # 通过不能对象访问18 obj.func() # 类内部可以访问19 obj.__add() # 派生类中不能访问20 21 私有方法
总结:
对于这些私有成员来说,他们只能在类的内部使用,不能再类的外部以及派生类中使用.
ps:非要访问私有成员的话,可以通过 对象._类__属性名,但是绝对不允许!!!
为什么可以通过._类__私有成员名访问呢?因为类在创建时,如果遇到了私有成员(包括私有静态字段,私有普通字段,私有方法)它会将其保存在内存时自动在前面加上_类名.
1.3面向对象的成员
1.3.1 字段
字段包括:普通字段和静态字段,他们在定义和使用中有所区别,而最本质的区别是内存中保存的位置不同,
- 普通字段属于对象
- 静态字段属于类
1 class Province: 2 3 # 静态字段 4 country = '中国' 5 6 def __init__(self, name): 7 8 # 普通字段 9 self.name = name10 11 12 # 直接访问普通字段13 obj = Province('河北省')14 print obj.name15 16 # 直接访问静态字段17 Province.country18 19 示例
上述代码可以看出【普通字段需要通过对象来访问】【静态字段通过类访问】,在使用上可以看出普通字段和静态字段的归属是不同的。其在内容的存储方式类似如下图:
由上图可是:
- 静态字段在内存中只保存一份
- 普通字段在每个对象中都要保存一份
应用场景: 通过类创建对象时,如果每个对象都具有相同的字段,那么就使用静态字段
1.3.2方法
方法包括:普通方法、静态方法和类方法,三种方法在内存中都归属于类,区别在于调用方式不同。
- 普通方法:由对象调用;至少一个self参数;执行普通方法时,自动将调用该方法的对象赋值给self;
- 类方法:由类调用; 至少一个cls参数;执行类方法时,自动将调用该方法的类复制给cls;
- 静态方法:由类调用;无默认参数;
1 class Foo: 2 3 def __init__(self, name): 4 self.name = name 5 6 def ord_func(self): 7 """ 定义普通方法,至少有一个self参数 """ 8 9 # print self.name10 print '普通方法'11 12 @classmethod13 def class_func(cls):14 """ 定义类方法,至少有一个cls参数 """15 16 print '类方法'17 18 @staticmethod19 def static_func():20 """ 定义静态方法 ,无默认参数"""21 22 print '静态方法'23 24 25 # 调用普通方法26 f = Foo()27 f.ord_func()28 29 # 调用类方法30 Foo.class_func()31 32 # 调用静态方法33 Foo.static_func()34 35 分别示例
相同点:对于所有的方法而言,均属于类(非对象)中,所以,在内存中也只保存一份。
不同点:方法调用者不同、调用方法时自动传入的参数不同。
1.3.2属性
什么是特性property
property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值
例一:BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属性,更便于理解)成人的BMI数值:过轻:低于18.5正常:18.5-23.9过重:24-27肥胖:28-32非常肥胖, 高于32 体质指数(BMI)=体重(kg)÷身高^2(m) EX:70kg÷(1.75×1.75)=22.86
1 class People: 2 def __init__(self,name,weight,height): 3 self.name=name 4 self.weight=weight 5 self.height=height 6 @property 7 def bmi(self): 8 return self.weight / (self.height**2) 9 10 p1=People('egon',75,1.85)11 print(p1.bmi)12 13 例一代码
为什么要用property
将一个类的函数定义成特性以后,对象再去使用的时候obj.name,根本无法察觉自己的name是执行了一个函数然后计算出来的,这种特性的使用方式遵循了统一访问的原则
由于新式类中具有三种访问方式,我们可以根据他们几个属性的访问特点,分别将三个方法定义为对同一个属性:获取、修改、删除
1 class Goods(object): 2 3 def __init__(self): 4 # 原价 5 self.original_price = 100 6 # 折扣 7 self.discount = 0.8 8 9 @property10 def price(self):11 # 实际价格 = 原价 * 折扣12 new_price = self.original_price * self.discount13 return new_price14 15 @price.setter16 def price(self, value):17 self.original_price = value18 19 @price.deltter20 def price(self, value):21 del self.original_price22 23 obj = Goods()24 obj.price # 获取商品价格25 obj.price = 200 # 修改商品原价26 del obj.price # 删除商品原价27 28 商品实例