110、信号、Django的信号、Flask-Script、Sqlalchemy介绍和快速使用

84 阅读8分钟

今日内容概要

  • 信号
  • Django的信号
  • Flask-Script
  • Sqlalchemy介绍和快速使用

今日内容详细

信号

1.Flask框架中的信号基于blinker,其主要就是让开发者可是在flask请求过程中定制一些用户行为
2.信号分为内置信号和自定义信号

内置信号

1.安装:pip3 install blinker
2.有哪些内置信号?
  request_started = _signals.signal('request-started')                # 请求到来前执行
  request_finished = _signals.signal('request-finished')              # 请求结束后执行

  before_render_template = _signals.signal('before-render-template')  # 模板渲染前执行
  template_rendered = _signals.signal('template-rendered')            # 模板渲染后执行

  got_request_exception = _signals.signal('got-request-exception')    # 请求执行出现异常时执行

  request_tearing_down = _signals.signal('request-tearing-down')      # 请求执行完毕后自动执行(无论成功与否)
  appcontext_tearing_down = _signals.signal('appcontext-tearing-down')# 应用上下文执行完毕后自动执行(无论成功与否)

  appcontext_pushed = _signals.signal('appcontext-pushed')            # 应用上下文push时执行
  appcontext_popped = _signals.signal('appcontext-popped')            # 应用上下文pop时执行
  message_flashed = _signals.signal('message-flashed')                # 调用flask在其中添加数据时,自动触发

内置信号的使用

1.第一步:写一个函数
def test(app, **kwargs):
    print(app)
    print(type(kwargs))
    # 请求地址是根路径,才记录日志,其它都不记录
    print(kwargs['context']['request'].path)
    if kwargs['context']['request'].path == '/':
        print('记录日志了')


2.第二步:跟内置信号绑定
# signals 中有很多内置信号
signals.before_render_template.connect(test)

3.第三步:等待信号被触发(不需要咱们做)--->只要执行到内置信号位置,绑定的函数就会执行
	@app.route('/')
  def index():
      session['uu'] = '00'
      session_input.send(app, session=session, kk={'name': 'uu'})
      return render_template('index.html')

自定义信号

1.第一步:定义一个自定义 信号
	session_input = _signals.signal('session_input')
2.第二步:写个函数
	def test2(*args, **kwargs):
    print(args)  # app
    print(kwargs)  # {session   kk}
    print('session放值了')
    if kwargs.get('kk').get('name') == 'xx':
        print('记录日志')
3.第三步:函数跟自己定义信号绑定
	session_input.connect(test2)
4.第四步:触发自定义信号---》我们做
from flask import Flask, session, render_template, signals
from flask.signals import _signals

# pip3 install blinker
app = Flask(__name__)
app.debug = True
app.secret_key = 'asdfasdfasdf'


# 需求: 使用内置信号实现,每次模板渲染之前,都执行某个代码

# 第一步:写一个函数
def test(app, **kwargs):
    print(app)
    print(type(kwargs))
    # 请求地址是根路径,才记录日志,其它都不记录
    print(kwargs['context']['request'].path)
    if kwargs['context']['request'].path == '/':
        print('记录日志了')


# 第二步:跟内置信号绑定
# signals 中有很多内置信号
# signals.before_render_template.connect(test)

# 第三步:等待信号被触发(不需要咱们做)--->只要执行到内置信号位置,绑定的函数就会执行


#### 自定义信号--->session每次放一个值,我们就执行信号
# 1 定义信号
# 自定义信号
session_input = _signals.signal('session_input')


# 2 写个函数
def test2(*args, **kwargs):
    print(args)  # app
    print(kwargs)  # {session   kk}
    print('session放值了')
    if kwargs.get('kk').get('name') == 'xx':
        print('记录日志')


# 3 绑定信号
session_input.connect(test2)


# 4 触发信号
@app.route('/')
def index():
    session['uu'] = '00'
    session_input.send(app, session=session, kk={'name': 'uu'})
    return render_template('index.html')


@app.route('/home')
def home():
    return render_template('home.html')


@app.route('/order')
def order():
    session['xx'] = 'xx'
    session_input.send(app, session=session, kk={'name': 'xx'})
    return "order"


if __name__ == '__main__':
    app.run(port=8080)

信号的作用-->对代码进行解耦

1.记录日志:只要是张三用户,访问index页面,就记录日志
2.只要用户表中,插入一条记录,就给用户发个短信通知
	-User.object.create--->调用发短信方法-->找到10个地址--->改10个地方
  -如果有个内置信号--->只要表中增加记录,就会触发这个信号--->通过信号内判断这个表是不是User表,决定要不要发短信
  -flask中没有这个内置信号--->自定义

