kaixin
Published on 2023-05-23 / 20 Visits
0

Falsk-Restful

Falsk-Restful

属于一种快速构建api的一种工具,和django的还是不相同的
​
安装
pip install flask-restful
​
可以和django一样进行 定义 类视图的方式
但是falskrestful中可以直接返回字典格式,它自动转换为json格式给客户端

1.基本的使用

from flask import Flask
from flask_restful import Resource, Api # 导入restful中的 Resource 和api
​
app = Flask(__name__)
api = Api(app) # 创建一个restful的api对象,将flaks的实例对象app传入
​
# 没有了django的序列化器的概念
# 设置路视图 继承flask_restful 中的 Resource
class HellWord(Resource):
    def get(self):
        return {'name': 'hello'} # 不需要在进行转换json格式,内部已经 帮助转换过了
​
    def post(self):
        return {'age': '18'}
​
​
# 使用api对象,将类视图,和路由信息进行绑定
# add_resource 帮我们调用了django的 as
api.add_resource(HellWord, '/') # hellword 类视图, / 为绑定的路径
if __name__ == '__main__':
    app.run()
​
127.0.0.1/ get  {'name': 'hello'}
127.0.0.1/psot {'age': '18'}
​
​
# 补充  在对象api.add_resource将 类视图和路由绑定的时候,可以进行设置一个路由的别名
api.add_resource(HellWord, '/',endpoint='HelloWorld')
通过这个参数endpoint

2.蓝图使用restful

from flask import Flask, Blueprint
from flask_restful import Resource, Api
​
app = Flask(__name__)
bp = Blueprint('bp', __name__)  # 1.创建蓝图对象
​
bp_api = Api(bp)  # 2.将蓝图对象传入,创建一个蓝图的api对象(绑定蓝图对象api对象)
​
​
class HellWord(Resource):
    def get(self):
        return {'name': 'hello'}
​
​
​
bp_api.add_resource(HellWord, '/bp/helloword')  # 3.由蓝图对象调用add_resource 绑定视图类与路径
# 蓝图注册,你不注册到flask全局对象中,蓝图的url是不会显示的
app.register_blueprint(bp)  # 4.还是需要将蓝图注册到全局的app  flask对象中的
if __name__ == '__main__':
    app.run()

3.restful视图使用装饰器

使用Resource 继承的 类视图的父类,中的method_decorators 属性进行使用
​
​
1.列表方式装饰器
    # 这个属性是由flask_restful框架中的 resource的属性当每次访问时,就会读取当前的 method_decorators
    # 1.当成一个列表 如果装饰器不做区分的话,使用列表,那么全部的请求方式都会加上装饰器
    # 2.当成一个字典 如果类视图中想要不同的条件加上不同的装饰器,那么就是用字典的形式
​
​
from flask import Flask
from flask_restful import Resource, Api
​
app = Flask(__name__)
​
​
def decorator1(func):
    def wrapper(*args, **kwargs):
        print('装饰器1')
        return func(*args, **kwargs)
​
    return wrapper
​
​
def decorator2(func): # 先打印
    def wrapper(*args, **kwargs):
        print('装饰器2')
        return func(*args, **kwargs)
​
    return wrapper
​
​
class UserInfo(Resource):
    method_decorators = [decorator1, decorator2]  # 装饰器2 先被打印
    '''    
    内部原理
    for i in method_decorators:
        func =  i(get) # 将当前的get请求的方法传入
        
        第一次遍历:
        get1  = func1(get) # 装饰器 
        第二次遍历
        get2 = func2(get1) # 将上面的装饰器对象get1传入装饰器方法2中
        
        相当于:
        @decorator2  最外层的装饰器 将下面的内容当成参数
        
        @decorator1
        def func()
        
        理解:
        decorator2(decorator1(get)) # 将下面的东西全部传入
    '''
    def get(self):
        return {'name': 'wkx'}
​
    def post(self):
        return {'pwd': 123}
