✨✨ 欢迎大家来到景天科技苑✨✨

养成好习惯,先赞后看哦~

所属专栏:Fastapi
景天的主页:景天科技苑

上一章,我们讲到了fastapi数据库操作ORM的配置和查询操作,感兴趣的小伙伴可以回顾下:
Python web框架fastapi数据库操作ORM(一)

今天,我们一起探讨一下fastapi的增删改查逻辑实现

文章目录

  • ORM数据库增删改查操作
    • 1.添加单条记录
      • 1.添加方式一
      • 2.添加方式二
    • 2.多对多表添加记录
      • 1.一对多查询
      • 2.多对多查询
    • 3.更新学生信息接口
    • 4.删除学生接口

ORM数据库增删改查操作

需要导入的包
from pydantic import BaseModel

from typing import Union,List

#保存数据到数据库,对用户输入数据进行校验
class StudentIn(BaseModel):
name: str
pwd: str
sno: int
clas_id: int
courses: List[int] = []

1.添加单条记录

api接口实现

1.添加方式一

@api_student.post("/",tags=["添加学生"])async def addStudent(stu: StudentIn):# 添加数据库操作# 方式1student = Student(name=stu.name, pwd=stu.pwd, sno=stu.sno, clas_id=stu.clas_id)#插入数据库动作await student.save()print(student, dir(student))return student

接口测试

查看数据库,数据已入库

2.添加方式二

# 方式2student = await Student.create(name=stu.name, pwd=stu.pwd, sno=stu.sno, clas_id=stu.clas_id)print(student, dir(student))

数据库查看,添加成功

2.多对多表添加记录

由于多对多那张表,没有对应的模型类,所以一般无法通过模型类直接添加删除
需要借助学生表的多对多字段courses来处理

# 添加多对多关系记录courses = await Course.filter(id__in=stu.courses)print("courses", courses)#添加的时候,要用*打散的方式传参await student.courses.add(*courses)print("student", student.courses)

查看数据库

可以看到id 13 对15和16,多对多插入成功

1.一对多查询

#一对多查询student = await Student.get(name="wangfang")#直接通过外键字段查询,班级名称clasname = await student.clas.values('name')print("班级名称",clasname)#查询所有学生的班级名称,通过外键__字段值students = await Student.all().values("name","clas__name")print(students)

2.多对多查询

#多对多查询,根据某个学生查询报的班名称和老师名称student = await Student.get(name="wangfang")print(await student.courses.all().values('name','teacher__name'))#查询所有学生的课程名称和老师名称students = await Student.all().values("name", "clas__name",'courses__name')print(students)

查询某一个学生的接口

#查看某个学生,基于路径参数@api_student.get("/{student_id}")async def getOneStudent(student_id:int):#注意,与数据库的操作要加awaitstudent = await Student.get(id=student_id)print(student.name)return student


3.更新学生信息接口

#更新学生信息,使用put请求@api_student.put("/student/{student_id}")async def update_student(student_id: int, student: StudentIn):#将对象转换为字典data = student.dict(exclude_unset=True)#将courses字段排除,因为student表中没有courses字段courses = data.pop("courses")print(data, courses)#更新用filter方法。因为filter查出来的对象有update方法await Student.filter(id=student_id).update(**data)#设置多对多课程courses = await Course.filter(id__in=student.courses)edit_student = await Student.get(id=student_id)#先把原来课程清空,再添加更新后的await edit_student.courses.clear()await edit_student.courses.add(*courses)return student

先看下源数据库13号学生信息

多对多表

在接口执行更新

更新后

多对多表

4.删除学生接口

导入异常包
from fastapi.exceptions import HTTPException
该异常类可以设置响应状态码,描述信息等

@api_student.delete("/student/{student_id}")async def delete_student(student_id: int):#返回删除的数量deleted_count = await Student.filter(id=student_id).delete()# 条件删除#输入的id不存在删除报错if not deleted_count:raise HTTPException(status_code=404, detail=f"Student {student_id} not found")#能正产给删除,返回空return {}

删除对应学生的时候,也会级联删除对应的多对多表中数据
我们删除13号学生

执行删除请求

查看数据库,13号学生已被删除

