类,函数,模块

1.png
image.png
image.png
image.png
image.png
image.png
image.png

作者:yang元祐
链接:https://www.zhihu.com/question/27699413/answer/267906889
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

2、继承的是方法而非自身属性??
所以,对于面向对象的继承来说,其实就是将多个类共有的方法提取到父类中,子类仅需继承父类而不必一一实现每个方法。

3、多继承和方法同名

1)、Python的类可以继承多个类,Java和C#中则只能继承一个类

2)、Python的类如果继承了多个类,那么其寻找方法的方式有两种,分别是:深度优先广度优先

272315068126604.jpg

  • 当类是经典类时,多继承情况下,会按照深度优先方式查找

  • 当类是新式类时,多继承情况下,会按照广度优先方式查找

经典类和新式类,从字面上可以看出一个老一个新,新的必然包含了跟多的功能,也是之后推荐的写法,从写法上区分的话,如果 当前类或者父类继承了object类,那么该类便是新式类,否则便是经典类。

272341313127410.jpg

272341553282314.jpg

eg:


image.png

经典类:首先去A类中查找,如果A类中没有,则继续去B类中找,如果B类中么有,则继续去D类中找,如果D类中么有,则继续去C类中找,如果还是未找到,则报错

新式类:首先去A类中查找,如果A类中没有,则继续去B类中找,如果B类中么有,则继续去C类中找,如果C类中么有,则继续去D类中找,如果还是未找到,则报错

注意:在上述查找过程中,一旦找到,则寻找过程立即中断,便不会再继续找了

4、
面向对象的应用场景:

1). 多函数需使用共同的值,如:数据库的增、删、改、查操作都需要连接数据库字符串、主机名、用户名和密码

![image](http://upload-images.jianshu.io/upload_images/9991116-b0be4bb45007cb06.gif?imageMogr2/auto-orient/strip)

![复制代码](http://upload-images.jianshu.io/upload_images/9991116-ea0e5ab8bb07f92e.gif?imageMogr2/auto-orient/strip) 

class SqlHelper:

    def __init__(self, host, user, pwd):

        self.host = host
        self.user = user
        self.pwd = pwd

    def 增(self):
        # 使用主机名、用户名、密码(self.host 、self.user 、self.pwd)打开数据库连接
        # do something
        # 关闭数据库连接

    def 删(self):
        # 使用主机名、用户名、密码(self.host 、self.user 、self.pwd)打开数据库连接
        # do something
        # 关闭数据库连接

    def 改(self):
        # 使用主机名、用户名、密码(self.host 、self.user 、self.pwd)打开数据库连接
        # do something
        # 关闭数据库连接

    def 查(self):
    # 使用主机名、用户名、密码(self.host 、self.user 、self.pwd)打开数据库连接
        # do something
        # 关闭数据库连接# do something
![复制代码](http://upload-images.jianshu.io/upload_images/9991116-0aa9f1d59503252c.gif?imageMogr2/auto-orient/strip)

2). 需要创建多个事物,每个事物属性个数相同,但是值的需求
如:张三、李四、杨五,他们都有姓名、年龄、血型,但其都是不相同。即:属性个数相同,但值不相同

![image](http://upload-images.jianshu.io/upload_images/9991116-c686f901f56ce582.gif?imageMogr2/auto-orient/strip)

![复制代码](http://upload-images.jianshu.io/upload_images/9991116-9b0e76e28bfdbbd5.gif?imageMogr2/auto-orient/strip) 

class Person:

    def __init__(self, name ,age ,blood_type):

        self.name = name
        self.age = age
        self.blood_type = blood_type

    def detail(self):
        temp = "i am %s, age %s , blood type %s " % (self.name, self.age, self.blood_type)
        print temp

zhangsan = Person('张三', 18, 'A')
lisi = Person('李四', 73, 'AB')
yangwu = Person('杨五', 84, 'A')

5、通过类创建对象时,如果每个对象都具有相同的字段,那么就使用静态字段
class Province:

# 静态字段
country = '中国'

6、方法
普通方法:由对象调用;至少一个self参数;执行普通方法时,自动将调用该方法的对象赋值给self;
类方法:由类调用; 至少一个cls参数;执行类方法时,自动将调用该方法的类复制给cls;
静态方法:由类调用;无默认参数;

class Foo:

def __init__(self, name):
    self.name = name

def ord_func(self):
    """ 定义普通方法,至少有一个self参数 """

    # print self.name
    print '普通方法'

@classmethod
def class_func(cls):
    """ 定义类方法,至少有一个cls参数 """

    print '类方法'

@staticmethod
def static_func():
    """ 定义静态方法 ,无默认参数"""

    print '静态方法'

调用普通方法

f = Foo()
f.ord_func()

调用类方法

Foo.class_func()

调用静态方法

Foo.static_func()

相同点:对于所有的方法而言,均属于类(非对象)中,所以,在内存中也只保存一份。

不同点:方法调用者不同、调用方法时自动传入的参数不同。

类方法有类变量cls传入,从而可以用cls做一些相关的处理。并且有子类继承时,调用该类方法时,传入的类变量cls是子类,而非父类。 对于类方法,可以通过类来调用,就像C.f(),有点类似C++中的静态方法, 也可以通过类的一个实例来调用,就像C().f(),这里C(),写成这样之后它就是类的一个实例了。
静态方法则没有,它基本上跟一个全局函数相同,一般来说用的很少

7、属性
因为Python中的属性其实是普通方法的变种。
class Foo:

def func(self):
    pass

# 定义属性
@property
def prop(self):
    pass

############### 调用

foo_obj = Foo()

foo_obj.func()
foo_obj.prop #调用属性


image.png

由属性的定义和调用要注意一下几点:

定义时,在普通方法的基础上添加 @property 装饰器;
定义时,属性仅有一个self参数
调用时,无需括号
方法:foo_obj.func()
属性:foo_obj.prop
注意:属性存在意义是:访问属性时可以制造出和访问字段完全相同的假象

属性由方法变种而来,如果Python中没有属性,方法完全可以代替其功能。

注:经典类中的属性只有一种访问方式,其对应被 @property 修饰的方法
新式类中的属性有三种访问方式,并分别对应了三个被@property、@方法名.setter、@方法名.deleter修饰的方法

由于新式类中具有三种访问方式,我们可以根据他们几个属性的访问特点,分别将三个方法定义为对同一个属性:获取、修改、删除

class Goods(object):

def __init__(self):
    # 原价
    self.original_price = 100
    # 折扣
    self.discount = 0.8

@property
def price(self):
    # 实际价格 = 原价 * 折扣
    new_price = self.original_price * self.discount
    return new_price

@price.setter
def price(self, value):
    self.original_price = value

@price.deltter
def price(self, value):
    del self.original_price

obj = Goods()
obj.price # 获取商品价格
obj.price = 200 # 修改商品原价
del obj.price # 删除商品原价

静态字段方式,创建值为property对象的静态字段:
class Foo:

def get_bar(self):
    return 'wupeiqi'

BAR = property(get_bar)

obj = Foo()
reuslt = obj.BAR # 自动调用get_bar方法,并获取方法的返回值
print reuslt

8、对于 from module_name import *
导入语句,如果想阻止导入模块内的全局变量可以使用旧有的规范,在全局变量上加一个前导的下划线

你可能感兴趣的:(类,函数,模块)