黑马程序员技术交流社区

标题: 粗谈python的面向对象 [打印本页]

作者: 阿木木的鞋带    时间: 2018-6-22 21:34
标题: 粗谈python的面向对象

类与对象


1.定义类:


类是具有相同属性和行为的事物的统称


class Cat:
    """
    函数体
    """
    pass
2.创建对象:


某一个具体事物的存在 现实生活中看得见 摸得着


对象名1 = 类名()


对象名2 = 类名()
多个对象之间互不干扰 相互独立


3.为对象添加属性


tom = Cat()
tom.name = '汤姆'
print(tom.name)
4.在方法内获取对象的属性:


class Cat:
    """
    函数体
    """
    def eat(self):
        print('%s在喝水' % tom.name)
tom = Cat()
tom.name = '汤姆'
tom.eat()
print(tom.name)
5.self的作用


self 代表了当前调用方法的对象


6.__init__的使用


使用: 方法在创建对象后执行 一般用来初始化对象的共有属性


__init__传参的作用: 可以在self 后面定义需要用到的参数 init需要的参数,在创建对象时传递


def __init__(self,new_name):
        """
        魔法方法:在恰当的时机会自动执行的方法
        
        """
        print('init 执行了')
        self.name = new_name
class Cat:


    def __init__(self, new_name):
        self.name = new_name
        
xiaobai = Cat('伊丽莎白')


7.全局变量与属性的区别:


全局变量 在程序中只能有一个 self引用了一个对象,添加的各个属性在各个对象间独立存在


8.__str__方法的使用:


