用轻量级ORM--Dapper调用MySQL存储过程

阅读本文你的收获

  1. 写一个MySQL的分页存储过程
  2. 学会使用Dapper去调用存储过程

《用轻量级ORM--Dapper实现泛型仓储》中分享了轻量级ORM框架的基本使用,并实现了一个泛型仓储。本文继续分享用Dapper执行存储过程,区别于上文用的SQL Server数据库,本次我们用MySQL。闲话少絮,直接上案例。

一、场景描述

分页是系统中常用的功能,首先我们在MySQL中创建一个通用的分页存储过程,然后用Dapper去调用分页存储过程,实现用户列表的分页查询。

开发环境:

平台版本是:.NET6

开发框架:ASP.NET Core WebApi

开发工具:Visual Studio 2022

数据库版本:MySQL 5.7

二、在MySQL中创建存储过程

1.创建存储过程

写一个通用分页存储过程,可以进行任何单表的分页查询。

sql 复制代码
-- 0. 如果存在该存储过程,则删除
DROP PROCEDURE IF EXISTS p_pageList;

-- 1. 创建存储过程
CREATE PROCEDURE p_pageList
(
  IN _pageIndex INT,  -- 页码  IN是输入参数
  IN _pageSize  INT,  -- 页的容量
  IN _tableName varchar(30), -- 表的名字
  IN _condition varchar(500), -- 查询条件
  OUT _totalCount INT -- 总条数  OUT是输出参数
)
BEGIN
   -- 用Declare关键词 定义一个局部变量
   DECLARE skip int;
   -- 给skip赋值, 跳过的条数
   set skip = (_pageIndex-1)*_pageSize;

   -- mysql的分页怎么写,用什么关键字??
   -- SELECT SQL_CALC_FOUND_ROWS * FROM users  where xxxx='' LIMIT skip ,_pageSize;
   
   -- 用Concat函数,拼接SQL语句,存入全局变量@sqlStr
   set @sqlStr = CONCAT("SELECT SQL_CALC_FOUND_ROWS "
                        ," * From "
                        , _tableName
                        , " where 1=1 "
                        , _condition
                        , " LIMIT "
                        , skip
                        ,","
                        , _pageSize);
   -- 打印拼接的SQL语句
   -- select @sqlStr;

   -- 执行SQL字符串语句
   PREPARE stmt FROM @sqlStr;  -- 对sql语句进行预处理
   EXECUTE stmt;              -- 执行sql语句
   DEALLOCATE PREPARE stmt;   -- 释放预处理的资源

   -- 在mysql中如何获取总条数,比SQL Server要简单
   SET _totalCount = FOUND_ROWS(); -- FOUND_ROWS()获取总条数

END;

2.测试存储过程

sql 复制代码
-- 调用存储过程
CALL p_pageList(1,3,"users", "and userName like '%张三%'", @total);  -- @开头变量,类似于全局变量
select @total;

三、使用Dapper调用存储过程

1.定义分页数据响应模型

csharp 复制代码
/// <summary>
/// 分页返回模型
/// </summary>
/// <typeparam name="T"></typeparam>
public class PagedList<T>
{
    public List<T> PagedData { get; set; } //分页结果集

    public int TotalCount { get; set; }    //总条数
}

2.定义用户模块的数据库访问接口

csharp 复制代码
//IUserDal.cs

/// <summary>
/// 用户模块的数据库访问接口
/// </summary>
public interface IUserDal
{
    PagedList<User> GetPage(int pageIndex, int pageSize, string name);
}

3.实现IUserDal接口

csharp 复制代码
//UserDal.cs代码如下

using Dapper;  //引用Dapper命名空间 
using System.Data;

/// <summary>
/// 用户的仓储接口实现类
/// </summary>
public class UserDal: IUserDal
{
	//获取连接字符串(本例中是硬编码的,最好写在appsettings.json中)
	private const string CONNECT_STRING = "Server=localhost;Database=testdb1;User=root;Password=123abc!;Port=3306";

	/// <summary>
	/// 用户分页查询
	/// </summary>
	/// <param name="pageIndex">页索引</param>
	/// <param name="pageSize">页容量</param>
	/// <param name="name">姓名查询条件</param>
	/// <returns></returns>
    public PagedList<User> GetUserPage(int pageIndex, int pageSize, string name)
    {
        //定义并添加参数 
        string strWhere = string.Empty;
        if (!string.IsNullOrEmpty(name))
        {
            strWhere += $" and Username like '{name}%'";
        }
        //1. Dapper动态参数赋值:定义DynamicParameters对象
        DynamicParameters paras = new DynamicParameters();
        paras.Add("_tableName", "Users");
        paras.Add("_condition", strWhere);
        paras.Add("_pageIndex", pageIndex);
        paras.Add("_pageSize", pageSize);
        paras.Add("_totalCount", dbType: DbType.Int32,
                                 direction: ParameterDirection.Output);

        //2. 实例化MySql数据库连接对象
        using var dbConnection = new MySqlConnection(CONNECT_STRING );

        PagedList<User> result = new PagedList<User>();
        
        //3. 用Dapper扩展方法来执行存储过程
        result.PagedData = dbConnection.Query<User>("p_pageList",
                                        commandType: CommandType.StoredProcedure, //存储过程
                                        param: paras
                                      ).ToList();
        //获取总条数
        result.TotalCount = paras.Get<int>("_totalCount");

        return result; 
    }
}

剖析以上案例,要点就是:

  1. 创建MySqlConnection连接对象dbConnection,注意要用using结构来确保资源释放;
  2. 用dbConnection对象调用Dapper扩展的方法Query来执行存储过程。其中"p_pageList"是存储过程的名称,commandType请设置为CommandType.StoredProcedure枚举值。给Query方法传参使用DynamicParameters动态参数对象

怎么样?用Dapper执行存储过程,是不是挺简单的?

那今天就分享到这里,如果本文对你有帮助的话,请点赞+评论+关注,或者转发给需要的朋友。

相关推荐
世界哪有真情3 分钟前
Trae 蓝屏问题
前端·后端·trae
Moment6 分钟前
NestJS 在 2025 年:对于后端开发者仍然值得吗 😕😕😕
前端·后端·github
一匹电信狗12 分钟前
【MySQL】数据库表的操作
linux·运维·服务器·数据库·mysql·ubuntu·小程序
Java水解13 分钟前
【Spring】Spring事务和事务传播机制
后端·spring
文心快码BaiduComate13 分钟前
文心快码实测Markdown排版工具开发
前端·后端·程序员
慧都小妮子19 分钟前
基于.NET UA Client SDK构建跨平台OPC UA客户端应用
.net·opc ua·automation·跨平台应用·unified
用户9047066835721 分钟前
java hutool 跟 JSONUtil 的关系
后端
渣哥21 分钟前
原文来自于:[https://zha-ge.cn/java/128](https://zha-ge.cn/java/128)
javascript·后端·面试
渣哥22 分钟前
项目写得再多也没用!Spring Bean 的核心概念要是没懂,迟早踩坑
javascript·后端·面试
白衣鸽子24 分钟前
MySQL 时间类型深度解析:精度、时区陷阱与版本兼容
数据库·后端·mysql