​
​
api = Api(app)
api.add_resource(UserInfo, '/')
if __name__ == '__main__':
    app.run(debug=True)
​
​
    # 为什么先打印第二个不是第一个:
        decorator2(decorator1(get))
            --> decorator2( decorator1(get) --> wrapper内部的函数名字 内部的func指的就是get函数)
                -->decorator2 --> 返回wrapper 内部的函数名字 其中里面的func指定就是
                        ###  decorator1(get) --> wrapper内部的函数 
                    
           当 get方法被调用的使用
                执行 decorator2 --> 返回wrapper 方法 中的 func(是decorator1 中的wrapper)
                执行后 就会还行decorator1中的wrapper 内部的func(就是get方法)
                
                
  调用装饰器  :外部的方法执行
调用被装饰的函数 :才是装饰器内部的方法被执行
​
​
​
####字典形式装饰 根据不同的请求装饰 不同数量的装饰器
from flask import Flask
from flask_restful import Resource, Api
​
app = Flask(__name__)
​
​
def decorator1(func):
    def wrapper(*args, **kwargs):
        print('装饰器1')
        return func(*args, **kwargs)
​
    return wrapper
​
​
def decorator2(func):
    def wrapper(*args, **kwargs):
        print('装饰器2')
        return func(*args, **kwargs)
​
    return wrapper
​
​
class UserInfo(Resource):
    method_decorators = {
        'get': [decorator1, decorator2], # 给get 装饰两个
        'post': [decorator2] # 给 post 装饰一个
    }
​
    def get(self):
        return {'name': 'wkx'}
​
    def post(self):
        return {'pwd': 123}
​
​
api = Api(app)
api.add_resource(UserInfo, '/')
if __name__ == '__main__':
    app.run(debug=True)
​

4.怎么获取请求参数

原生的是request对象获取请求。
restful 提供了requestParset类 用来帮我们校验和转换请求数据的
# 简单的操作
from flask_restful.reqparse import RequestParser # 对request的进一步的封装
​
from flask import Flask
from flask_restful import Resource, Api
from flask_restful.reqparse import RequestParser
​
app = Flask(__name__)
​
​
class UserInfo(Resource):
    # http://127.0.0.1:5000/?a=10&b=20
    def get(self):
        # 作用帮我们省去了前端数据传入时的判断
        # 1. 创建对象
        rp = RequestParser()
        # 2.声明前端传入参数
        rp.add_argument('a')  # 参数名(存在几个就调用几个方法)
        rp.add_argument('b')
        # 3.执行对参数执行校验
        res = rp.parse_args()  # 可以将这个返回校验内容可以当对象和字典
        print(res)  # {'a': '10', 'b': '20'}
        # res.a 或者res.['a']
​
        return {'name': 'wkx'}
​
​
api = Api(app)
api.add_resource(UserInfo, '/')
if __name__ == '__main__':
    app.run(debug=True)
​
​
    
参数说明
1.required # 参数到底需不需要传递
参数默认为 False 可以不用传值
rp.add_argument('a', required=True)  报错400
# 如果设置为true,不传值的情况下,就会提示错误提示,参数必须要传
​
​
2. help 参数,定制错误的提示
rp.add_argument('a', required=True,help='参数必须传入')
# 如果这个参数不传入 的情况下,就会提示help定制的内容
​
3.action 如果传入的参数同名的情况下
默认 action='store' 保留出现的第一个
action = 'append' 一列表的形式追加保存同名参数的值
​
 rp.add_argument('a', required=True,help='参数必须传入',action='append')
 http://127.0.0.1:5000/?a=10&a=40&a=50
  {a:[10,40,50]}   
​
4.type 类型的制定
    1.自带
        str  会将全部的参数全部转换为字符串
        rp.add_argument('a', required=True,help='参数必须传入',type=str)
        int 传入其他类型报错,如果可以转换,那么他会转换为int
        rp.add_argument('a', required=True,help='参数必须传入',type=int) 