__str__方法会在打印对象时执行,一般用于返回对象的描述信息
__str__方法必须有返回值,并且返回值必须是字符串类型
def __str__(self):
        # print('str 执行了')
        return '我是 %s' % self.name
        ```


###### 9.__del__方法的使用:


__del__方法当对象被销毁时执行
__del__方法一般用于验证对象是否被销毁以及释放没有用的资源(关闭文件等)


接面向对象二10.私有属性:


属性前面加两个下划线,只有定义属性的类才能用,


如果在类之外为私有属性赋值,产生一个新的属性。


class Animal(object):
    def __init__(self, ):
        self.__age = 18  #  定义私有属性
11.私有方法:


只能在 类内部用的方法,方法前面加双下划线


私有权限:在属性名和方法名 前面 加上两个下划线 __ 类的私有属性 和 私有方法,都不能通过对象直接访问,但是可以在本类内部访问; 类的私有属性 和 私有方法,都不会被子类继承,子类也无法访问; 私有属性 和 私有方法 往往用来处理类的内部事情,不通过对象处理,起到安全作用。


12.父类,子类


Python3 没写父类的继承,就默认继承 object


Python2 如果没写继承 就是经典类,
写明object 是新式类


13.继承


子类可以继承父类的属性和方法


子类如果要使用父类的私有属性,只能通过共有方法


只要创建子类的对象,就默认执行了那个继承过来的__init__方法


(1)单继承


只继承了一个父类
(2)多继承


当父类有同名方法时 一般用强制调用方法指定调用哪一个父类


可以继承多个父类 也继承了父类的所有属性和方法


如果多个父类中有同名的 属性和方法,则默认使用第一个父类的属性和方法(根据类的魔法属性mro的顺序来查找)


is-a has-a 思维工具、逻辑工具


is-a 表示继承关系 is 前面是子类


has-a 表示属性关系 a 后面 是属性


当父类有同名方法时 一般用强制调用方法指定调用哪一个父类


14.重写


定义 父类不满足子类功能 子类写一个与父类同名的方法


重写后的方法,在调用时会调用自己类里面的方法, 变量 是重新赋值 不算重写方法


重写后 仍需要父类同名方法代码里的数据 内容:


1) 父类名.方法名(self,...) (多继承里用


super(子类名,子类对象“self”).父类方法()
super().父类方法() 只能在 py3中使用
上述方法传参: 如果被重写的方法需要接受参数 在子类调用方法时传参 一般父类需要的 参数 子类也要作为形参


15.__new__方法:


创建对象时执行,能够控制一个类是否要创建对象。


cls:表示调用方法时的类对象。


必须要 return后 才能创建对象。


__new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供


__new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例


    def __new__(cls, *args, **kwargs):
        obj = object.__new__(cls)
        return obj
__new__方法的传参:


用 不定长参数来接受参数 创建对象时的参数解释器会给__new__方法传来,但是__new__方法一般不会使用参数,此时可以使用不定长参数来接收


16. 类对象 与 类属性 实例对象 与 实例属性


类对象:


遇到class关键字 解释器会自动创建类对象


类属性:


当前类的所有实例对象共享一个类属性。


修改类属性:


类名.类属性 = 类属性


实例对象:


类创建出的对象,真实存在的东西


实例属性:


彼此相互独立,互补干扰。


17.实例方法 与 类方法、静态方法


实例方法:


方法的第一个参数是实例对象,叫实例方法


类方法


方法上写@classmethod 方法的第一个参数 一定是接受类对象的


静态方法:


方法上面加上 @staticmethod 方法不接受任何强制性参数


方法的调用 实例方法: 对象名.实例方法()


类方法: 类名.类方法()


静态方法:类名.静态方法()


class Person(object):


    __num = 0  # 私有类属性


    def __init__(self, name):
        self.name = name
        Person.__num += 1


    def drink(self):  # 方法的第一个参数接收的是实例对象,叫做实例方法
        print('%s 能喝' % self.name)


    @classmethod  # 装饰器
    def get_num(cls):  # 类方法,在方法上写一个 @classmethod ,方法的第一个参数必定是类对象
        print('get_num,cls=',cls)
        return cls.__num


    @staticmethod
    def add2num(a, b):  # 静态方法,在方法上写一个 @staticmethod ,方法不需要接收任何强制性的参数
        return a+b


zs = Person('张三')
# zs.drink()


# print(Person.get_num())  # 类方法调用,类名.类方法()
print(Person.add2num(11,22))  # 静态方法调用,类名.静态方法()
18.单例模式:


设计模式:


* 设计模式是由技术大牛们总结出来的一套代码模板
* 即使是技术不够强大的程序员,只要按照模板的套路来写,就可以解决比较复杂的问题
定义:一个类只允许创建一个对象 intance = None


    def __new__(cls, *args, **kwargs):
        # if cls.instance == None:  # 当类属性引用为 None 才创建新对象
        if not cls.instance:  # 当类属性引用为 None 才创建新对象
            cls.instance = object.__new__(cls)
        return cls.instance
19.if判断


class A(object):
    pass


tmp = 0  # False
tmp = 1  # True
tmp = -1  # True
tmp = 0.0  # False
tmp = 0.2  # True
tmp = True
tmp = False
tmp = ''  # False
tmp = ' '  # True
tmp = []  # False
tmp = [11]  # True
tmp = ()  # False
tmp = (11,)  # True
tmp = {}  # False
tmp = {'age': 18}  # True
tmp = set()  # False
tmp = {11}  # True


tmp = A()  # True
tmp = None  # False


if tmp:
    print('True')
else:
    print('False')
20.多态:


多态定义:只要是父类能工作的地方,子类都能工作,并且不同子类的执行效果不同


多态的写法


1)定义父类 提供基本方法 2)定义子类 重写父类方法 3)把子类对象传递给调用者,会产生不同的执行效果


多态的好处:


提高调用的灵活性


面向对象三大特性


封装 , 这是定义类的 准则,单个类 根据 职责 将 属性 和 方法 封装 到一个抽象的 类 中,


继承 , 这是设计类的 技巧,父与子 主要体现是实现代码的 重用,相同的代码不需要重复的编写 子类可以在父类功能上进行重写,扩展类的功能


多态, 不同的 子类对象调用 相同的 父类方法,产生 不同的 执行结果,可以增加代码的外部 调用灵活度,


多态以 继承 和 重写 父类方法 为前提 多态是调用方法的技巧,不会影响到类的内部设计


class Dog(object):


    def work(self):
        pass


class ArmyDog(Dog):


    def work(self):
        print('追击敌人')


class DrugDog(Dog):


    def work(self):
        print('追查毒品')




class XiaoTianQuan(Dog):


    def work1(self):
        print('吃月亮')


class Person(object):


    def work_with_dog(self, dog):  # 多态定义:只要是父类能工作的地方,子类都能工作,并且不同子类的执行效果不同
        dog.work()


# 多态的好处:提高调用的灵活性
# 多态的写法:
# 1.定义父类,提供基本方法
# 2.定义子类,重写父类方法
# 3.把子类对象传递给调用者,会产生不同的执行效果


army = ArmyDog()
drug = DrugDog()
xiaotian = XiaoTianQuan()


p = Person()
p.work_with_dog(army)
p.work_with_dog(drug)
p.work_with_dog(xiaotian)
21.异常与捕获异常:


异常:让程序崩溃的代码 捕获异常: try...except...


try:
    print('try start')
    print(a)
    print('try end')  # 即使捕捉了异常,异常之后的代码也不会执行


except:  # 当出现异常的时候会执行 except
    print('捕捉到异常,正在进行修复')


1)捕获多个异常和指定异常


根据工作经验和 解释器给出的异常名


多个异常捕捉 用同样方法解决 可以用元组 里写异常名


except (NameError, FileNotFoundError):  # 捕捉多个指定异常,except (异常名称1, 异常名称2, ...)
    print('捕捉到资源不存在异常,正在进行修复')


except ValueError:
    print('类型转换发生异常,正在进行修复')
2)捕捉任意异常


查找父类:__mro__方法,能通过查找顺序来查找一个类的父类


except Exception:是py里大部分异常处理的父类 (可以设置别名) except 后面不写 :则是就收所有的异常,(不可以使用别名)


except Exception:  # Exception 是Python里大部分异常的父类,可以捕捉绝大部分的异常
    print('Exception 捕捉到异常')


except:  # except 后不指定异常,可以捕捉任意异常
    print('捕捉到未知异常,正在修复~~')
3)异常的别名:


except 异常名 as 变量: 打印变量 时就会打印出异常信息。 except (多个异常名称)as 变量: 打印变量只会打印出一个异常信息,(因为一次只捕获一次)


except ValueError as exp:  # except 异常名称 as 别名
    print('类型转换发生异常,正在进行修复, exp=', exp)


except (NameError, FileNotFoundError) as exp:  # except (异常1,异常2,..) as 别名
    print('捕捉到资源不存在异常,正在进行修复, exp=', exp)


except Exception as exp:  # Exception 可以设置别名
    print('Exception 捕捉到异常, exp=', exp)


except:  # except 不能设置别名
    print('捕捉到未知异常,正在修复~~')
4)try ...except ... else ... finally.


else :在没有异常时会执行 finally 无论有没有异常都会执行


except ValueError as exp:
    print('类型转换发生异常,正在进行修复, exp=', exp)


else:
    print('else 当没有异常发生时才会执行')


finally:
    print('finally 不管有没有异常都会执行')
5)异常的传递:


被调用的函数发生异常,会把异常传回调用者。


def func1():
    print('func1 --- start')
    x = 1 / 0  # 被调用的函数发生异常,会把异常传递给调用者
    print('func1 --- end')




def func2():
    print('func2 --- start')
    try:
        func1()
    except Exception:
        print('捕捉到异常')
    print('func2 --- end')


func2()
print('程序结束')
6)自定义异常:


自定义异常必须继承Exception
通过raise 执行异常
李龙 21:31:49
类与对象


1.定义类:


类是具有相同属性和行为的事物的统称


class Cat:
    """
    函数体
    """
    pass
2.创建对象:


某一个具体事物的存在 现实生活中看得见 摸得着


对象名1 = 类名()


对象名2 = 类名()
多个对象之间互不干扰 相互独立


3.为对象添加属性


tom = Cat()
tom.name = '汤姆'
print(tom.name)
4.在方法内获取对象的属性:


class Cat:
    """
    函数体
    """
    def eat(self):
        print('%s在喝水' % tom.name)
tom = Cat()
tom.name = '汤姆'
tom.eat()
print(tom.name)
5.self的作用


self 代表了当前调用方法的对象


6.__init__的使用


使用: 方法在创建对象后执行 一般用来初始化对象的共有属性


__init__传参的作用: 可以在self 后面定义需要用到的参数 init需要的参数,在创建对象时传递


def __init__(self,new_name):
        """
        魔法方法:在恰当的时机会自动执行的方法
        
        """
        print('init 执行了')
        self.name = new_name
class Cat:


    def __init__(self, new_name):
        self.name = new_name
        
xiaobai = Cat('伊丽莎白')


7.全局变量与属性的区别:


全局变量 在程序中只能有一个 self引用了一个对象,添加的各个属性在各个对象间独立存在


8.__str__方法的使用:


__str__方法会在打印对象时执行,一般用于返回对象的描述信息
__str__方法必须有返回值,并且返回值必须是字符串类型
def __str__(self):
        # print('str 执行了')
        return '我是 %s' % self.name
        ```


