零碎知识点:装饰器、MD5、异常、日志

装饰器

首先说一下装饰器

import time
def timer(func):
    def inner():
        start = time.time()
        func()
        print(time.time() - start)
    return inner

@timer   #==> func1 = timer(func1)
def func1():
    print(‘in func1‘)

func1()

装饰器

上边一段代码就是简单的装饰器,,装饰器的作用是在不改变原函数的内容以及调用方式返回值的情况下,给函数添加功能(个人的理解),具体的书写方式就是先写外部函数,然后写内部函数,在内部函数内执行想要添加功能的原函数,在原函数执行前后可以进行功能的添加。

以上虽然是一个装饰器但还是不完整,如果原函数是带参数和返回值的,我们就改变了原函数的调用方式和返回值,这就违背了装饰器的初衷,应该将原函数的参数传递进去

import time
def timer(func):
    def inner(*args,**kwargs):
        start = time.time()
        re = func(*args,**kwargs)
        print(time.time() - start)
        return re
    return inner

@timer   #==> func1 = timer(func1)
def func1(a,b):
    print(‘in func1‘)

@timer   #==> func2 = timer(func2)
def func2(a):
    print(‘in func2 and get a:%s‘%(a))
    return ‘fun2 over‘

func1(‘aaaaaa‘,‘bbbbbb‘)
print(func2(‘aaaaaa‘))

hold住各种传参方式

import time
def timer(func):
    def inner(*args,**kwargs):
        start = time.time()
        re = func(*args,**kwargs)
        print(time.time() - start)
        return re
    return inner

@timer   #==> func2 = timer(func2)
def func2(a):
    print(‘in func2 and get a:%s‘%(a))
    return ‘fun2 over‘

func2(‘aaaaaa‘)
print(func2(‘aaaaaa‘))

带返回值的装饰器

刚刚那个装饰器已经非常完美了,但是正常我们情况下查看函数的一些信息的方法在此处都会失效

def index():
    ‘‘‘这是一个主页信息‘‘‘
    print(‘from index‘)

print(index.__doc__)    #查看函数注释的方法
print(index.__name__)   #查看函数名的方法

查看函数信息的一些方法

为了不让他们失效,我们还要在装饰器上加上一点来完善它:

from functools import wraps

def deco(func):
    @wraps(func) #加在最内层函数正上方
    def wrapper(*args,**kwargs):
        return func(*args,**kwargs)
    return wrapper

@deco
def index():
    ‘‘‘哈哈哈哈‘‘‘
    print(‘from index‘)

print(index.__doc__)
print(index.__name__)

利用wraps

装饰器的主要功能和装饰器的固定结构

装饰器的主要功能:

在不改变函数调用方式的基础上在函数的前、后添加功能。

装饰器的固定格式:

from functools import wraps

def deco(func):
    @wraps(func) #加在最内层函数正上方
    def wrapper(*args,**kwargs):
        return func(*args,**kwargs)
    return wrapper

带参数的装饰器

假如你有成千上万个函数使用了一个装饰器,现在你想把这些装饰器都取消掉,你要怎么做?

一个一个的取消掉? 没日没夜忙活3天。。。

过两天你领导想通了,再让你加上。。。

def outer(flag):
    def timer(func):
        def inner(*args,**kwargs):
            if flag:
                print(‘‘‘执行函数之前要做的‘‘‘)
            re = func(*args,**kwargs)
            if flag:
                print(‘‘‘执行函数之后要做的‘‘‘)
            return re
        return inner
    return timer

@outer(False)
def func():
    print(111)

func()

带参数的装饰器

多个装饰器装饰同一个函数

有些时候,我们也会用到多个装饰器装饰同一个函数的情况。

这种情况下装饰顺序按照就近原则

MD5

MD5是一种不可逆的加密算法. 它是可靠的. 并且安全的. 在python中我们不需要手写
这一套算法. 只需要引入一个叫hashlib的模块就能搞定MD5的加密工作

import hashlibobj = hashlib.md5()
obj.update("alex".encode("utf-8")) # 加密的必须是字节
miwen = obj.hexdigest()
print(miwen) # 534b44a19bf18d20b71ecc4eb77c572f

那这样的密文安全么? 其实是不安全的. 当我们用这样的密文去找一个所谓的MD5解密
工具. 是有可能解密成功的.