补充:信号量

# acquire():消耗信号量
# release():释放信号量
import threading
import time
def run(n):
    semaphore.acquire()  # 计数器获取锁
    time.sleep(5)  # 程序休眠5秒
    print (n)
    semaphore.release()  # 计数器释放锁

if __name__ == '__main__':
    # 添加一个计数器,最大并发线程数量5(最多同时运行5个线程)
    semaphore = threading.Semaphore(5)
    for i in range(50):
        t = threading.Thread(target=run, args=(i, ))  # 创建线程
        t.start()

Django的信号

内置信号-Model signals

pre_init                    # django的modal执行其构造方法前,自动触发
post_init                   # django的modal执行其构造方法后,自动触发
pre_save                    # django的modal对象保存前,自动触发
post_save                   # django的modal对象保存后,自动触发
pre_delete                  # django的modal对象删除前,自动触发
post_delete                 # django的modal对象删除后,自动触发
m2m_changed                 # django的modal中使用m2m字段操作第三张表(add,remove,clear)前后,自动触发
class_prepared              # 程序启动时,检测已注册的app中modal类,对于每一个类,自动触发
Management signals
pre_migrate                 # 执行migrate命令前,自动触发
post_migrate                # 执行migrate命令后,自动触发
Request/response signals
request_started             # 请求到来前,自动触发
request_finished            # 请求结束后,自动触发
got_request_exception       # 请求异常后,自动触发
Test signals
setting_changed             # 使用test测试修改配置文件时,自动触发
template_rendered           # 使用test测试渲染模板时,自动触发
Database Wrappers
connection_created          # 创建数据库连接时,自动触发

内置信号使用

1.内置信号使用(需求:当user表创建用户,就给用户发个邮件)
	1.写个函数   #放到__init__里
    from django.db.models.signals import pre_save
    import logging
    def callBack(sender, **kwargs):
      logging.debug('%s创建了一个%s对象'%(sender._meta.model_name,kwargs.get('instance').title))

        
  2.绑定内置信号   
		pre_save.connect(callBack)
	3.等待触发
    
    
2.内置信号
    from django.db.models.signals import pre_save
    from django.dispatch import receiver
    @receiver(pre_save)
    def my_callback(sender, **kwargs):
        print("对象创建成功")
        print(sender)
        print(kwargs)
        
3.在django中 有两种绑定信号的方式
	1 @receiver(信号名)  装饰对应的函数
	2 信号名.connect(对应的函数名)
    

自定义信号

1.定义信号(一般创建一个py文件)(toppings,size 是接受的参数)
  	import django.dispatch
    pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"])
2.注册信号
    def callback(sender, **kwargs):
        print("callback")
        print(sender,kwargs)
    pizza_done.connect(callback)
3.触发信号
    from 路径 import pizza_done
    pizza_done.send(sender='seven',toppings=123, size=456)
    
4.自定义信号干过什么?
	-做双写一致性的缓存更新
  

flask-script

1.用于实现类似于django中 python3 manage.py runserver ...类似的命令(安装:pip3 install flask-script)

2.django中,有命令  
  python manage.py runserver
  python manage.py makemigrations
  自定制命令(django如何自定制命令)...
 			 -python manage.py init_db  excel文件路径  指定表名

3.需求:flask启动项目,像djagno一样,通过命令启动
	下载:pip install Flask==2.2.2
  		 pip install Flask_Script==2.0.3
  		
4.借助于:flask-script 实现
	-安装:pip3.8 install flask-script
  -修改代码:
    from flask_script import Manager
    manager=Manager(app)
    manager.run()
	-用命令启动
    	python38 manage.py runserver
        
5.自定制命令
	1.简单自定制命令
    @manager.command
    def custom(arg):
        # 命令的代码,比如:初始化数据库, 有个excel表格,使用命令导入到mysql中
        print(arg)

    2.复杂一些的自定制命令
    @manager.option('-n', '--name', dest='name')
    @manager.option('-u', '--url', dest='url')
    def cmd(name, url):
        # python run.py cmd -n lqz -u xxx
        # python run.py cmd --name lqz --url uuu
        print(name, url)
    
6.django 中如何自定制命令-->搜索

manage.py

from flask import Flask
from flask_script import Manager

app = Flask(__name__)

manager=Manager(app)
app.debug = True
app.secret_key = 'asdfasdfasdf'


@manager.command
def custom(arg):
    # 命令的代码,比如:初始化数据库, 有个excel表格,使用命令导入到mysql中
    print(arg)