###### 9.__del__方法的使用:


__del__方法当对象被销毁时执行
__del__方法一般用于验证对象是否被销毁以及释放没有用的资源(关闭文件等)


接面向对象二10.私有属性:


属性前面加两个下划线,只有定义属性的类才能用,


如果在类之外为私有属性赋值,产生一个新的属性。


class Animal(object):
    def __init__(self, ):
        self.__age = 18  #  定义私有属性
11.私有方法:


只能在 类内部用的方法,方法前面加双下划线


私有权限:在属性名和方法名 前面 加上两个下划线 __ 类的私有属性 和 私有方法,都不能通过对象直接访问,但是可以在本类内部访问; 类的私有属性 和 私有方法,都不会被子类继承,子类也无法访问; 私有属性 和 私有方法 往往用来处理类的内部事情,不通过对象处理,起到安全作用。


12.父类,子类


Python3 没写父类的继承,就默认继承 object


Python2 如果没写继承 就是经典类,
写明object 是新式类


13.继承


子类可以继承父类的属性和方法


子类如果要使用父类的私有属性,只能通过共有方法


只要创建子类的对象,就默认执行了那个继承过来的__init__方法


(1)单继承


只继承了一个父类
(2)多继承


当父类有同名方法时 一般用强制调用方法指定调用哪一个父类