这就尴尬了. MD5不是不可逆么? 注意. 这里有一个叫撞库的问题. 就是. 由于MD5的原
始算法已经存在很久了. 那就有一些人用一些简单的排列组合来计算MD5. 然后当出现相同
的MD5密文的时候就很容易反推出原来的数据是什么. 所以并不是MD5可逆, 而是有些别有
用心的人把MD5的常见数据已经算完并保留起来了.
那如何应对呢? 加盐就星了. 在使?MD5的时候. 给函数的参数传递一个byte即可

import hashlib
obj = hashlib.md5(b"fjlksajflkjasfsalwer123dfskjf") # 加盐
obj.update("alex".encode("utf-8")) # 加密的必须是字节
miwen = obj.hexdigest()
print(miwen) # 99fca4b872fa901aac30c3e952ca786d

import hashlib
def my_md5(s):
obj = hashlib.md5(b"fjlksajflkjasfsalwer123dfskjf")
obj.update(s.encode("utf-8")) # 加密的必须是字节
miwen = obj.hexdigest()
return miwen
# alex: 99fca4b872fa901aac30c3e952ca786d
username = input("请输??户名:")
password = input("请输?密码:")
# 数据存储的时候.
# username: my_md5(password)# 假设现在的?户名和密码分别是
# wusir: 99fca4b872fa901aac30c3e952ca786d ==> wusir: alex
# ?户登录
if username == "wusir" and my_md5(password) ==
"99fca4b872fa901aac30c3e952ca786d":
print("成功")
else:
print("失败")

MD5的简单应用

异常

异常处理

首先, 我们先说?下, 什么是异常? 异常是程序在运行过程中产生的错误. 就好比. 你在
回家路上突然天塌了. 那这个就属于?个异常. 总之就是不正常. 那如果程序出现了异常. 怎
么处理呢? 在之前的学习中我们已经写过类似的代码了.
我们先制造?个错误. 来看看异常长什么样.

def chu(a, b):
return a/b
ret = chu(10, 0)
print(ret)
结果:
Traceback (most recent call last):
File "/Users/sylar/PycharmProjects/oldboy/?向对象/day05.py", line 100, in
<module>
ret = chu(10, 0)
File "/Users/sylar/PycharmProjects/oldboy/?向对象/day05.py", line 98, in
chu
return a/b
ZeroDivisionError: division by zero

异常

什么错误呢. 除法中除数不能是0. 那如果真的出了这个错. 你把这一堆信息抛给客户
么? 肯定不能. 那如何处理呢?

def chu(a, b):
return a/b
try:
ret = chu(10, 0)
print(ret)
except Exception as e:
print("除数不能是0")
结果:
除数不能是0

异常处理

那try...except是什么意思呢? 尝试着运行xxxxx代码. 出现了错误. 就执行except后面的
代码. 在这个过程中. 当代码出现错误的时候. 系统会产生一个异常对象. 然后这个异常会向
外抛. 被except拦截. 并把接收到的异常对象赋值给e. 那这里的e就是异常对象. 那这里的
Exception是什么? Exception是所有异常的基类, 也就是异常的跟. 换句话说. 所有的错误都
是Exception的子类对象. 我们看到的ZeroDivisionError 其实就是Exception的?类. 那这样
写好像有点儿问题撒. Exception表示所有的错误. 太笼统了. 所有的错误都会被认为是Exception.
当程序中出现多种错误的时候, 就不好分类了, 最好是出什么异常就用什么来处理. 这样就更加合理了.
所以在try...execpt语句中. 还可以写更多的except

try:
  print("各种操作....")
except ZeroDivisionError as e:
  print("除数不能是0")
except FileNotFoundError as e:
  print("?件不存在")
except Exception as e:
  print("其他错误")

此时. 程序运行过程中. 如果出现了ZeroDivisionError就会被第一个except捕获. 如果出
现了FileNotFountError就会被第二个except捕获. 如果都不是这两个异常. 那就会被最后的
Exception捕获. 总之最后的Exception就是我们异常处理的最后一个守门员. 这时我们最常
用的一套写法. 接下来. 给出一个完整的异常处理写法(语法):

try:‘‘‘操作‘‘‘
except Exception as e:
‘‘‘异常的?类,可以捕获所有的异常‘‘‘
else:
‘‘‘保护不抛出异常的代码, 当try中?异常的时候执?‘‘‘
finally:
‘‘‘最后总是要执?我‘‘‘

