约束/自定义异常/hashlib/logging

一.为检验上一篇博客学过后的水平

class Course:
    def __init__(self,name,price,period):
        self.name = name
        self.price = price
        self.period = period

class Student:
    def __init__(self,name):
        self.name = name
        self.courses = []

    def select_course(self):
        """
        选择课程,已选则不能再选
        :return:
        """
        pass

    def show_selected_course(self):
        """
        查看自己所选课程
        :return:
        """
        pass

    def show_del_course(self):
        """
        删除已选课程
        :return:
        """
        pass
def run():
    """
    主程序
        1.根据Course类创建10个课程
        2.用户输入学生姓名,动态创建学生对象
        3.查看所有课程
        4.为学生选课
        5.查看学生已选课程
        6.删除已选课程
    :return:
    """
    pass
套餐:补充选课系统(必须用反射)
total_course_list = []

class Course:
    def __init__ (self,name,price,period):
        self.name = name
        self.price = price
        self.period = period
class Student:

    func_list = [
        {'text':'选课','name':'select_course'},
        {'text':'查看课程','name':'select_course'},
        {'text':'删除课程','name':'select_course'},
    ]

    def __init__ (self,name):
        self.name = name
        self.courses = []
    def select_course(self):
        for i,item in enumerate(total_course_list,1):
            print(i,item.name,item.price,item.period)
        num = int(input('请选择要选择的课程'))
        num = num - 1
        course_obj = total_course_list[num]
        if course_obj not in self.courses:
            self.courses.append(course_obj)

    def show_selected_course(self):
        pass
    def show_del_course(self):
        pass

def run():

    for i in range(1,11):
        obj = Course('xx-%s' % i, 90, 90)
        total_course_list.append(obj)

    stu_name = input('请输入学生姓名')
    stu = Student(stu_name)

    for i,item in enumerate(stu.func_list,1):
        print(i,item['text'])

    while True:
        num = int(input('请选择要执行的功能序号:'))
        num = num-1
        row = stu.func_list[num]
        name = row['name']
        func = getattr(stu,name)
        func()

if __name__ == '__main__':
    run()
方法一:
class Course:
    def __init__ (self,name,price,period):
        self.name = name
        self.price = price
        self.period = period
class Student:

    func_list = [
        {'text':'选课','name':'select_course'},
        {'text':'查看课程','name':'show_selected_course'},
        {'text':'删除课程','name':'show_del_course'},
    ]

    def __init__ (self,name):
        self.name = name
        self.courses = []
    def select_course(self,csl):
        pass
    def show_selected_course(self,csl):
        pass
    def show_del_course(self,csl):
        pass

def run():
    total_course_list = []
    for i in range(1,11):
        obj = Course('xx-%s' % i, 90, 90)
        total_course_list.append(obj)

    stu_name = input('请输入学生姓名')
    stu = Student(stu_name)

    for i,item in enumerate(stu.func_list,1):
        print(i,item['text'])


    while True:
        num = int(input('请选择要执行的功能序号:'))
        num = num-1
        row = stu.func_list[num]
        name = row['name']
        func = getattr(stu,name)
        func(total_course_list)

if __name__ == '__main__':
    run()
方法二:

二.约束

1.Python中建议使用:

class BaseMessage(object):
    def send(self,x1):
        """
        必须继承BaseMessage,然后其中必须编写send方法。用于完成具体业务逻辑。
        """
        raise NotImplementedError(".send() 必须被重写.")

class Email(BaseMessage):
    def send(self,x1):
        """
        必须继承BaseMessage,然后其中必须编写send方法。用于完成具体业务逻辑。
        """
        pass

obj = Email()
obj.send(1)

BaseMessage类用于约束,约束其派生类:保证派生类中必须编写send方法,不然执行可能就会报错。

2.小练习加补充

class BaseMessage(object):
    def send(self):
        """
        必须继承BaseMessage,然后其中必须编写send方法。用于完成具体业务逻辑。
        """
        raise NotImplementedError(".send() 必须被重写.")
                # raise Exception(".send() 必须被重写.")

class Email(BaseMessage):
    def send(self):
        pass # 发送邮件

    def f1(self):
        pass

    def f2(self):
        pass
class Wechat(BaseMessage):
    def send(self):
        pass # 发送微信

    def f1(self):
        pass

    def f2(self):
        pass
class Msg(BaseMessage):
    def send(self):
        pass # 发送短信

    def f1(self):
        pass

    def f2(self):
        pass


def func(arg):
    """
    报警通知的功能
    """
    arg.send()    
小练习走一波:
from abc import ABCMeta,abstractmethod

class Base(metaclass=ABCMeta): # 抽象类

    def f1(self):
        print(123)

    @abstractmethod
    def f2(self):   # 抽象方法
        pass

class Foo(Base):

    def f2(self):
        print(666)

obj = Foo()
obj.f1()
补充:以抽象类,抽象方法的约束

抽象类,约束,约束继承它的派生类必须实现它其中的抽象方法。

3.需要知道的:接口

接口,接口中不允许在方法内部写代码,只能约束继承它的类必须实现接口中定义的所有方法。

interface IFoo:
    
    def f1(self,x1):pass 
    
    def f2(self,x1):pass 
    
interface IBar:
    
    def f3(self,x1):pass 
    
    def f4(self,x1):pass 

class Foo(IFoo,IBar):# 实现了2个接口
    
    def f1(self,x1):pass 
    
    def f2(self,x1):pass 
    
    def f3(self,x1):pass 
    
    def f4(self,x1):pass 