可以继承多个父类 也继承了父类的所有属性和方法


如果多个父类中有同名的 属性和方法,则默认使用第一个父类的属性和方法(根据类的魔法属性mro的顺序来查找)


is-a has-a 思维工具、逻辑工具


is-a 表示继承关系 is 前面是子类


has-a 表示属性关系 a 后面 是属性


当父类有同名方法时 一般用强制调用方法指定调用哪一个父类


14.重写


定义 父类不满足子类功能 子类写一个与父类同名的方法


重写后的方法,在调用时会调用自己类里面的方法, 变量 是重新赋值 不算重写方法


重写后 仍需要父类同名方法代码里的数据 内容:


1) 父类名.方法名(self,...) (多继承里用


super(子类名,子类对象“self”).父类方法()
super().父类方法() 只能在 py3中使用
上述方法传参: 如果被重写的方法需要接受参数 在子类调用方法时传参 一般父类需要的 参数 子类也要作为形参


15.__new__方法:


创建对象时执行,能够控制一个类是否要创建对象。


cls:表示调用方法时的类对象。


必须要 return后 才能创建对象。


__new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供


__new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例


    def __new__(cls, *args, **kwargs):
        obj = object.__new__(cls)
        return obj
__new__方法的传参:


用 不定长参数来接受参数 创建对象时的参数解释器会给__new__方法传来,但是__new__方法一般不会使用参数,此时可以使用不定长参数来接收


16. 类对象 与 类属性 实例对象 与 实例属性


类对象:


遇到class关键字 解释器会自动创建类对象


类属性:


当前类的所有实例对象共享一个类属性。


修改类属性:


类名.类属性 = 类属性


实例对象:


类创建出的对象,真实存在的东西


实例属性:


彼此相互独立,互补干扰。


17.实例方法 与 类方法、静态方法


实例方法:


方法的第一个参数是实例对象,叫实例方法


类方法


方法上写@classmethod 方法的第一个参数 一定是接受类对象的


静态方法:


方法上面加上 @staticmethod 方法不接受任何强制性参数


方法的调用 实例方法: 对象名.实例方法()


类方法: 类名.类方法()


静态方法:类名.静态方法()


class Person(object):


    __num = 0  # 私有类属性


    def __init__(self, name):
        self.name = name
        Person.__num += 1


    def drink(self):  # 方法的第一个参数接收的是实例对象,叫做实例方法
        print('%s 能喝' % self.name)


    @classmethod  # 装饰器
    def get_num(cls):  # 类方法,在方法上写一个 @classmethod ,方法的第一个参数必定是类对象
        print('get_num,cls=',cls)
        return cls.__num


    @staticmethod
    def add2num(a, b):  # 静态方法,在方法上写一个 @staticmethod ,方法不需要接收任何强制性的参数
        return a+b


zs = Person('张三')
# zs.drink()


# print(Person.get_num())  # 类方法调用,类名.类方法()
print(Person.add2num(11,22))  # 静态方法调用,类名.静态方法()
18.单例模式:


设计模式:


* 设计模式是由技术大牛们总结出来的一套代码模板
* 即使是技术不够强大的程序员,只要按照模板的套路来写,就可以解决比较复杂的问题
定义:一个类只允许创建一个对象 intance = None


    def __new__(cls, *args, **kwargs):
        # if cls.instance == None:  # 当类属性引用为 None 才创建新对象
        if not cls.instance:  # 当类属性引用为 None 才创建新对象
            cls.instance = object.__new__(cls)
        return cls.instance
19.if判断


class A(object):
    pass


tmp = 0  # False
tmp = 1  # True
tmp = -1  # True
tmp = 0.0  # False
tmp = 0.2  # True
tmp = True
tmp = False
tmp = ''  # False
tmp = ' '  # True
tmp = []  # False
tmp = [11]  # True
tmp = ()  # False
tmp = (11,)  # True
tmp = {}  # False
tmp = {'age': 18}  # True
tmp = set()  # False
tmp = {11}  # True


tmp = A()  # True
tmp = None  # False


if tmp:
    print('True')
else:
    print('False')
20.多态:


多态定义:只要是父类能工作的地方,子类都能工作,并且不同子类的执行效果不同


多态的写法


1)定义父类 提供基本方法 2)定义子类 重写父类方法 3)把子类对象传递给调用者,会产生不同的执行效果


