面向对象中关于元类的介绍与异常处理

一、异常处理

1. 什么是异常处理     异常是错误发生的信号,一旦程序出错就会产生一个异常,如果该异常     没有被应用程序处理,那么该异常就会抛出来,程序的执行也随之终止

异常包含三个部分:         1. traceback异常的追踪信息         2. 异常的类型         3. 异常的信息

错误分为两大类:         1. 语法上的错误:在程序运行前就应该立即修正         2. 逻辑上的错误

2. 为何要异常处理

避免程序因为异常而崩溃,所以在应用程序中应该对异常进行处理,从而增强程序的健壮性

3. 如何异常处理

try:
    代码1
    代码2
    代码3
    ......
except NameError:
    当抛出的异常是NameError时执行的子代码块
except ....:
    pass
except ...:
    pass
else:
    pass
finally:
    pass

3.1常见的逻辑错误导致的异常

 print(‘adsfsadf‘

 age=input(‘>>: ‘).strip()
 print(age > 10) #TypeError

 for i in 10: #TypeError
    pass

 import os
 os.xxx #AttributeError

 1 / 0 #ZeroDivisionError:

 print(‘=====1‘)
 print(‘=====2‘)
 print(‘=====3‘)
 l=[1,2,3]# l[1000] #IndexError
 print(‘=====4‘)
 d={‘x‘:1,‘y‘:2}
 d[‘z‘] #KeyError
 print(‘=====5‘)

3.2异常处理
  3.2.1异常处理的单分支

try:
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     d[‘z‘]  # KeyError
     print(‘=====4‘)
     l = [1, 2, 3]
     l[1000]  # IndexError
     print(‘=====5‘)
except IndexError:
    print(‘IndexError‘)
print(‘other code‘)

3.2.2异常处理的多分支

 try:
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     d[‘z‘]  # KeyError
     print(‘=====4‘)
     l = [1, 2, 3]
     l[1000]  # IndexError
        print(‘=====5‘)
 except KeyError as e:
     print(‘KeyError‘,e)
 except IndexError as e:
     print(‘IndexError‘,e)

 print(‘other code‘)

 try:
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     # d[‘z‘]  # KeyError
     print(‘=====4‘)
     l = [1, 2, 3]
     l[1000]  # IndexError
     print(‘=====5‘)
 except (KeyError,IndexError) as e:
     print(e)
 print(‘other code‘)

3.2.3万能异常类型Exception:可以匹配任意类型的异常

 try:
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     # d[‘z‘]  # KeyError
     # xxx
     print(‘=====4‘)
     l = [1, 2, 3]
     l[1000]  # IndexError
     print(‘=====5‘)
 except IndexError as e:
     print(‘IndexError:‘, e)
 except KeyError as e:
    print(‘KeyError:‘, e)
 except Exception as e:
     print(‘Exception:‘,e)

 print(‘other code‘)

 try... else...
 try:
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     # d[‘z‘]  # KeyError
     # xxx
     print(‘=====4‘)
     l = [1, 2, 3]
     # l[1000]  # IndexError
     print(‘=====5‘)
 except IndexError as e:
     print(‘IndexError:‘, e)
 except KeyError as e:
     print(‘KeyError:‘, e)
 except Exception as e:
     print(‘Exception:‘,e)
 else:
     print(‘else必须放到后面,else的子代码块会在被检测的代码没有异常的情况下执行‘)
 # print(‘other code‘)

 try... finally...
 try:
     f=open(‘a.txt‘,‘w‘)
     print(‘=====1‘)
     print(‘=====2‘)
     print(‘=====3‘)
     d = {‘x‘: 1, ‘y‘: 2}
     # d[‘z‘]  # KeyError
     # xxx
     ‘xx‘ > 10
     print(‘=====4‘)
     l = [1, 2, 3]
     # l[1000]  # IndexError
     print(‘=====5‘)

 except IndexError as e:
     print(‘IndexError:‘, e)
 except KeyError as e:
     print(‘KeyError:‘, e)
 # except Exception as e:
 #     print(‘Exception:‘,e)
 else:
     print(‘else必须放到后面,else的子代码块会在被检测的代码没有异常的情况下执行‘)
 finally:
     print(‘无论被检测的代码有没有异常都会执行‘)
     f.close()

4几个异常的用法
4.1主动触发异常

class People:
    def __init__(self,name,age):
        self.__name=name
        self.__age=age
    def tell_info(self):
        print(self.__name,self.__age)
    def set_info(self,name,age):
        if not isinstance(name,str):
            raise TypeError(‘name必须是字符串类型str‘)
        if not isinstance(age,int):
            raise TypeError(‘age必须是整型int‘)
        self.__name=name
        self.__age=age
obj=People(‘liu‘,18)
obj.set_info(‘egon‘,2.0)
obj.tell_info()

4.2自定义异常(了解)

class Mymeta(BaseException):
    def __init__(self,name):
        self.name=name
    def __str__(self):
        return ‘%s‘%self.name
raise Mymeta(‘LIU‘)

4.3断言(了解)

print(‘上半部分,生产数据‘)
l=[1,2,3]
# if len(l)!=5:
#     raise TypeError
assert len(l)==5
print(‘下半部分,处理数据‘)

二、元类的介绍
1、什么是元类?

源自一句话:在Python中一切皆为对象,二对象都是由类实例化得到的

class OldboyTeacher:
    def __init__(self,name,age,sex):
        self.name=name
        self.age=age
        self.sex=sex

    def score(self):
        print(‘%s is scoring‘ %self.name)

tea1=OldboyTeacher(‘egon‘,18,‘male‘)
print(type(tea1))
print(type(OldboyTeacher))
对象tea1是调用OldboyTeacher类得到的,如果说一切皆对象,那么OldboyTeacher也是一个对象,只要是对象 都是调用一个类实例化得到的,即OldboyTeacher=元类(...),内置的元类是type
关系: 1. 调用元类---->自定义的类 2. 调用自定义的类---->自定义的对象 
class关键字创建自定义类的底层的工作原理,分为四步 1. 先拿到类名:‘OldboyTeacher‘2. 再拿到类的基类们:(object,) 3. 然后拿到类的名称空间???(执行类体代码,将产生的名字放到类的名称空间也就是一个字典里,补充exec)4. 调用元类实例化得到自定义的类: OldboyTeacher=type(‘OldboyTeacher‘,(object,),{...})
class OldboyTeacher: #OldboyTeacher=type(...)
    school = ‘Oldboy‘
    def __init__(self,name,age,sex):
        self.name=name
        self.age=age
        self.sex=sex

    def score(self):
        print(‘%s is scoring‘ %self.name)
print(OldboyTeacher)
注意:
自定义类的三个关键组成部分:1. 类名 2. 类的基类们3. 类的名称空间
不依赖class关键字创建一个自定义类
# 1. 拿到类名
class_name=‘OldboyTeacher‘
#2. 拿到类的基类们:(object,)
class_bases=(object,)
#3. 拿到类的名称空间
class_dic={}
class_body="""

"""
exec(class_body,{},class_dic)
print(class_dic)
#4. 调用type得到自定义的类
OldboyTeacher=type(class_name,class_bases,class_dic)

print(OldboyTeacher)
# print(OldboyTeacher.school)
# print(OldboyTeacher.score)

tea1=OldboyTeacher(‘egon‘,18,‘male‘)
print(tea1.__dict__)
3、自定义元类来控制自定义类的产生于调用过程
# 模板
# class Mymeta(type): #但凡继承了type的类才能称之为自定义的元类,否则就是只是一个普通的类
#     def __init__(self,class_name,class_bases,class_dic):
#         print(self)
#         print(class_name)
#         print(class_bases)
#         print(class_dic)
#
# class OldboyTeacher(object,metaclass=Mymeta): #OldboyTeacher=Mymeta(‘OldboyTeacher‘,(object,),{...})
#     school = ‘Oldboy‘
#
#     def __init__(self,name,age,sex):
#         self.name=name
#         self.age=age
#         self.sex=sex
#
#     def score(self):
#         print(‘%s is scoring‘ %self.name)

# 控类的产生
# 1.类名必须用驼峰体
# 2.类体必须有文档注释,且文档注释不能为空
class Mymeta(type): #但凡继承了type的类才能称之为自定义的元类,否则就是只是一个普通的类
    def __init__(self,class_name,class_bases,class_dic):
        if class_name.islower():
            raise TypeError(‘类名必须使用驼峰体‘)

        doc=class_dic.get(‘__doc__‘)
        if doc is None or len(doc) == 0 or len(doc.strip(‘\n ‘)) == 0:
            raise TypeError(‘类体中必须有文档注释,且文档注释不能为空‘)

class OldboyTeacher(object,metaclass=Mymeta): #OldboyTeacher=Mymeta(‘OldboyTeacher‘,(object,),{...})
    school = ‘Oldboy‘

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

    def score(self):
        print(‘%s is scoring‘ %self.name)

print(OldboyTeacher.__dict__)
补充:属性查找
class Mymeta(type):  # 但凡继承了type的类才能称之为自定义的元类,否则就是只是一个普通的类
    # n=444
    def __call__(self, *args, **kwargs): #self=OldboyTeacher这个类
        # 1. 先产生一个空对象
        tea_obj = self.__new__(self)  # tea_obj是OldboyTeacher这个类的对象
        # print(self.__new__ is object.__new__)
        # tea_obj=object.__new__(self)

        # 2. 执行__init__方法,完成对象的初始属性操作
        self.__init__(tea_obj, *args, **kwargs)
        # 3. 返回初始化好的那个对象
        return tea_obj

class Bar:
    # n = 33
    pass

class Foo(Bar):
    # n = 222
    pass

class OldboyTeacher(Foo, metaclass=Mymeta):  # OldboyTeacher=Mymeta(‘OldboyTeacher‘,(object,),{...})
    # n = 111
    school = ‘Oldboy‘

    def __init__(self, name, age, sex):
        self.name = name #None.name=‘egon‘
        self.age = age
        self.sex = sex

    def score(self):
        print(‘%s is scoring‘ % self.name)

    def __new__(cls, *args, **kwargs):
        # print(‘=====>‘)
        return super().__new__(cls)

tea1 = OldboyTeacher(‘egon‘, 18, ‘male‘)
# print(tea1)
print(tea1.__dict__)

# print(OldboyTeacher.n)

# print(object.__new__)



 




原文地址:https://www.cnblogs.com/ageliu/p/9545324.html

时间: 2024-10-06 06:07:08

面向对象中关于元类的介绍与异常处理的相关文章

深刻理解Python中的元类(metaclass)以及元类实现单例模式

深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例模式的那一节有些疑惑.因此花了几天时间研究下元类这个概念.通过学习元类,我对python的面向对象有了更加深入的了解.这里将一篇写的非常好的文章基本照搬过来吧,这是一篇在Stack overflow上很热的帖子,我看http://blog.jobbole.com/21351/这篇博客对其进行了翻译. 一.理解

Python中的元类

从前面"Python对象"文章中了解到,在Python中一切都是对象,类可以创建实例对象,但是类本身也是对象. class C(object): pass c = C() print c.__class__ print C.__class__ 代码中,通过"__class__"属性来查看对象的类型,对于类C对象本身,它的类型是type. 由于类也是对象,所以就可以在运行时动态的创建类,那么这时候就要用到内建函数type了. 再看type 从前面的文章了解到,可以通过

深刻理解Python中的元类(metaclass)

译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了985点的赞同点数,更有人评论说这段回复应该加入到Python的官方文档中去.而e-satis同学本人在Stack Overflow中的声望积分也高达6

Python中的元类(metaclass)

推荐+收藏:深刻理解Python中的元类(metaclass) 做一些笔记学习学习: 在大多数编程语言中,类就是用来描述如何生成一个对象的代码段,在Python中类也是一个对象,这个(类)对象自身拥有创建对象(类实例)的能力. 因为它的本质是一个对象: 可以将它赋值给一个变量 可以拷贝它 增加属性 作为参数进行传递 可以在运行时动态地创建他们,可以在函数中创建类,只需要使用class关键字即可 当使用class关键字的时候,Python解释器会自动地创建这个对象,Python还提供了手动处理的方

深刻理解Python中的元类metaclass(转)

本文由 伯乐在线 - bigship 翻译 英文出处:stackoverflow 译文:http://blog.jobbole.com/21351/ 译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了9

深刻理解Python中的元类

译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了985点的赞同点数,更有人评论说这段回复应该加入到Python的官方文档中去.而e-satis同学本人在Stack Overflow中的声望积分也高达6

python——深刻理解Python中的元类(metaclass)

译注:这是一篇在Stack overflow上 很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子 和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了985点的赞同点数,更有人评论说这段 回复应该加入到Python的官方文档中去.而e-satis同学本人在Stack Overflow中的声望积分也

[转] 深刻理解Python中的元类(metaclass)

非常详细的一篇深入讲解Python中metaclass的文章,感谢伯乐在线-bigship翻译及作者,转载收藏. 本文由 伯乐在线 - bigship 翻译.未经许可,禁止转载!英文出处:stackoverflow.欢迎加入翻译组. 译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及

深入理解Python中的元类(metaclass)

注:本文原稿来自stackoverflow,原文链接,目前已收获5k高赞. 一.类也是对象 在理解元类之前,你需要先掌握Python中的类.Python中类的概念借鉴于Smalltalk,这显得有些奇特.在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.在Python中这一点仍然成立: >>> class ObjectCreator(object): … pass … >>> my_object = ObjectCreator() >>>