域名 AXUM.RS 将于2025年10月到期。我们无意再对其进行续费,我们希望你能够接续这个域名,让更多 AXUM 开发者继续受益。
  • 方案1️⃣AXUM.RS 域名 = 3000
  • 方案2️⃣方案1️⃣ + 本站所有专题原始 Markdown 文档 = 5000
  • 方案3️⃣方案2️⃣ + 本站原始数据库 = 5500
如果你有意接续这份 AXUM 情怀,请与我们取得联系。
说明:
  1. 如果有人购买 AXUM.RS 域名(方案1️⃣),或者该域名到期,本站将启用新的免费域名继续提供服务。
  2. 如果有人购买了 AXUM.RS 域名,且同时购买了内容和/或数据库(方案2️⃣/方案3️⃣),本站将关闭。届时我们或许会以另一种方式与你再相遇。

使用 SeaORM 查询数据

本章我们将讨论如何使用 SeaORM 进行查询:查询条件、分页、查询单条记录等。

category-paginate

准备数据

为了更好地演示分页效果,我们通过 SQL 语句添加一些示例数据:

ALTER SEQUENCE categoies_id_seq RESTART WITH 4;
INSERT INTO categoies (name) VALUES
('U3A0CsWdiy'),
('SWACTQFa0Y'),
('GYqfhaKJ6J'),
('0sjsXVArdZ'),
('MiN8lR1g9B'),
('oBorPeyIvH'),
('cqS4jGnmxG'),
('dc0qqvbDNP'),
('jq8K6LgUFy'),
('K1tKtlvzgf'),
('Z5kEYZYEdp'),
('y3K6ryqRMF'),
('hwPu60bq1u'),
('2Idzt9CmAV'),
('vbLGfMJNHz'),
('6tTPkRtpWB'),
('sWBfrpOAIB'),
('zgmXGcYsGt'),
('WH2EBpojIS'),
('m1rsNTknqS');

接收参数

为了让我们的分类列表能接收到各种参数,我们需要进行如下操作:

定义参数结构体

// src/param.rs

#[derive(Debug, Clone, Deserialize)]
pub struct CategoryParams {
    pub keyword: Option<String>, // 用于搜索的关键字
    pub is_del: Option<i32>, // 是否删除
    pub sort: Option<String>, // 排序
    pub page_size: Option<usize>, // 每页记录条数
    pub page: Option<usize>, // 当前页码
}
// ...

在handler中接收参数

// src/handler/category.rs
pub async fn index(
    Extension(state): Extension<Arc<AppState>>,
    Query(params): Query<param::CategoryParams>,
) -> Result<HtmlRespon> {
  // ...
}

模板

<!-- templates/category.html -->
<form class="row" method="get" action="/category">
    <div class="col-auto">
    <label class="visually-hidden" for="keyword">关键字</label>
    <div class="input-group">
      <div class="input-group-text">关键字</div>
      <input type="text" class="form-control" id="keyword" name="keyword" placeholder="输入关键字" value="{{ params.keyword() }}">
    </div>
  </div>
  <div class="col-auto">
    <label class="visually-hidden" for="is_del">是否删除</label>
    <select class="form-select" id="is_del" name="is_del">
        <option value="-1"{% if params.is_del() == -1%} selected{%endif%}>全部</option>
      <option value="0"{% if params.is_del() == 0%} selected{%endif%}>未删除</option>
      <option value="1"{% if params.is_del() == 1%} selected{%endif%}>已删除</option>
    </select>
  </div>
  <div class="col-auto">
    <label class="visually-hidden" for="sort">排序</label>
    <select class="form-select" id="sort" name="sort">
        <option value=""{% if params.sort().is_empty() %} selected{%endif%}>默认排序</option>
      <option value="asc"{% if params.sort() == "asc" %} selected{%endif%}>升序</option>
      <option value="desc"{% if params.sort() == "desc" %} selected{%endif%}>降序</option>
    </select>
  </div>
  <div class="col-auto">
    <label class="visually-hidden" for="page_size">每页条数</label>
    <select class="form-select" id="page_size" name="page_size">
        <option value="0"{% if params.page_size() == 0 %} selected{%endif%}>默认条数</option>
        <option value="3"{%if params.page_size() == 3 %} selected{%endif%}>每页3条</option>
      <option value="5"{%if params.page_size() == 5 %} selected{%endif%}>每页5条</option>
      <option value="10"{%if params.page_size() == 10 %} selected{%endif%}>每页10条</option>
    </select>
  </div>
  <div class="col-auto">
    <button type="submit" class="btn btn-primary"><i class="bi bi-search"></i> 搜索</button>
  </div>
</form>

SeaORM 提供多过滤数据的方式,对于多条记录而言,filter()是最常用的方法。

官方文档

官方文档

简单查询

category::Entity::find()
	.filter(category::Column::IsDel.eq(false))
// SELECT * FROM categories WHERE is_del=false

模糊查询

比如,要查询名字中带有a的分类:

category::Entity::find()
	.filter(category::Column::Name.contains("a"))
// SELECT * FROM categories WHERE name LIKE '%a%'

组合过滤

我们可以使用Condition来组合复杂的查询条件, 然后将其放到 filter()里。比如,要查询所有未删除的、并且名字中带有a的分类:

category::Entity::find()
	.filter(
		Condition.all()
    	.add(category::Column::IsDel.eq(false))
    	.add(category::Column::Name.contains("a"))
)
// SELECT * FROM categories WHERE is_del=false AND name LIKE '%a%'

可选过滤

let name: Option<String> = Some(String::from("a"));
category::Entity::find()
	.filter(
		Condition.all()
    	.add_option(
    		name.map(|n| category::Column::Name.contains(&n))
    	)
)
// -- 当 name 是 None 时:
// SELECT * FROM categories
// -- 当 name 是 Some("a") 时:
// SELECT * FROM categories WHERE is_del=false AND name LIKE '%a%'

组合可选过滤

同样的,我们也可以组合多个可选过滤

let name: Option<String> = Some(String::from("a"));
let is_del:Option<bool> = Some(false);
category::Entity::find()
	.filter(
		Condition.all()
    	.add_option(
    		name.map(|n| category::Column::Name.contains(&n))
    	)
    	.add_option(
    		is_del.map(|n| category::Column::IsDel.eq(n))
    	)
)

可以使用 order_by(字段,排序方式)来对数据进行排序。其中的 字段通常是一个Column,比如category::Column::Id,而 排序方式Order枚举:

  • Order::Asc:升序
  • Order::Desc:降序

比如,要让所有记录按id的降序排列:

category::Entity::find()
	.order_by(category::Column::Id, Order::Desc)
// SELECT * FROM categoies ORDER BY id DESC

由于ASCDESC太常见了,所以 SeaORM 直接提供了两个方法来简化操作:

  • order_by_asc(字段):即 order_by(字段, Order::Asc)
  • order_by_desc(字段):即 order_by(字段, Order:Desc)

分页 - paginate()

SeaORM 很贴心的提供了 paginate(conn,page_size)->Paginator方法,方便地进行分页。

它的conn 参数是实现了ConnectionTrait的对象,我们的项目中使用的是 DatabaseConnectionsrc/state.rs),它实现了这个ConnectionTrait。而 page_size是次获取记录的条数。

它的返回值是一个Paginator对象,它提供了诸多有用的方法:

  • async num_pages():获取分页的总页数
  • async num_items():获取总的记录数
  • async fetch_page(page):获取第page页的记录,page是从0开始算的。

简单分页

有了这些知识的铺垫,现在来看一下如何实现一个简单的分页:

let page:usize = 0;
let page_size = 15usize;
let paginator = category::Entity::find().paginate(conn, page_size);
let page_total = paginator.num_pages().await.unwrap();
let categies: Vec<category::Model> = paginator
        .fetch_page(page)
        .await
				.unwrap();

复杂分页

配合前面讨论的组合过滤、排序等,可以实现复杂的分页:

src/handler/category.rs

src/handler/category.rs

pub async fn index(
    Extension(state): Extension<Arc<AppState>>,
    Query(params): Query<param::CategoryParams>,
) -> Result<HtmlRespon> {
    let handler_name = "category/index";
    let page = params.page(); // 当前页码
    let page_size = params.page_size(); // 每页条数,默认15
    let conn = get_conn(&state);
    let mut selc = category::Entity::find().filter(
        Condition::all()
            .add_option(
                params
                    .keyword_opt()
                    .map(|n| category::Column::Name.contains(&n)),
            )
            .add_option(params.is_del_opt().map(|n| category::Column::IsDel.eq(n))),
    );
    if let Some(ord) = params.order() {
        selc = selc.order_by(category::Column::Id, ord);
    };
    let paginator = selc.paginate(conn, page_size);
    let page_total = paginator.num_pages().await.map_err(AppError::from)?;
    let categies: Vec<category::Model> = paginator
        .fetch_page(page)
        .await
        .map_err(AppError::from)
        .map_err(log_error(handler_name))?;
    let tpl = view::CategoryTemplate {
        categies,
        params,
        page_total,
    };
    render(tpl, handler_name)
}

模板中显示分页链接

<!-- templates/category.html  -->
<nav>
  <ul class="pagination pagination">
    {% for page in 0..page_total %} 
        {% if page == params.page() %}
    <li class="page-item active">
        <span class="page-link">{{ page + 1}}</span>
    </li>
        {% else %}
    <li class="page-item">
        <a class="page-link" href="?page={{page}}&keyword={{params.keyword()}}&is_del={{params.is_del()}}&sort={{params.sort()}}&page_size={{params.page_size()}}">{{ page + 1}}</a>
        </li>
        {% endif %}
    {% endfor %}
  </ul>
</nav>

查询单条数据

let cate: Option<category::Model> = category::Entity::find()
	.filter(category::Column::Id.eq(1))
	.one()
	.await
	.unwrap();

对于通过主键来查询,我们可以使用更简单的方法 find_by_id()

src/handler/category.rs

src/handler/category.rs

pub async fn find(
    Extension(state): Extension<Arc<AppState>>,
    Path(id): Path<i32>,
) -> Result<String> {
    let handler_name = "category/find";
    let conn = get_conn(&state);
    let cate: Option<category::Model> = category::Entity::find_by_id(id)
        .one(conn)
        .await
        .map_err(AppError::from)
        .map_err(log_error(handler_name))?;
    match cate {
        Some(cate) => Ok(format!("id: {}, 名称: {}", cate.id, cate.name)),
        None => Err(AppError::notfound()),
    }
}

本章代码位于02/查询数据分支。

要查看完整内容,请先登录