Java,C#

4.总结:

(1).什么是接口以及作用

  接口是一种数据类型,主要用于约束派生类中必须实现指定的方法

  Python中,不存在;JAVA,C# 中是存在的

(2).Python中使用过什么来约束呐?

  抽象类+抽象方法 --> 编写上麻烦

  人为主动抛出异常

(3).约束时,抛出的异常是否可以用其他的?

  不专业:raise Exception(".send() 必须被重写")

  专业:raise NotImplementedError(".send() 必须被重写")

(4).以后看代码揣摩心思

5.应用场景

多个类.内部都必须有某些方法时,需要使用基类+异常进行约束

class IBase:
    def login():
        raise NotImplementedError(".send() 必须被重写.")

class Student:
    def login(self):
        pass 
        
    def score(self):
        pass 
        
class Teacher:
    def login(self):
        pass 
        
    def exam(self):
        pass 
        
class Manager(self):
    def login(self):
        pass         
    ....
学生管理程序

三.异常处理

import os

class ExistsError(Exception):
    pass

class KeyInvalidError(Exception):
    pass

def new_func(path,prev):
    """
    去path路径的文件中,找到前缀为prev的一行数据,获取数据并返回给调用者。
        1000,成功
        1001,文件不存在
        1002,关键字为空
        1003,未知错误
        ...
    :return:
    """
    response = {'code':1000,'data':None}
    try:
        if not os.path.exists(path):
            raise ExistsError()

        if not prev:
            raise KeyInvalidError()
        pass
    except ExistsError as e:
        response['code'] = 1001
        response['data'] = '文件不存在'
    except KeyInvalidError as e:
        response['code'] = 1002
        response['data'] = '关键字为空'
    except Exception as e:
        response['code'] = 1003
        response['data'] = '未知错误'
    return response


def func(path,prev):
    """
    去path路径的文件中,找到前缀为prev的一行数据,获取数据并返回给调用者。
        1000,成功
        1001,文件不存在
        1002,关键字为空
        1003,未知错误
        ...
    :return:
    """
    response = {'code':1000,'data':None}
    try:
        if not os.path.exists(path):
            response['code'] = 1001
            response['data'] = '文件不存在'
            return response
        if not prev:
            response['code'] = 1002
            response['data'] = '关键字为空'
            return response
        pass
    except Exception as e:
        response['code'] = 1003
        response['data'] = '未知错误'
    return response

def show():
    return 8

def run():
    pass
异常处理
# 知识点:如何自定义异常类?
class MyException(Exception):
    def __init__(self,code,msg):
        self.code = code
        self.msg = msg
try:
    # 知识点:主动抛出异常 
    raise MyException(1000,'操作异常')

except KeyError as obj:
    print(obj,1111)
except MyException as obj: # 知识点:捕获异常 
    print(obj,2222)
except Exception as obj:
    print(obj,3333)
自定义异常处理

四.加密,hashlib模块

hashlib主要提供字符加密功能,将md5和sha模块整合到了一起,支持md5,sha1, sha224, sha256, sha384, sha512等算法

撞库:md5对象,md5不能反解,但是加密是固定的,就是关系是一一对应,所以有缺陷,可以被对撞出来

加盐:创建一个只有自己知道的字符串加入到你要加密的密码中,然后再进行加密,这样不容易被撞出来

import hashlib

SALT = b'2erer3asdfwerxdf34sdfsdfs90'   # 加盐,为防止被撞库获取到密码,对密码进行加盐

def md5(pwd):
    # 实例化对象
    obj = hashlib.md5(SALT)
    # 写入要加密的字节
    obj.update(pwd.encode('utf-8'))  #以UTF-8的方式进行编码
    # 获取密文
    return obj.hexdigest() # 21232f297a57a5a743894a0e4a801fc3 # 66fbdc0f98f68d69cd458b0cee975fe3 # c5395258d82599e5f1bec3be1e4dea4a

user = input("请输入用户名:")
pwd = input("请输入密码:")
if user == 'oldboy' and md5(pwd) == 'c5395258d82599e5f1bec3be1e4dea4a':
    print('登录成功')
else:
    print('登录失败')

# 明文密码:admin
加密

 五.日志

1.用于便捷记录日志且线程安全的模块

import logging

logger = logging.basicConfig(filename='xxxxxxx.txt',
                             format='%(asctime)s - %(name)s - %(levelname)s -%(module)s:  %(message)s',
                             datefmt='%Y-%m-%d %H:%M:%S',
                             level=30)

# logging.debug('x1') # 10
# logging.info('x2')  # 20
# logging.warning('x3') # 30
# logging.error('x4')    # 40
# logging.critical('x5') # 50
# logging.log(10,'x6')


import traceback

def func():
    try:
        a = a +1
    except Exception as e:
        # 获取当前错误的堆栈信息
        msg = traceback.format_exc()
        logging.error(msg)
func()

 这样获取到的是报错的全部信息,错误的优先级在logging默认是30

 

2.上方的日志不支持把日志分别添加到不同的文件中去,想要实现这一操作,必须得自定义日志

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('123123123')


# 在创建一个操作日志的对象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('666')
自定义日志

这个是把不同的日志放到不同的文件中去

3.为什么要写日志?

  给开发人员看,用于排查错误

posted @ 2018-08-31 18:02  骑驴老神仙  阅读(188)  评论(0)    收藏  举报