子依赖¶
您可以创建具有子依赖的依赖项。
它们可以根据您的需要深度嵌套。
FastAPI 将负责解决它们。
第一个依赖 “dependable”¶
您可以创建第一个依赖项(“dependable”)如下:
from typing import Annotated
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[str | None, Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
🤓 其他版本和变体
from typing import Annotated, Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[Union[str, None], Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor), last_query: str | None = Cookie(default=None)
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from typing import Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor),
last_query: Union[str, None] = Cookie(default=None),
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
它声明了一个可选的查询参数 q,类型为 str,然后直接返回它。
这很简单(不是很有用),但有助于我们关注子依赖项的工作方式。
第二个依赖,“dependable” 和 “dependant”¶
然后您可以创建另一个依赖项函数(一个“dependable”),该函数同时声明了自己的依赖项(因此它也是一个“dependant”)
from typing import Annotated
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[str | None, Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
🤓 其他版本和变体
from typing import Annotated, Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[Union[str, None], Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor), last_query: str | None = Cookie(default=None)
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from typing import Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor),
last_query: Union[str, None] = Cookie(default=None),
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
让我们关注声明的参数
- 尽管此函数本身是一个依赖项(“dependable”),但它也声明了另一个依赖项(它“依赖”于其他内容)。
- 它依赖于
query_extractor,并将query_extractor返回的值赋给参数q。
- 它依赖于
- 它还声明了一个可选的
last_querycookie,类型为str。- 如果用户没有提供查询
q,我们将使用上次使用的查询,该查询已保存在 cookie 中。
- 如果用户没有提供查询
使用依赖¶
然后我们可以使用依赖项:
from typing import Annotated
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[str | None, Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
🤓 其他版本和变体
from typing import Annotated, Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: Annotated[str, Depends(query_extractor)],
last_query: Annotated[Union[str, None], Cookie()] = None,
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(
query_or_default: Annotated[str, Depends(query_or_cookie_extractor)],
):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: str | None = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor), last_query: str | None = Cookie(default=None)
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
提示
如果可能,请优先使用 Annotated 版本。
from typing import Union
from fastapi import Cookie, Depends, FastAPI
app = FastAPI()
def query_extractor(q: Union[str, None] = None):
return q
def query_or_cookie_extractor(
q: str = Depends(query_extractor),
last_query: Union[str, None] = Cookie(default=None),
):
if not q:
return last_query
return q
@app.get("/items/")
async def read_query(query_or_default: str = Depends(query_or_cookie_extractor)):
return {"q_or_cookie": query_or_default}
信息
请注意,我们只在路径操作函数中声明了一个依赖项:query_or_cookie_extractor。
但是,FastAPI 将知道它必须先解决 query_extractor,然后才能在调用 query_or_cookie_extractor 时将 query_extractor 的结果传递给它。
graph TB
query_extractor(["query_extractor"])
query_or_cookie_extractor(["query_or_cookie_extractor"])
read_query["/items/"]
query_extractor --> query_or_cookie_extractor --> read_query
多次使用同一个依赖¶
如果您的一个依赖项在同一个路径操作中被声明了多次,例如,多个依赖项有一个共同的子依赖项,FastAPI 将知道每请求只调用该子依赖项一次。
它将在请求中将返回的值保存在一个“缓存”中,并将其传递给需要它的所有“dependant”,而不是在同一个请求中多次调用该依赖项。
在一种高级场景中,如果您知道需要在同一个请求中的每个步骤(可能多次)调用该依赖项,而不是使用“缓存”的值,您可以在使用 Depends 时设置参数 use_cache=False。
async def needy_dependency(fresh_value: Annotated[str, Depends(get_value, use_cache=False)]):
return {"fresh_value": fresh_value}
提示
如果可能,请优先使用 Annotated 版本。
async def needy_dependency(fresh_value: str = Depends(get_value, use_cache=False)):
return {"fresh_value": fresh_value}
总结¶
除了这里使用的所有花哨的词语外,依赖注入系统非常简单。
只是函数,它们看起来与路径操作函数相同。
但是,它仍然非常强大,允许您声明任意深度嵌套的依赖项“图”(树)。
提示
使用这些简单的示例,所有这些可能看起来没那么有用。
但您将在关于安全性的章节中看到它有多么有用。
您还将看到它将为您节省多少代码。