多态的好处:


提高调用的灵活性


面向对象三大特性


封装 , 这是定义类的 准则,单个类 根据 职责 将 属性 和 方法 封装 到一个抽象的 类 中,


继承 , 这是设计类的 技巧,父与子 主要体现是实现代码的 重用,相同的代码不需要重复的编写 子类可以在父类功能上进行重写,扩展类的功能


多态, 不同的 子类对象调用 相同的 父类方法,产生 不同的 执行结果,可以增加代码的外部 调用灵活度,


多态以 继承 和 重写 父类方法 为前提 多态是调用方法的技巧,不会影响到类的内部设计


class Dog(object):


    def work(self):
        pass


class ArmyDog(Dog):


    def work(self):
        print('追击敌人')


class DrugDog(Dog):


    def work(self):
        print('追查毒品')




class XiaoTianQuan(Dog):


    def work1(self):
        print('吃月亮')


class Person(object):


    def work_with_dog(self, dog):  # 多态定义:只要是父类能工作的地方,子类都能工作,并且不同子类的执行效果不同
        dog.work()


# 多态的好处:提高调用的灵活性
# 多态的写法:
# 1.定义父类,提供基本方法
# 2.定义子类,重写父类方法
# 3.把子类对象传递给调用者,会产生不同的执行效果