解读: 程序先执行操作, 然后如果出错了会走except中的代码. 如果不出错, 执?else中
的代码. 不论处不出错. 最后都要执?finally中的语句. 一般我们用try...except就够用了. 顶多
加上finally. finally一般用来作为收尾工作.
上面是处理异常. 我们在执行代码的过程中如果出现了一些条件上的不对等. 根本不符
合我的代码逻辑. 比如. 参数. 我要求你传递一个数字. 你非得传递一个字符串. 那对不起. 我
没办法帮你处理. 那如何通知你呢? 两个方案.
方案一. 直接返回即可. 我不管你还不行么?
方案二. 抛出一个异常. 告诉你. 我不好惹. 乖乖的听话.
第一种方案是我们之前写代码经常用到的方案. 但这种方案并不够好. 无法起到警示作用. 所
以. 以后的代码中如果出现了类似的问题. 直接抛一个错误出去. 那怎么抛呢? 我们要用到
raise关键字

def add(a, b):
‘‘‘
给我传递两个整数. 我帮你计算两个数的和
:param :param a:
:param :param b:
:return :return:
‘‘‘
if not type(a) == int and not type(b) == int:
# 当程序运?到这句话的时候. 整个函数的调?会被中断. 并向外抛出?个异常.
raise Exception("不是整数, 朕不能帮你搞定这么复杂的运算.")
return a + b
# 如果调??不处理异常. 那产?的错误将会继续向外抛. 最后就抛给了?户
# add("你好", "我叫赛利亚")
# 如果调??处理了异常. 那么错误就不会丢给?户. 程序也能正常进?
try:
add("胡辣汤", "滋滋冒油的?腰?")
except Exception as e:
print("报错了. ??处理去吧")

当程序运行到raise. 程序会被中断. 并实例化后面的异常对象. 抛给调用方. 如果调用方
不处理. 则会把错误继续向上抛出. 最终抛给用户. 如果调用方处理了异常. 那程序可以正常
的进行执行.
说了这么多. 异常也知道如何抛出和处理了. 但是我们现在用的都是人家python给的异
常. 如果某一天. 你写的代码中出现了一个无法用现有的异常来解决问题. 那怎么办呢? 别着
急. python可以自定义异常.
自定义异常: 非常简单. 只要你的类继承了Exception类. 那你的类就是一个异常类. 就这
么简单. 比如. 你要写一个男澡堂子程序. 那这时要是来个女的. 你怎么办? 是不是要抛出一个
性别异常啊? 好. 我们来完成这个案例:

# 继承Exception. 那这个类就是?个异常类
class GenderError(Exception):
pass
class Person:
def __init__(self, name, gender):
self.name = name
self.gender = gender
def nan_zao_tang_xi_zao(person):
if person.gender != "男":
raise GenderError("性别不对. 这?是男澡堂?")
p1 = Person("alex", "男")
p2 = Person("eggon", "蛋")
# nan_zao_tang_xi_zao(p1)
# nan_zao_tang_xi_zao(p2) # 报错. 会抛出?个异常: GenderError
# 处理异常
try:
nan_zao_tang_xi_zao(p1)
nan_zao_tang_xi_zao(p2)
except GenderError as e:
print(e) # 性别不对, 这?是男澡堂?
except Exception as e:
print("反正报错了")

自定义异常

ok搞定. 但是, 如果是真的报错了. 我们在调试的时候, 最好是能看到错误源自于哪里?
怎么办呢? 需要引入另?个模块traceback. 这个模块可以获取到我们每个方法的调用信息.
又被称为堆栈信息. 这个信息对我们拍错是很有帮助的.

import traceback
# 继承Exception. 那这个类就是?个异常类
class GenderError(Exception):
pass
class Person:
def __init__(self, name, gender):
self.name = name
self.gender = genderdef nan_zao_tang_xi_zao(person):
if person.gender != "男":
raise GenderError("性别不对. 这?是男澡堂?")
p1 = Person("alex", "男")
p2 = Person("eggon", "蛋")
# nan_zao_tang_xi_zao(p1)
# nan_zao_tang_xi_zao(p2) # 报错. 会抛出?个异常: GenderError
# 处理异常
try:
nan_zao_tang_xi_zao(p1)
nan_zao_tang_xi_zao(p2)
except GenderError as e:
val = traceback.format_exc() # 获取到堆栈信息
print(e) # 性别不对. 这?是男澡堂?
print(val)
except Exception as e:
print("反正报错了")
结果:
性别不对. 这?是男澡堂?
Traceback (most recent call last):
File "/Users/sylar/PycharmProjects/oldboy/?向对象/day05.py", line 155, in
<module>
nan_zao_tang_xi_zao(p2)
File "/Users/sylar/PycharmProjects/oldboy/?向对象/day05.py", line 144, in
nan_zao_tang_xi_zao
raise GenderError("性别不对. 这?是男澡堂?")
GenderError: 性别不对. 这?是男澡堂?