查看多对多表13号也被删除

完整代码:

from fastapi import Requestfrom fastapi.exceptions import HTTPException#导入模板的包from fastapi.templating import Jinja2Templates# 实例化Jinja2对象,并将文件夹路径设置为以templates命名的文件夹templates = Jinja2Templates(directory="templates")from pydantic import BaseModelfrom typing import Union,List#导入modelsfrom models import *from fastapi import APIRouterapi_student = APIRouter()#保存数据到数据库,对用户输入数据进行校验class StudentIn(BaseModel):name: strpwd: strsno: intclas_id: intcourses: List[int] = []#查看所有学生@api_student.get("/")async def getAllStudent():#注意,与数据库的操作要加await,得到的是queryset类型数据,[Student(),Student(),Student()....]# students = await Student.all()# print('students',students,type(students))## #循环打印# for stu in students:# print(stu.name, stu.sno)#过滤查询,查询指定内容filter,得到的依然是list类型数据# student = await Student.filter(name='liuxin')# print(student,type(student))# #得到具体数据# print(student[0].name)#get方法# student = await Student.get(name="wangfang")# print(student,type(student))# print(student.name,student.sno)#模糊查询,查询学号大于2001的学生# students = await Student.filter(sno__gt=2001)#查询学号是2001和2002的学生# students = await Student.filter(sno__in=[2001,2002])# print(students)#values查询,只查出指定字段数据# students = await Student.filter(sno__range=[1, 10000]).values('name','sno')# print(students)# for stu in students:# print(stu)#一对多查询# student = await Student.get(name="wangfang")## #直接通过外键字段查询,班级名称# clasname = await student.clas.values('name')# print("班级名称",clasname)## #查询所有学生的班级名称,通过外键__字段值# students = await Student.all().values("name","clas__name")# print(students)#多对多查询,根据某个学生查询报的班名称和老师名称student = await Student.get(name="wangfang")print(await student.courses.all().values('name','teacher__name'))#查询所有学生的课程名称和老师名称students = await Student.all().values("name", "clas__name",'courses__name')print(students)return students#将学生信息显示到页面@api_student.get("/index")async def show_student(request:Request):students = await Student.all()return templates.TemplateResponse('index.html', #第一个参数放模板文件{'request': request,# 注意,返回模板响应时,必须有request键值对,且值为Request请求对象'students':students}, #context上下文对象,是个字典)#查看某个学生,基于路径参数@api_student.get("/{student_id}")async def getOneStudent(student_id:int):#注意,与数据库的操作要加awaitstudent = await Student.get(id=student_id)print(student.name)return student#更新学生信息,使用put请求@api_student.put("/student/{student_id}")async def update_student(student_id: int, student: StudentIn):#将对象转换为字典data = student.dict(exclude_unset=True)#将courses字段排除,因为student表中没有courses字段courses = data.pop("courses")print(data, courses)#更新用filter方法。因为filter查出来的对象有update方法await Student.filter(id=student_id).update(**data)#设置多对多课程courses = await Course.filter(id__in=student.courses)edit_student = await Student.get(id=student_id)#先把原来课程清空,再添加更新后的await edit_student.courses.clear()await edit_student.courses.add(*courses)return student@api_student.delete("/student/{student_id}")async def delete_student(student_id: int):#返回删除的数量deleted_count = await Student.filter(id=student_id).delete()# 条件删除#输入的id不存在删除报错if not deleted_count:raise HTTPException(status_code=404, detail=f"Student {student_id} not found")#能正产给删除,返回空return {}@api_student.post("/",tags=["添加学生"])async def addStudent(stu: StudentIn):# 添加数据库操作# 方式1# student = Student(name=stu.name, pwd=stu.pwd, sno=stu.sno, clas_id=stu.clas_id)# #插入数据库动作# await student.save()# print(student, dir(student))# 方式2student = await Student.create(name=stu.name, pwd=stu.pwd, sno=stu.sno, clas_id=stu.clas_id)print(student, dir(student))# 添加多对多关系记录courses = await Course.filter(id__in=stu.courses)print("courses", courses)#添加的时候,要用*打散的方式传参await student.courses.add(*courses)print("student", student.courses)return courses