类别:学习文档
日期:2021-07-15 浏览:2281 评论:0
理论知识
分表 - 从表面意思上看呢,就是把一张表分成N多个小表,每一个小表都是完正的一张表。分表后数据都是存放在分表里,总表只是一个外壳,存取数据发生在一个一个的分表里面。分表后单表的并发能力提高了,磁盘I/O性能也提高了。并发能力为什么提高了呢,因为查寻一次所花的时间变短了,如果出现高并发的话,总表可以根据不同 的查询,将并发压力分到不同的小表里面。
分库 - 把原本存储于一个库的数据分块存储到多个库上,把原本存储于一个表的数据分块存储到多个表上。数据库中的数据量不一定是可控的,在未进行分表分库的情况下,随着时间和业务的发展,库中的表会越来越多,表中的数据量也会越来越大,相应地,数据操作,增删改查的开销也会越来越大;另外,一台服务器的资源(CPU、磁盘、内存、IO等)是有限的,最终数据库所能承载的数据量、数据处理能力都将遭遇瓶颈。
分表 AsTable
FreeSql 原生用法、FreeSql.Repository 仓储用法 都提供了 AsTable 方法对分表进行 CRUD 操作,例如:
var repo = fsql.GetRepository<Log>(); repo.AsTable(oldname => $"{oldname}_201903"); //对 Log_201903 表 CRUDrepo.Insert(new Log { ... });
跨库,但是在同一个数据库服务器下,也可以使用 AsTable(oldname => $"db2.dbo.{oldname}")
//跨表查询var sql = fsql.Select<User>() .AsTable((type, oldname) => "table_1") .AsTable((type, oldname) => "table_2") .ToSql(a => a.Id);//select * from (SELECT a."Id" as1 FROM "table_1" a) ftb //UNION ALL//select * from (SELECT a."Id" as1 FROM "table_2" a) ftb
分表总结:
分表、相同服务器跨库 可以使用 AsTable 进行 CRUD;
AsTable CodeFirst 会自动创建不存在的分表;
不可在分表分库的实体类型中使用《延时加载》;
SqlServer 跨库事务 可以使用 TransactionScope,如下:
var repoLog = fsql.GetRepository<Log>();var repoComment = fsql.GetRepository<Comment>(); repoLog.AsTable(oldname => $"{201903}.dbo.{oldname}"); repoComment.AsTable(oldname => $"{201903}.dbo.{oldname}");using (TransactionScope ts = new TransactionScope()) { repoComment.Insert(new Comment { ... }); repoLog.Insert(new Log { ... }); ts.Complete(); }
分库 IdleBus
IFreeSql 对应一个数据库,分库是不是要定义N个 IFreeSql?分库的租户场景,那不要定义10000个?
IdleBus 空闲对象管理容器,有效组织对象重复利用,自动创建、销毁,解决【实例】过多且长时间占用的问题。有时候想做一个单例对象重复使用提升性能,但是定义多了,有的又可能一直空闲着占用资源。专门解决:又想重复利用,又想少占资源的场景。https://github.com/2881099/IdleBus
dotnet add package IdleBus
static IdleBus<IFreeSql> ib = new IdleBus<IFreeSql>(TimeSpan.FromMinutes(10)); ib.Register("db1", () => new FreeSqlBuilder().UseConnectionString(DataType.MySql, "str1").Build()); ib.Register("db2", () => new FreeSqlBuilder().UseConnectionString(DataType.MySql, "str2").Build()); ib.Register("db3", () => new FreeSqlBuilder().UseConnectionString(DataType.SqlServer, "str3").Build());//...注册很多个ib.Get("db1").Select<T>().Limit(10).ToList();
IdleBus 也是【单例】设计!主要的两个方法,注册,获取。使用 IdleBus 需要弱化 IFreeSql 的存在,每次使用 ib.Get 获取。
public static class IdleBusExtesions{ static AsyncLocal<string> asyncLocalTenantId = new AsyncLocal<string>(); public static IdleBus<IFreeSql> ChangeTenant(this IdleBus<IFreeSql> ib, string tenantId) { asyncLocalTenantId.Value = tenantId; return ib; } public static IFreeSql Get(this IdleBus<IFreeSql> ib) => ib.Get(asyncLocalTenantId.Value ?? "db1"); public static IBaseRepository<T> GetRepository<T>(this IdleBus<IFreeSql> ib) where T : class => ib.Get().GetRepository<T>(); //------------------------------------------------------- static void test() { IdleBus<IFreeSql> ib = null; //单例注入 var fsql = ib.Get(); //获取当前租户对应的 IFreeSql var fsql00102 = ib.ChangeTenant("00102").Get(); //切换租户,后面的操作都是针对 00102 var songRepository = ib.GetRepository<Song>(); var detailRepository = ib.GetRepository<Detail>(); } //------------------------------------------------------- public static IServiceCollection AddIdleBusRepository(this IServiceCollection services, IdleBus<IFreeSql> ib, params Assembly[] assemblies) { services.AddSingleton(ib); services.AddScoped(typeof(IBaseRepository<>), typeof(YourDefaultRepository<>)); services.AddScoped(typeof(BaseRepository<>), typeof(YourDefaultRepository<>)); services.AddScoped(typeof(IBaseRepository<,>), typeof(YourDefaultRepository<,>)); services.AddScoped(typeof(BaseRepository<,>), typeof(YourDefaultRepository<,>)); if (assemblies?.Any() == true) foreach (var asse in assemblies) //批量注册 foreach (var repo in asse.GetTypes().Where(a => a.IsAbstract == false && typeof(IBaseRepository).IsAssignableFrom(a))) services.AddScoped(repo); return services; } } class YourDefaultRepository<T> : BaseRepository<T> where T : class{ public YourDefaultRepository(IdleBus<IFreeSql> ib) : base(ib.Get(), null, null) { } } class YourDefaultRepository<T, TKey> : BaseRepository<T, TKey> where T : class{ public YourDefaultRepository(IdleBus<IFreeSql> ib) : base(ib.Get(), null, null) { } }
分库总结:
跨库 可以使用 ib.Get() 获取 IFreeSql 进行 CRUD;
跨库 事务不好处理,注意了;
跨库 查询不好处理,注意了;
系列文章导航
FreeSql (一)入门及安装 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二)自动迁移实体 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三)实体特性 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (四)实体特性 Fluent Api - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (五)插入数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (六)批量插入数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (七)插入数据时忽略列 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (八)插入数据时指定列 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (九)删除数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十)更新数据 - 学习文档 - 零一的世界 (vtzw.com)FreeSql (十一)更新数据 Where - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十二)更新数据时指定列 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十三)更新数据时忽略列 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十四)批量更新数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十五)查询数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十六)分页查询 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十七)联表查询 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十八)导航属性 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (十九)多表查询 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十)多表查询 WhereCascade - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十一)查询返回数据 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十二)Dto 映射查询 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十三)分组、聚合 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十四)Linq To Sql 语法使用介绍 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十五)延时加载 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十六)贪婪加载 Include、IncludeMany - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十七)将已写好的 SQL 语句,与实体类映射进行二次查询 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十八)事务 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (二十九)Lambda 表达式 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十)读写分离 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十一)分表分库 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十二)Aop - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十三)CodeFirst 类型映射 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十四)CodeFirst 迁移说明 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十五)CodeFirst 自定义特性 - 学习文档 - 零一的世界 (vtzw.com)
FreeSql (三十六)进阶 仓储系列文档 - 学习文档 - 零一的世界 (vtzw.com)
发表评论 / 取消回复