SqlSugar多库/多租户
1、 多库和多租户
如果是接口ISugarClient先看一下标题6,看完在看这儿
1.1 固定多数据库模式
数据库数量是固定的一般在声明的全部加上
//通过ConfigId进行数据库区分 var db = new SqlSugarClient( new List<ConnectionConfig>() { //这儿声名所有上下文都生效 new ConnectionConfig(){ConfigId= "0" ,DbType=DbType.SqlServer,ConnectionString=..,IsAutoCloseConnection= true }, new ConnectionConfig(){ConfigId= "1" ,DbType=DbType.MySql,ConnectionString=..,IsAutoCloseConnection= true } }); //定义实体对应ConfigId ,下面0表示ConfigId为0的数据库 [TenantAttribute( "0" )] //对应ConfigId public class C1Table { public string Id { get ; set ; } } //根据特性直接CRUD var list=db.QueryableWithAttr<Order>().ToList(); //5.0.9.1 全自动切换库查询 var list1=db.AsTenant().QueryableWithAttr<Order>().ToList(); //接口需要AsTenant()转成租户 var list2=db.Queryable<Order>().AsWithAttr().ToList(); //强制名库+表名一般在嵌套或者unionall不能切换库中用 db.InsertWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库插入 db.UpdateWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库更新 db.DeleteableWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库删除 //根据特性获取获取db var childDb=db.GetConnectionWithAttr<C2Table>(); //线程安全用GetConnectionWithAttrScope var list=childDb.Queryable<Order>().ToList(); //手动获取db var childA=db.GetConnection( "0" ); //线程安全用GetConnectionScope var list=childA.Queryable<Order>().ToList(); //事务直接用就行,支持回滚(标题3有原理介绍) |
1.2 动态多数据库模式
就是一个用户对应一个库,或一个企业对应一个数据库,通过后台维护用户和数据库的方式
//当前上下文不存在则添加 if (!db.IsAnyConnection(configId)) //添加一个db到当前上下文 (Add部分不线上下文不会共享) db.AddConnection( new ConnectionConfig(){ DbType = SqlSugar.DbType.SqlServer, ConfigId = "1" , //设置库的唯一标识 IsAutoCloseConnection = true , ConnectionString = Config.ConnectionString2 }); var currentDb=db.GetConnection(configId); //获取当前上下文存在的db //单例SqlSugarScope中用AddConnection和IsAnyConnection多用户不会相互影响,不会共享 |
Saas分库详解:https://www.donet5.com/Home/Doc?typeId=2403
2、详细用例
2.1 子db操作多库 (推荐)
优点灵活,在库的数据不确定的情况下可以使用,比如SAAS分库结合AddConnection和IsAnyConnection使用
主db
SqlSugarClient或者SqlSugarScope我们称为主db
拥租有租户方法:db.BeginTran(事务)、db.GetConnection(获取子Db)等
在多租户中一般只用来 处理事务、创建子DB和获取子DB
可以用 ISqlSugarClien或者ITenant 接收
ISqlSugarClient.AsTenant() 可以转成 ITenant 调用租户方法
子db
通过租户方法GetConnection出来的我们称为子db,没有租户事务相关方法
我们一般用子db去操作指定ConfigId数据库
可以用 ISqlSugarClient 接收,不能转成租户对象 没有租户方法
//主db var db = new SqlSugarClient( new List<ConnectionConfig>() { new ConnectionConfig(){ConfigId= "A" ,DbType=DbType.SqlServer,ConnectionString=..,IsAutoCloseConnection= true }, new ConnectionConfig(){ConfigId= "B" ,DbType=DbType.MySql,ConnectionString=..,IsAutoCloseConnection= true }, new ConnectionConfig(){ConfigId= "C" ,DbType=DbType.MySql,ConnectionString=..,IsAutoCloseConnection= true } }); //获取子Db var childA=db.GetConnection( "A" ); var childB=db.GetConnection( "B" ); var childC=db.GetConnectionScope( "C" ); //线程安全 //使用子Db用 childA.Queryable<Order>().ToList(); childB.Queryable<Order>().ToList(); //事务看标题3 //线程安全 (推荐) //线程安全 |
2.2 根据特性获取(推荐)
适合一个实体和库是一对一的情况
var db = new SqlSugarClient( new List<ConnectionConfig>() { new ConnectionConfig(){ConfigId= "1" ,DbType=DbType.SqlServer,ConnectionString=..,IsAutoCloseConnection= true }, new ConnectionConfig(){ConfigId= "2" ,DbType=DbType.MySql,ConnectionString=..,IsAutoCloseConnection= true } }); //根据特性获取获取db var childDb=db.GetConnectionWithAttr<C2Table>(); childDb.Queryable<Order>().ToList(); //根据特性直接CRUD var list=db.QueryableWithAttr<Order>().ToList(); //5.0.9.1 全自动切换库查询 var list2=db.Queryable<Order>().AsWithAttr().ToList(); //强制名库+表名一般在嵌套或者unionall不能切换库中用 db.InsertWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库插入 db.UpdateWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库更新 db.DeleteableWithAttr(list).ExecuteCommand() ; //5.0.9.1 全自动切换库删除 //如果一个实体对应多个库看SAAS分库文档 //事务看标题3 [TenantAttribute( "1" )] //对应ConfigId public class C1Table { public string Id { get ; set ; } } [TenantAttribute( "2" )] public class C2Table { public string Id { get ; set ; } } |
2.3 通过切换数据库(不推荐)
切换对设计要求过高,很容切了忘记切回来, 特别是一个请求多次切换
var db = new SqlSugarClient( new List<ConnectionConfig>() { //这儿声名所有上下文都生效 new ConnectionConfig(){ConfigId= "0" ,DbType=DbType.SqlServer,ConnectionString=..,IsAutoCloseConnection= true }, new ConnectionConfig(){ConfigId= "1" ,DbType=DbType.MySql,ConnectionString=..,IsAutoCloseConnection= true } }); //使用默认数据库对象 db.Deleteable<Order>().ExecuteCommand(); //切换主db默认值数据库 ConfigId = 1 db.ChangeDatabase( "1" ); //改变db.的默认数据库 db.Deleteable<Order>().ExecuteCommand(); |
注意:只在同一个上下文生效,不同上下文不共享
3、多库事务
支持多库、跨服务器和多种数据库混合使用(老版本db.GetConnection要在事务外声名,然后在事务内用变量)
//开启事务 try { db.BeginTran(); //不能是db.Ado.BeginTran db.GetConnection( "1" ).Insertable( new Order() { CreateTime = DateTime.Now, CustomId = 1, Name = "a" , Price = 1 }).ExecuteCommand(); db.GetConnection( "2" ).Insertable( new Order() { CreateTime = DateTime.Now, CustomId = 1, Name = "a" , Price = 1 }).ExecuteCommand(); //提交事务 db.CommitTran(); //不能是db.ado.CommitTran } catch (Exception ex) { //回滚事务 db.Rollback(); } //主db //注入的SqlSugarClient或者SqlSugarScope我们称为主db //子db //通过租户方法GetConnection出来的我们称为子db,用来操作当前数据库,没有租户事务相关方法 //主db可以用事务管理多个子db ,也可以使用 GetConnection等租户方法 //目前底层是业务执行成功后统一提交事务,业务只要失败全部回滚,统一回滚过程中都有有3次重试回滚 //从目前用户使用情况来看,相当稳定几乎没有一例失败的反馈 //高安全级别数据:请使用差异日志+Catch(AggregateException ex)进行补偿机质 //如果回滚失败会throw new AggregateException |
4、多租户设置AOP
AOP在多租户是不共享的,需要单独设置,满足更多需求,你可以循环添加
//注意: //如果你用的 GetConnectionScope或者 GetConnectionScopeWithAttr AOP也应该用 GetConnectionScope //如果你用的 GetConnection或者 GetConnectionWithAttr AOP也应该用 GetConnectionScope SqlSugarClient Db= new SqlSugarClient( new ConnectionConfig(){ ConnectionString = "连接符字串" , DbType = DbType.SqlServer, IsAutoCloseConnection = true }, db=>{ //也可以这里面循环 db.GetConnection( "1" ).Aop.OnLogExecuting = (sql, pars) => { Console.WriteLine( "执行1库" +sql); }; db.GetConnection( "0" ).Aop.OnLogExecuting = (sql, pars) => { Console.WriteLine( "执行0库" +sql); }; }); |
5、对表进行过滤
db.GetConnection( "A" ).QueryFilter.Add( new TableFilterItem<Order>(it => it.Name.Contains( "a" ), true ) db.GetConnection( "B" ).QueryFilter.Add( new TableFilterItem<Order>(it => it.Name.Contains( "a" ), true ) |
如果要对表进行数据隔离可以看 查询过滤器的例子
https://www.donet5.com/Home/Doc?typeId=1205
6、ISugarClient使用多租户
问题:Db.GetConnection点不出来,出现这种情况一般是用的接口对象ISqlSugarClient
解决方案: Db.AsTenant().GetConnection(1)
原理如下:
ISqlSugarClient和SqlSugarClient不同,ISqlSugarClient不包含租户方法,原因如下
SqlSugarClient : ISqlSugarClient, ITenant //ISqlSugarClient和ITenant是平行关系,没有租户方法 |
我们可以通过自带转换实现
ISqlSugarClient db= 注入db ; db.AsTenant().BeginTran(); db.AsTenant().CommitTran(); db.AsTenant().RollbackTran(); db.AsTenant().GetConnection(1) db.AsTenant().IsAnyConnection(1) //低版本 (db as ITenant).BeginTran() |
SqlSugar多库/多租户的更多相关文章
- FreeSql 将 Saas 租户方案精简到极致[.NET ORM SAAS]
什么是多租户 维基百科:"软件多租户是指一种软件架构,在这种软件架构中,软件的一个实例运行在服务器上并且为多个租户服务".一个租户是一组共享该软件实例特定权限的用户.有了多租户架构 ...
- FreeSql 将 Saas 租户方案精简到极致[.NET ORM]
什么是多租户 维基百科:"软件多租户是指一种软件架构,在这种软件架构中,软件的一个实例运行在服务器上并且为多个租户服务".一个租户是一组共享该软件实例特定权限的用户.有了多租户架构 ...
- .NET 6 跨服务器联表查询
一.大家是否有这个需求 1.跨品种查询 :比如 MYSQL和一个SQLSERVER进行联表查询 ,或者SQLITE和MYSQL进行联表查询 2.跨服务器查询 : A服务器和B服务器查询 如果想同时支持 ...
- Oracle 12c 多租户 CDB 与 PDB 级别 expdb 与 impdb(表、用户、全库)
Oracle 数据库 12 c 多租户下,如何在容器数据库 (CDB) 和可插拔数据库 (PDB) 中使用 expdb 与 impdp (数据泵) 呢? 我们一起探讨下PDB 下进行表级,用户级别,全 ...
- ORM 创新解放劳动力 -SqlSugar 新功能介绍
介绍 SqlSugar是一款 老牌 .NET 开源ORM框架,由果糖大数据科技团队维护和更新 ,Github star数仅次于EF 和 Dapper 优点: 简单易用.功能齐全.高性能.轻量级.服务齐 ...
- .NET SAAS 架构与设计 -SqlSugar ORM
1.数据库设计 常用的Saas分库分为2种类型的库 1.1 基础信息库 主要存组织架构 .权限.字典.用户等 公共信息 性能优化:因为基础信息库是共享的,所以我们可以使用 读写分离,或者二级缓存来进行 ...
- .NET 开源SqlServer ORM框架 SqlSugar 3.0 API
3.1.x ,将作为3.X系统的最后一个版本,下面将会开发 全新的功能 更新列表:https://github.com/sunkaixuan/SqlSugar/releases 优点: SqlSuga ...
- 【Network】OVS、VXLAN/GRE、OVN等 实现 Docker/Kubernetes 网络的多租户隔离
多租户隔离 DragonFlow与OVN | SDNLAB | 专注网络创新技术 Neutron社区每周记(6.25~7.6)| OVS将加入Linux基金会?OVN或抛弃ovsdb? | Unite ...
- 【开源框架】SqlSugarRepository 全库ORM 正式发布
SqlSugarRepository.dll 全库开发框架支持 四种数据库:SqlServer. MySql .Oracle和Sqlite. SqlSugarRepository是为全库开发而生的OR ...
- SqlSugar ORM已经支持读写分离
目前只有MYSQL版 3.5.2.9 支持,其库版本12月3号更新该功能 用例讲解 using (var db = new SqlSugarClient("主连接字符串", &qu ...
随机推荐
- 【JAVA基础】数值处理
#BigDecimal处理 ##保留两位小数 https://www.cnblogs.com/jpfss/p/8072379.html /** * 保留两位小数 */ @org.junit.Test ...
- SpringBoot 项目实战 | 瑞吉外卖 Day05
该系列将记录一份完整的实战项目的完成过程,该篇属于第五天 案例来自B站黑马程序员Java项目实战<瑞吉外卖>,请结合课程资料阅读以下内容 该篇我们将完成以下内容: 新增套餐 套餐信息分页查 ...
- 供应链安全情报 | 恶意py包伪装代理SDK进行后门攻击,目标锁定python开发者
概述 2023年11月28号,悬镜供应链安全实验室在Pypi官方仓库(https://pypi.org)监测到两起伪装成http和socks5代理SDK的开源组件投毒事件.python开发者一旦下载安 ...
- 实战指南 | Serverless 架构下的应用开发
作者 | 刘宇.田初东.卢萌凯.王仁达 UC Berkeley认为Serverless架构的出现过程类似于40多年前从汇编语言转向高级语言的过程,在未来Serverless架构的使用会飙升,或许服务器 ...
- 从0开发3D引擎(十三):使用领域驱动设计,从最小3D程序中提炼引擎(第四部分)
目录 上一篇博文 下一篇博文 本文流程 回顾上文 解释基本的操作 如何在浏览器上运行index.html 开始实现 准备 建立代码的文件夹结构,约定模块文件的命名规则 模块文件的命名原则 一级和二级文 ...
- uni-app打包到安卓步骤
1.打包到安卓 https://www.bilibili.com/video/BV1BJ411W7pX?p=56 2.打包到IOS https://www.jianshu.com/p/ef6e6e01 ...
- Vue2知识点简要
一.双向绑定原理 Vue2采用的是观察者-发布订阅模式,利用Object.defineProperty实现对数据已定义属性的监控(定义观察者模式), 编译DOM时解析v-model等属性以及对inpu ...
- 时空图预测的方法论,以及 diffusion model 基本概念
前天和善良的同学聊天,请教了 ① 时空图预测的方法论,② diffusion model 基本概念,记录下来. ① 时空图预测的方法论: 首先,构造 0 ~ t-1 时刻的 t 个图,每个图都有 n ...
- MAUI使用Masa blazor组件库
上一篇(点击阅读)我们实现了UI在Web端(Blazor Server/Wasm)和客户端(Windows/macOS/Android/iOS)共享,这篇我加上 Masa Blazor组件库的引用,并 ...
- SpringBoot实现限流注解
SpringBoot实现限流注解 在高并发系统中,保护系统的三种方式分别为:缓存,降级和限流. 限流的目的是通过对并发访问请求进行限速或者一个时间窗口内的的请求数量进行限速来保护系统,一旦达到限制速率 ...