堆栈信息

搞定了. 这样我们就能收放自如了. 当测试代码的时候把堆栈信息打印出来. 但是当到了
线上的生产环境的时候把这个堆栈去掉即可

日志

首先, 你要知道在编写任何一款软件的时候, 都会出现各种各样的问题或者bug. 这些问
题或者bug一般都会在测试的时候给处理掉. 但是多多少少的都会出现一些意想不到的异常
或者错误. 那这个时候, 我们是不知道哪里出了问题的. 因为很多BUG都不是必现的bug. 如果
是必现的. 测试的时候肯定能测出来. 最头疼的就是这种不必现的bug. 我这跑没问题. 客户那
一用就出问题. 那怎么办呢?我们需要给软件准备一套日志系统. 当出现任何错误的时候. 我
们都可以去日志系统里去查. 看哪里出了问题. 这样在解决问题和bug的时候就多了一个帮手.
那如何在python中创建这个日志系统呢? 很简单

1. 导入logging模块.
2. 简单配置一下logging
3. 出现异常的时候(except). 向日志里写错误信息

# filename: ?件名
# format: 数据的格式化输出. 最终在?志?件中的样?
# 时间-名称-级别-模块: 错误信息
# datefmt: 时间的格式
# level: 错误的级别权重, 当错误的级别权重?于等于leval的时候才会写??件
logging.basicConfig(filename=‘x1.txt‘,
format=‘%(asctime)s - %(name)s - %(levelname)s -%
(module)s: %(message)s‘,
datefmt=‘%Y-%m-%d %H:%M:%S‘,
level=0) # 当前配置表示 10以上的分数会被写??件
# CRITICAL = 50
# FATAL = CRITICAL
# ERROR = 40
# WARNING = 30
# WARN = WARNING
# INFO = 20
# DEBUG = 10
# NOTSET = 0
logging.critical("我是critical") # 50分. 最贵的
logging.error("我是error") # 40分logging.warning("我是警告") # 警告 30
logging.info("我是基本信息") # 20
logging.debug("我是调试") # 10
logging.log(2, "我是?定义") # ?定义. 看着给分

最后, 如果你系统中想要把日志文件分开. 比如. 一个大项目, 有两个子系统, 那两个子系
统要分开记录日志. 方便调试. 那怎么办呢? 注意. 用上面的basicConfig是搞不定的. 我们要
借助文件助手(FileHandler), 来帮我们完成日志的分开记录