army = ArmyDog()
drug = DrugDog()
xiaotian = XiaoTianQuan()


p = Person()
p.work_with_dog(army)
p.work_with_dog(drug)
p.work_with_dog(xiaotian)
21.异常与捕获异常:


异常:让程序崩溃的代码 捕获异常: try...except...


try:
    print('try start')
    print(a)
    print('try end')  # 即使捕捉了异常,异常之后的代码也不会执行


except:  # 当出现异常的时候会执行 except
    print('捕捉到异常,正在进行修复')


1)捕获多个异常和指定异常


根据工作经验和 解释器给出的异常名


多个异常捕捉 用同样方法解决 可以用元组 里写异常名


except (NameError, FileNotFoundError):  # 捕捉多个指定异常,except (异常名称1, 异常名称2, ...)
    print('捕捉到资源不存在异常,正在进行修复')


except ValueError:
    print('类型转换发生异常,正在进行修复')
2)捕捉任意异常


查找父类:__mro__方法,能通过查找顺序来查找一个类的父类


except Exception:是py里大部分异常处理的父类 (可以设置别名) except 后面不写 :则是就收所有的异常,(不可以使用别名)


except Exception:  # Exception 是Python里大部分异常的父类,可以捕捉绝大部分的异常
    print('Exception 捕捉到异常')


except:  # except 后不指定异常,可以捕捉任意异常
    print('捕捉到未知异常,正在修复~~')
3)异常的别名:


except 异常名 as 变量: 打印变量 时就会打印出异常信息。 except (多个异常名称)as 变量: 打印变量只会打印出一个异常信息,(因为一次只捕获一次)


except ValueError as exp:  # except 异常名称 as 别名
    print('类型转换发生异常,正在进行修复, exp=', exp)


except (NameError, FileNotFoundError) as exp:  # except (异常1,异常2,..) as 别名
    print('捕捉到资源不存在异常,正在进行修复, exp=', exp)


except Exception as exp:  # Exception 可以设置别名
    print('Exception 捕捉到异常, exp=', exp)


except:  # except 不能设置别名
    print('捕捉到未知异常,正在修复~~')
4)try ...except ... else ... finally.


else :在没有异常时会执行 finally 无论有没有异常都会执行


except ValueError as exp:
    print('类型转换发生异常,正在进行修复, exp=', exp)


else:
    print('else 当没有异常发生时才会执行')


finally:
    print('finally 不管有没有异常都会执行')
5)异常的传递:


被调用的函数发生异常,会把异常传回调用者。


def func1():
    print('func1 --- start')
    x = 1 / 0  # 被调用的函数发生异常,会把异常传递给调用者
    print('func1 --- end')




def func2():
    print('func2 --- start')
    try:
        func1()
    except Exception:
        print('捕捉到异常')
    print('func2 --- end')


func2()
print('程序结束')
6)自定义异常:


自定义异常必须继承Exception
通过raise 执行异常


class MyError(Exception):


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


    def __str__(self):  # __str__ 方法返回的数据,就是异常的描述信息
        return '年龄不能小于 0,您输入的是 %d' % self.age




try:
    age = int(input('请输入您的年龄:'


class MyError(Exception):


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


    def __str__(self):  # __str__ 方法返回的数据,就是异常的描述信息
        return '年龄不能小于 0,您输入的是 %d' % self.age




try:
    age = int(input('请输入您的年龄:'




作者: wuqiong    时间: 2018-6-25 10:45

作者: 不二晨    时间: 2018-6-25 14:01
可以说很专业了
作者: 小影姐姐    时间: 2018-6-26 09:24

作者: 梦缠绕的时候    时间: 2018-6-26 15:23
加油加油
作者: 不二晨    时间: 2018-6-27 14:47
上海分校-小影 发表于 2018-6-26 09:24

非常的厉害吧
作者: 吴琼老师    时间: 2018-6-28 14:53





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2