​
    在inputs模块中指定的其他类型
    from flask_restful import inputs 
    int_range :整数的返回最小1最大5 整数的返回
    rp.add_argument('a', required=True, help='参数必须传入',type=inputs.int_range(1, 5))
​
    natural 自然数 0...10
            rp.add_argument('a', required=True, help='参数必须传入',
                            type=inputs.natural)
​
    positive 正整数
            rp.add_argument('a', required=True, help='参数必须传入',
                            type=inputs.positive)
​
    regex 传入正则表达式    
             rp.add_argument('a', required=True, help='参数必须传入',
                            type=inputs.regex(r'\d'))
    url 传入url
            rp.add_argument('a', required=True,
                            type=inputs.url)
​
​
    自定义类型
    def mobile(mobile_str):
        '''    
        :param mobile_str:  接受被验证的参数(接受参数)
        :return: 返回验证过的数据,如果错误就抛出异常 
        '''
​
        if re.match(r'^1[3-9]\d{9}$',mobile_str):
            return mobile_str
        else:
            raise ValueError('{} 格式不正确'.format(mobile_str))
​
    rp.add_argument('a', required=True,type=mobile)
    
    
 5.指明参数传递的位置 location
        rp.add_argument('a', required=True,
                        type=int,location='form')
    
    json files cookies headers args form 等位置进行传参
    可以指明多个属性 location = [json,files]

5.使用序列化工具使用

可以直接返回字典类型转为json格式给前端
把模型类转换字典,指定需要的内容。
工具:marshal
from flask_restful import marshal_with,marshal
​
marshal_with : 以视图装饰器的形式进行返回数据,对原始的marshl进行封装
marshal :直接返回数据
​
​
from flask import Flask
from flask_restful import Resource, Api, marshal_with, fields(定义类型), marshal
​
​
app = Flask(__name__)
# 指明返回数据的类型,指明以后数据会根据这个指明的类型进行转换
resouecr_fields = {
    'name': fields.String,
    'age': fields.Integer,
    'address': fields.String
}
​
​
class UserInfo(Resource):
    # @marshal_with(resouecr_fields,envelope='data)
    # def get(self):
         # 如果数据少传,就会根据少的字段赋值为null
    #     return {'name':'wkx','age':123}
​
    def get(self):
        # 效果和装饰器一样的
        data = {'name': 'wkx', 'age': 123}
        return marshal(data, resouecr_fields,envelope='data')  # ,需要将返回前端的参数给传入到marshal中
​
​
api = Api(app)
api.add_resource(UserInfo, '/')
if __name__ == '__main__':
    app.run(debug=True)
    
    
# envelope的参数意思就是将返回的参数内嵌到一个字典中
{
    "data": {
            "name": "wkx",
            "age": 123,
            "address": null
            }
}   

6.怎么定制响应数据

from flask import make_response, current_app
from flask_restful.utils import PY3
from json import dumps
​
​
# 这个是restful中api中的提供的一个带参数的装饰器,允许定制返回数据的呈现格式
# 需要经过这里转变为json返回给前端
@api.representations('application/json')
def handle_json(data, code, headers):
    '''
    :param data: 接受视图的返回字典数据(需要返回前端)
    :param code: 想返回的状态码
    :param headers: 返回的表头
    :return: 返回给前端的数据
    '''
    # 如果传入的数据,没有带message 就给它包裹一层
    if 'message' not in data:
        data = {
            'message': 'OK',
            'data': data
        }
    # 原代码不动
    settings = current_app.config.get('RESTFUL_JSON', {})
    if current_app.debug:
        settings.setdefault('indent', 4)
        settings.setdefault('sort_keys', not PY3)
    dumped = dumps(data, **settings) + "\n"
    resp = make_response(dumped, code)
    resp.headers.extend(headers or {})
    return resp
​
​
调用原代码 进行原始的封装,该原来的原代码
​