面向对象:单例模式实现的多种方式、pickle序列化模块、选课系统需求分析等

一、单例模式实现的多种方式

方式一:使用类

class C1:
    __instance = None

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

    @classmethod
    # 使用装饰器,让类加括号调用的时候也可以省略一个参数
    def singleton(cls):
        # 判断cls(也就是self)有没有这个属性,也就是判断之前内部有没有创建出__instance这个对象,如果有了就直接把他返回出去,如果没有就根据设定的参数创建一个
        if not cls.__instance:
            cls.__instance = cls('jason', 18)
        return cls.__instance


# 这里的意思是如果不传参数产生新的对象,始终调用的是同一个对象
obj1 = C1.singleton()
obj2 = C1.singleton()
obj3 = C1.singleton()
print(id(obj1), id(obj2), id(obj3))
obj4 = C1('kevin', 28)
obj5 = C1('tony', 38)
print(id(obj4), id(obj5))

方法二:使用metaclass方式(自定义元类)

class Mymeta(type):
    def __init__(self, name, bases, dic):  # 定义类Mysql时触发
        # 事先先从配置文件中取配置来造一个Mysql的实例出来(创建类的时候就给这个类创建一个默认对象)
        self.__instance = object.__new__(self)  # 产生对象
        self.__init__(self.__instance, 'jason', 18)  # 初始化对象
        # 上述两步可以合成下面一步
        # self.__instance=super().__call__(*args,**kwargs)

        # 这里就是用type的代码生成新的类
        super().__init__(name, bases, dic)

    def __call__(self, *args, **kwargs):  # Mysql(...)时触发
        # 当类定义好了之后,被用来创建对象的时候就会触发双下call,如果没有传参就直接返回之前创建类时定义的默认对象,如果有参数就会生成新的对象返回出去
        if args or kwargs:  # args或kwargs内有值
            obj = object.__new__(self)
            self.__init__(obj, *args, **kwargs)
            return obj
        return self.__instance


class Mysql(metaclass=Mymeta):
    def __init__(self, name, age):
        self.name = name
        self.age = age

# 这里的思路跟上面的相似,如果不传参数产生新的对象,始终调用的是同一个对象
obj1 = Mysql()
obj2 = Mysql()
print(id(obj1), id(obj2))
obj3 = Mysql('tony', 321)
obj4 = Mysql('kevin', 222)
print(id(obj3), id(obj4))

方法三:自定义双下new

# 类加括号无论执行多少次永远只会产生一个对象
class Singleton_:
    def __new__(self, *args, **kwargs):
        if not hasattr(self, '_instance'):
            self._instance = super(Singleton_, self).__new__(self)
        return self._instance

# 我们通过今天的学习得知双下new是产生对象的方法,因此我们在这里进行派生操作,
# 用hasatter检测当前类是是否已经有创建过对象,如果没有就新建一个(使用super拿元类的双下new直接创建),如果有就返回有的那个对象
a = Singleton_()
b = Singleton_()
print(a)
print(b)

方法四:基于模块的单例模式

在一个py文件中创建一个对象,然后让外面的代码一直调用这个类,从而达到单例模式的目的。

'''基于模块的单例模式:提前产生一个对象 之后导模块使用'''
class C1:
    def __init__(self, name):
        self.name = name

obj = C1('jason')


def outer(cls):
    _instance = cls('jason', 18)
    def inner(*args, **kwargs):
        if args or kwargs:
            obj = cls(*args, **kwargs)
            return obj
        return _instance

    return inner


@outer  # Mysql=outer(Mysql)
class Mysql:
    def __init__(self, host, port):
        self.host = host
        self.port = port


obj1 = Mysql()
obj2 = Mysql()
obj3 = Mysql()
print(obj1 is obj2 is obj3)  # True

obj4 = Mysql('1.1.1.3', 3307)
obj5 = Mysql('1.1.1.4', 3308)
print(obj3 is obj4)  # False

二、pickle序列化模块

优势:

python中几乎所有的数据类型(列表,字典,集合,类等)都可以用pickle来序列化,我们的对象也可以。

缺陷:

1、只能够在python中使用,无法跨语言传输

2、pickle序列化后的数据,可读性差,人一般无法识别。(都是二进制,能看懂也不是一般人了)

方法介绍:


pickle.dump(obj, file[, protocol])
  序列化对象,并将结果数据流写入到文件对象中。参数protocol是序列化模式,默认值为0,表示以文本的形式序列化。protocol的值还可以是1或2,表示以二进制的形式序列化。


pickle.load(file)
  反序列化对象。将文件中的数据解析为一个Python对象。

其中要注意的是,在load(file)的时候,要让python能够找到类的定义,否则会报错:

注:pickle模块保存数据的文件是没有格式的

"""
需求:产生一个对象并保存到文件中 取出来还是一个对象
"""


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

    def func1(self):
        print('from func1')

    def func2(self):
        print('from func2')


obj = C1('jason', 18)
# 这里的对象,并不能以对象的格式存储到文件中,因此我们使用pickle模块
import pickle

with open(r'a.txt', 'wb') as f:
    pickle.dump(obj, f)
with open(r'a.txt', 'rb') as f:
    data = pickle.load(f)
print(data)
# <__main__.C1 object at 0x0000026712C75BB0>

三、选课系统需求分析

选课系统项目需求

角色:

学校、学员、课程、讲师

要求:

   1. 创建北京、上海 2 所学校
   2. 创建linux , python , go 3个课程 , linux\py 在北京开, go 在上海开
   3. 课程包含,周期,价格,通过学校创建课程 
   4. 通过学校创建班级, 班级关联课程、讲师5. 创建学员时,选择学校,关联班级
   5. 创建讲师角色时要关联学校, 
   6. 提供三个角色接口  
       6.1 学员视图, 可以注册, 交学费, 选择班级,  
       6.2 讲师视图, 讲师可管理自己的班级, 上课时选择班级, 查看班级学员列表 , 修改所管理的学员的成绩   
       6.3 管理视图,创建讲师, 创建班级,创建课程
   7. 上面的操作产生的数据都通过pickle序列化保存到文件里

四、功能提炼

1.管理员功能

  • 登录功能
  • 创建学校
  • 创建课程
  • 创建老师

2.讲师功能

  • 登录功能
  • 选择课程
  • 查看课程
  • 查看学生分数
  • 修改学生分数

3.学生功能

  • 注册功能
  • 登录功能
  • 选择学校
  • 选择课程
  • 查看课程分数

五、选课系统架构设计

使用的也是三层架构结构

img

三层架构

功能展示层

  • src.py
  • admin_view.py
  • teacher_view.py
  • student_view.py

核心逻辑层

  • admin_interface.py
  • teacher_interface.py
  • student_interface.py

数据处理层

  • db_hanlder.py
  • model.py

六、选课系统目录搭建和功能划分

目录搭建

这里我们依旧使用软件开发目录规范

功能划分

包/文件 功能
interface 登录、注册、选课等接口
core 注册、登录、选课核心功能
db 存放注册的用户数据
lib 存放编写的公共方法文件
conf 存放配置文件
readme.txt 项目文档
requirements.txt 开发文件说明
posted @ 2022-11-09 18:29  wwwxxx123  阅读(31)  评论(0编辑  收藏  举报