import logging
# 创建?个操作?志的对象logger(依赖FileHandler)
file_handler = logging.FileHandler(‘l1.log‘, ‘a‘, encoding=‘utf-8‘)
file_handler.setFormatter(logging.Formatter(fmt="%(asctime)s - %(name)s - %
(levelname)s -%(module)s: %(message)s"))
logger1 = logging.Logger(‘s1‘, level=logging.ERROR)
logger1.addHandler(file_handler)
logger1.error(‘我是A系统‘)# 再创建?个操作?志的对象logger(依赖FileHandler)
file_handler2 = logging.FileHandler(‘l2.log‘, ‘a‘, encoding=‘utf-8‘)
file_handler2.setFormatter(logging.Formatter(fmt="%(asctime)s - %(name)s -
%(levelname)s -%(module)s: %(message)s"))
logger2 = logging.Logger(‘s2‘, level=logging.ERROR)
logger2.addHandler(file_handler2)
logger2.error(‘我是B系统‘)

原文地址:https://www.cnblogs.com/wangpanger/p/10199964.html

时间: 2024-11-11 10:48:53

零碎知识点:装饰器、MD5、异常、日志的相关文章

python中用修饰器进行异常日志记录

当脚本中需要进行的的相同的异常操作很多的时候,可以用修饰器来简化代码.比如我需要记录抛出的异常: 在log_exception.py文件中, import functools import logging def create_logger(): logger = logging.getLogger("test_log") logger.setLevel(logging.INFO) fh = logging.FileHandler("test.log") fmt =

装饰器的写法以及应用环境

1.授权(Authorization) 装饰器能有助于检查某个人是否被授权去使用一个web应用的端点(endpoint).它们被大量使用于Flask和Django web框架中.这里是一个例子来使用基于装饰器的授权: from functools import wraps # 最新版python引用是 import functools def requires_auth(f): # f 就是我们需要装饰的函数,一看就是不带参数的装饰器 @wraps(f) # 新版python写法 @functo

Python装饰器的写法以及应用场景

应用场景: 1.授权(Authorization) 装饰器能有助于检查某个人是否被授权去使用一个web应用的端点(endpoint).它们被大量使用于Flask和Django web框架中.这里是一个例子来使用基于装饰器的授权: from functools import wraps # 最新版python引用是 import functools def requires_auth(f): # f 就是我们需要装饰的函数,一看就是不带参数的装饰器 @wraps(f) # 新版python写法 @

python函数(3):装饰器

昨天学了很多函数方面的概念和知识其中有一个闭包的函数.很多人都对闭包的作用不是很清楚,今天我们就来认识一个新的知识点装饰器.它就是闭包函数的一个经典应用. 预习: 编写装饰器,为多个函数加上认证的功能(用户的账号密码来源于文件),要求登录成功一次,后续的函数都无需再输入用户名和密码 一.楔子 def func1(): print('in func1') 想要计算上面函数的执行时间: import time def func1(): start = time.time() print('in fu

[python] 之 装饰器

装饰器本质上是一个python函数,它可以让其它函数在不需要做任何代码变动的前提下增加额外的功能,装饰器的返回值也是一个函数对象.装饰器主要用于插入日志.性能测试.事务处理.缓存及权限校验等,解决代码重复使用.值得注意的是,内层函数保留(记忆)了外层函数的(参数)状态. 一.装饰器创建 装饰器的语法以@开头,接着是装饰器函数的变量名和可选参数,紧接着是被修饰函数的定义及被修饰函数的可选参数. 语法: @decorator(dec_opt_args) def fuc2Bdecorated(func

Python的函数式编程-传入函数、排序算法、函数作为返回值、匿名函数、偏函数、装饰器

函数是Python内建支持的一种封装,我们通过把大段代码拆成函数,通过一层一层的函数调用,就可以把复杂任务分解成简单的任务,这种分解可以称之为面向过程的程序设计.函数就是面向过程的程序设计的基本单元. 传入函数 函数的本身也可以作为参数. Python内建的mapreduce的函数.(来源于谷歌的,后来被道格这家伙开源了,成为当今处理大数据最火热的hadoop中的计算模型---MapReduce) 我们先看map.map()函数接收两个参数,一个是函数,一个是序列,map将传入的函数依次作用到序

python装饰器高级用法

1.装饰类 下面,直接来看代码是如何实现装饰类的: def decorator(aClass): class newClass: def __init__(self, age): self.total_display = 0 self.wrapped = aClass(age) def display(self): self.total_display += 1 print("total display", self.total_display) self.wrapped.displa

理解 Python 装饰器看这一篇就够了

讲 Python 装饰器前,我想先举个例子,虽有点污,但跟装饰器这个话题很贴切. 每个人都有的内裤主要功能是用来遮羞,但是到了冬天它没法为我们防风御寒,咋办?我们想到的一个办法就是把内裤改造一下,让它变得更厚更长,这样一来,它不仅有遮羞功能,还能提供保暖,不过有个问题,这个内裤被我们改造成了长裤后,虽然还有遮羞功能,但本质上它不再是一条真正的内裤了.于是聪明的人们发明长裤,在不影响内裤的前提下,直接把长裤套在了内裤外面,这样内裤还是内裤,有了长裤后宝宝再也不冷了.装饰器就像我们这里说的长裤,在不

Python入门之装饰器

讲 Python 装饰器前,我想先举个例子,虽有点污,但跟装饰器这个话题很贴切. 每个人都有的内裤主要功能是用来遮羞,但是到了冬天它没法为我们防风御寒,咋办?我们想到的一个办法就是把内裤改造一下,让它变得更厚更长,这样一来,它不仅有遮羞功能,还能提供保暖,不过有个问题,这个内裤被我们改造成了长裤后,虽然还有遮羞功能,但本质上它不再是一条真正的内裤了.于是聪明的人们发明长裤,在不影响内裤的前提下,直接把长裤套在了内裤外面,这样内裤还是内裤,有了长裤后宝宝再也不冷了.装饰器就像我们这里说的长裤,在不

Python 函数式编程、装饰器以及一些相关概念简介

Python 中的 Decorator(装饰器) 是对一个函数或者方法的封装,从而使其可以完成一些与自身功能无关的工作. 预备知识 一切皆对象 在 Python 中,所有的一切都被视为对象,任何的变量.函数.类等都是 object 的子类.因此除了变量之外,函数和类等也可以被指向和传递. >>> def foo(): ... pass ... >>> def Foo(): ... pass ... >>> v = foo >>> v