learn from
https://fastapi.tiangolo.com/zh/tutorial/
1. 第一步
from fastapi import FastAPI
my_app = FastAPI()
@my_app.get("/")
async def root():
return {"message" : "Hello World"}
data:image/s3,"s3://crabby-images/ce1c9/ce1c9f182846e3d59b1b520833c0f7020b993a31" alt=""
http 操作: POST:创建数据。 GET:读取数据。 PUT:更新数据。 DELETE:删除数据。
@my_app.get("/") 告诉 FastAPI 在它下方的函数负责处理如下访问请求:
函数可以返回一个 dict 、list ,像 str 、int 一样的单个值,等等。还可以返回 Pydantic 模型
1.1 小结
- 导入
FastAPI - 创建一个 app 实例
- 编写一个路径操作装饰器(如
@app.get("/") ) - 编写一个路径操作函数(如上面的
def root(): ... ) - 运行开发服务器(如
uvicorn main:app --reload )
2. 路径参数
@my_app.get("/items/{item_id}")
async def read_item(item_id):
return {"itemid": item_id}
@my_app.get("/items/{item_id}")
async def read_item(item_id: int):
return {"itemid": item_id}
- 文档
http://127.0.0.1:8000/docs ,http://127.0.0.1:8000/redoc
2.1 顺序很重要
@my_app.get("/users/me")
async def read_user_me():
return {"user_id": "the current user"}
@my_app.get("/users/{user_id}")
async def read_user(user_id: str):
return {"user_id": user_id}
data:image/s3,"s3://crabby-images/56fa4/56fa40117c3c484b91d9b1b74de8daeedda251b0" alt="在这里插入图片描述" 如果上面,两个函数顺序反了,如下结果
data:image/s3,"s3://crabby-images/2c113/2c11360d246700f22b51a9f170b321d68f8a27a7" alt="在这里插入图片描述"
2.2 预设值
from enum import Enum
app = FastAPI()
class ModelName(str, Enum):
alexnet = "alexnet"
resnet = "resnet"
lenet = "lenet"
@app.get("/models/{model_name}")
async def get_model(model_name: ModelName):
if model_name == ModelName.alexnet:
return {"model_name": model_name, "message": "Deep Learning FTW!"}
if model_name.value == "lenet":
return {"model_name": model_name, "message": "LeCNN all the images"}
return {"model_name": model_name, "message": "Have some residuals"}
可以使用 model_name.value 或通常来说 your_enum_member.value 来获取实际的值 data:image/s3,"s3://crabby-images/a2400/a2400ada40bdee94d38ef993d9a89b9699e2d68b" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/34502/34502a6e7b35ad0be375c9e0b140e7387c86eb63" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/2d77f/2d77f10afb514468884f51cc13f2e2a35f8d8e05" alt="在这里插入图片描述"
2.3 包含路径的路径参数
- 参数 { } 内
参数名:path : 前后均没有空格,不加 :path 无法识别 带有/ 的路径参数
@app.get("/files/{file_path:path}")
async def read_file(file_path: str):
return {"file_path": file_path}
3. 查询参数
fake_items_db = [{"item_name": "Foo"}, {"item_name": "Bar"}, {"item_name": "Baz"}]
@app.get("/items")
async def read_item(skip: int = 0, limit: int = 10):
return fake_items_db[skip:skip + limit]
- 使用
? 开始,参数间使用 & 分割 data:image/s3,"s3://crabby-images/21e06/21e06d951e25b320ac130ac2f0791a983b65e535" alt="在这里插入图片描述"
from typing import Optional
@app.get("/items/{item_id}")
async def read_item(item_id: str, q: Optional[str] = None):
if q:
return {"item_id": item_id, "q": q}
return {"item_id": item_id}
data:image/s3,"s3://crabby-images/ffa91/ffa91ae5232c91d427d96b3efcef9c38e0b7dc9a" alt=""
3.1 查询参数类型转换
from typing import Optional
@app.get("/items/{item_id}")
async def read_item(item_id: str, q: Optional[str] = None, short: bool = False):
item = {"item_id" : item_id}
if q:
item.update({"q" : q})
if not short:
item.update(
{"description": "This is an amazing item that has a long description"}
)
return item
输入 short= ,后面是 1,True, true, yes, on, On, YES 任意变形体都是一样的效果,都是 true data:image/s3,"s3://crabby-images/e7a15/e7a15d3d28a91d640faae4790598c3bfb8c32601" alt="在这里插入图片描述"
@app.get("/users/{user_id}/items/{item_id}")
async def read_user_item(
item_id: str, user_id: int, q: Optional[str] = None, short: bool = False
):
item = {"item_id": item_id, "owner_id": user_id}
if q:
item.update({"q": q})
if not short:
item.update(
{"description": "This is an amazing item that has a long description"}
)
return item
item_id: str, user_id: int, 更换两个变量的位置,没有关系 data:image/s3,"s3://crabby-images/4ae7a/4ae7a521d08b7050bdc91792f7df8d2809acefc5" alt="在这里插入图片描述"
4. 请求体
请求体是客户端发送给 API 的数据 响应体是 API 发送给客户端的数据
使用 Pydantic 模型 来声明请求体
from typing import Optional
from Pinyin2Hanzi import Item
from fastapi import FastAPI
from pydantic import BaseModel
class Item(BaseModel):
name: str
description: Optional[str]=None
price:float
tax:Optional[float] = None
app = FastAPI()
@app.put("/items/{item_id}")
async def create_item(item_id: int, item: Item, q: Optional[str] = None):
result = {"item_id": item_id, **item.dict()}
if q:
result.update({"q": q})
return result
data:image/s3,"s3://crabby-images/2f87a/2f87a0bf59526d1e8570022cbdec92f96cf1bdd9" alt=""
- 还可以同时声明请求体、路径参数和查询参数。
函数参数将依次按如下规则进行识别: 1.如果在路径中也声明了该参数,它将被用作路径参数 2.如果参数属于单一类型(比如 int、float、str、bool 等)它将被解释为查询参数 3.如果参数的类型被声明为一个 Pydantic 模型,它将被解释为请求体
|