#2 复杂一些的自定制命令
@manager.option('-n', '--name', dest='name')
@manager.option('-u', '--url', dest='url')
def cmd(name, url):
    # python38 manage.py cmd -n lqz -u xxx
    # python38 manage.py cmd --name lqz --url uuu
    print(name, url)


@app.route('/')
def index():
    return 'indx'

if __name__ == '__main__':
    manager.run()

Sqlalchemy介绍和快速使用

SQLAlchemy是一个基于Python实现的ORM框架。该框架建立在 DB API之上,使用关系对象映射进行数据库操作,简言之便是:将类和对象转换成SQL,然后使用数据API执行SQL并获取执行结果。

1.安装:pip3 install sqlalchemy
2.ORM框架---->django orm---->只能用在django中,不能独立使用
	1.python界的orm框架
	2.peewee
    -sqlalchemy:企业级
    -djagno rom
    -Tortoise ORM
    -GINO 
	3.go 界orm框架
		-gorm  国人写的
    -Xorm
    
3.java界orm框架
		-ssh 框架springmvc  structs   Hibernate(java的orm框架)
    -ssh  spring    springmvc   Hibernate
    -ssm  Spring    SpringMVC    MyBatis (orm框架)
    -springboot :sb框架 ---》java工程师就是spring工程师
    -spring cloud
    
4.分层:
  Engine,框架的引擎
  Connection Pooling ,数据库连接池
  Dialect,选择连接数据库的DB API种类(sqlite,mysql...)
  Schema/Types,架构和类型
  SQL Exprression Language,SQL表达式语言

5.操作不同数据库
  MySQL-Python
      mysql+mysqldb://<user>:<password>@<host>[:<port>]/<dbname>

  pymysql
      mysql+pymysql://<username>:<password>@<host>/<dbname>[?<options>]

  MySQL-Connector
      mysql+mysqlconnector://<user>:<password>@<host>[:<port>]/<dbname>

  cx_Oracle
      oracle+cx_oracle://user:pass@host:port/dbname[?key=value&key=value...]
更多:http://docs.sqlalchemy.org/en/latest/dialects/index.html

6.了解
	orm不能创建数据库--->只能创建表,删除表--->sqlalchemy不能增加删除字段-->借助于第三方插件实现

sqlalchemy的原生操作

import sqlalchemy
from sqlalchemy import create_engine
from sqlalchemy.engine.base import Engine

# 1.第一步:创建engine对象
engine = create_engine(
    "mysql+pymysql://root:123@127.0.0.1:3306/cars?charset=utf8",
    max_overflow=0,  # 超过连接池大小外最多创建的连接
    pool_size=5,  # 连接池大小
    pool_timeout=30,  # 池中没有线程最多等待的时间,否则报错
    pool_recycle=-1  # 多久之后对线程池中的线程进行一次连接的回收(重置)
)

# 2.第二步:通过engine获得链接
conn=engine.raw_connection()
cursor = conn.cursor()
cursor.execute(
    "select * from news"
)
result = cursor.fetchall()
print(result)
cursor.close()
conn.close()
import sqlalchemy
from sqlalchemy import create_engine
from sqlalchemy.engine.base import Engine
from models import Base
# 第一步:创建engine对象
engine = create_engine(
    "mysql+pymysql://root:123@127.0.0.1:3306/db001?charset=utf8",
    max_overflow=0,  # 超过连接池大小外最多创建的连接
    pool_size=5,  # 连接池大小
    pool_timeout=30,  # 池中没有线程最多等待的时间,否则报错
    pool_recycle=-1  # 多久之后对线程池中的线程进行一次连接的回收(重置)
)

#1 在数据库中创建表
# Base.metadata.create_all(engine)

# 2 删除表
# Base.metadata.drop_all(engine)
from sqlalchemy import Column, Integer, String, Text, ForeignKey, DateTime, UniqueConstraint, Index
from sqlalchemy.ext.declarative import declarative_base
import datetime

Base = declarative_base()


# print(type(Base))
class User(Base):
    # 以__开头的是配置
    __tablename__ = 'users'  # 数据库表名称,如果不写,以类名作为表名

    id = Column(Integer, primary_key=True)  # 主键索引,聚簇索引
    name = Column(String(64), index=True, nullable=False)  # name字段加辅助索引
    email = Column(String(32), unique=True)
    # datetime.datetime.now不能加括号,加了括号,以后永远是当前时间
    ctime = Column(DateTime, default=datetime.datetime.now())
    extra = Column(Text, nullable=True)

    __table_args__ = (
        UniqueConstraint('id', 'name', name='uix_id_name'), #联合唯一
        Index('ix_id_name', 'name', 'email'), #索引
    )