首页 > 其他分享 >MongoDriver 分表分页查询

MongoDriver 分表分页查询

时间:2023-04-21 12:11:35浏览次数:53  
标签:MongoDriver pageIndex 分页 pageSize Value collection startTime 分表 endTime

摘要:

  • 业务需求,分表也要兼容旧表。 技术有限,封装思路及代码如下,大佬们见笑。
  • 首先Mongdb的Collection及其内容字段都是可以动态创建的,所以这里需要的一个关键点是,分表时用什么字段。
  • 本文将使用数据的创建时间作为依据,按月分表(如果需要其它字段分表,也可以参考这个思路)
  • 首先本文使用泛型类约束为前提,类:MongoDriver<T>,这么做的目的是,每一个实例对应一个表操作类

上代码:

1. 封装根据时间获取Collection的方法,这里添加一个对事务的支持可选参数

 1         private IMongoCollection<T> GetCollection(DateTime? yearAndMonth, IClientSessionHandle session = null)
 2         {
 3             var tableName = typeof(T).Name;
 4             //如果没有传入时间范围就使用原表名
 5             var fixedTableName = yearAndMonth.HasValue ? $"{tableName}{yearAndMonth:yyyyMM}" : tableName;
 6 
 7             if (session == null)
 8             {
 9                 return _mongoDatabase.GetCollection<T>(fixedTableName);
10             }
11             else
12             {
13                 return session.Client.GetDatabase(_dBName).GetCollection<T>(fixedTableName);
14             }
15         }

 

2. 封装根据操作的数据的时间范围获取Collection

 1  private IEnumerable<IMongoCollection<T>> GetCollections(DateTime? startTime, DateTime? endTime, IClientSessionHandle session = null)
 2         {
 3             //没有时间范围最多查一年
 4             if (!startTime.HasValue && !endTime.HasValue)
 5             {
 6                 endTime = DateTime.Now;
 7                 startTime = endTime.Value.AddYears(-1);
 8             }
 9             //没有开始时间有结束时间,默认从开始时间往后查一年
10             else if (!startTime.HasValue && endTime.HasValue)
11             {
12                 startTime = endTime.Value.AddYears(-1);
13             }
14             else
15             {
16                 startTime ??= DateTime.Now;
17                 endTime ??= DateTime.Now;
18             }
19 
20             if (endTime.Value < startTime.Value)
21             {
22                 throw new InvalidOperationException("End time should greater than start time");
23             }
24             else if ((endTime.Value - startTime.Value).TotalDays > 365)
25             {
26                 throw new InvalidOperationException("时间跨度不能大于365天");
27             }
28             if (startTime.Value.Year == endTime.Value.Year && startTime.Value.Month == endTime.Value.Month)
29             {
30                 var collection = GetCollection( startTime.Value, session);
31                 if (collection != null)
32                 {
33                     yield return collection;
34                 }
35             }
36             else
37             {
38                 var cur = endTime.Value;
39                 while (cur.Month >= startTime.Value.Month)
40                 {
41                     var collection = GetCollection( new DateTime(cur.Year, cur.Month, 1), session);
42                     if (collection != null)
43                     {
44                         yield return collection;
45                     }
46                     cur = cur.AddMonths(-1);
47                 }
48             }
49 
50             yield return _mongoCollection;
51         }

 

3.在有了获取Collection的支撑后,剩下的事情就好办了,我们来实现分页查询的逻辑

 1     /// <summary>
 2         /// 按时间段分页查询
 3         /// </summary>
 4         /// <typeparam name="TKey"></typeparam>
 5         /// <param name="predicate">查询条件</param>
 6         /// <param name="pageIndex">第几页</param>
 7         /// <param name="pageSize">每页数量</param>
 8         /// <param name="keySelector">要排序的字段,不需要的话传null</param>
 9         /// <param name="totalCount">总数</param>
10         /// <param name="desc">默认降序</param>
11         /// <param name="startTime">查询范围的开始时间,精确到年月日</param>
12         /// <param name="endTime">查询范围的结束时间,精确到年月日</param>
13         /// <returns></returns>
14         public List<T> QueryPagedByTimeRange<TKey>(Expression<Func<T, bool>> predicate, int pageIndex, int pageSize, Expression<Func<T, TKey>> keySelector, out int totalCount, bool desc = true, DateTime? startTime = null, DateTime? endTime = null)
15         {
16             if (pageSize <= 1)
17             {
18                 throw new ArgumentException("arg:pageSize should greater than 1");
19             }
20 
21             var collections = GetCollections(startTime, endTime);
22             var result = new List<T>();
23             totalCount = 0;
24             var pageLeft = pageSize;
25 
26             foreach (var item in collections)
27             {
28                 //当前表符合条件的数量
29                 var currentTableCount = QueryPageCount(predicate, item);
30 
31                 //查到的总数为0 ,没必要再去查一遍数据了
32                 if (currentTableCount == 0)
33                 {
34                     continue;
35                 }
36 
37                 //查询到的总数
38                 totalCount += currentTableCount;
39                 //从第几条开始
40                 var startIndexNumber = (pageIndex - 1) * pageSize;
41                 //当前表查询到的数据
42                 var rangeResult = new List<T>();
43                 //第pageIndex页,从startIndexNumber条开始取,如果查到的数累计不够到startIndexNumber 就不执行查询了,继续下一张表的数量计算
44                 if (startIndexNumber > totalCount)
45                 {
46                     continue;
47                 }
48 
49                 rangeResult = QueryPaged(predicate, pageIndex, pageLeft, keySelector, item, desc);
50 
51                 //下一个表再查pageLeft条
52                 pageLeft -= rangeResult.Count;
53 
54                 if (rangeResult?.Count > 0)
55                 {
56                     result.AddRange(rangeResult);
57                 }
58 
59                 //当前表查询完不够一页了 下一个表从第一页算
60                 if (pageLeft > 0)
61                 {
62                     pageIndex = 1;
63                 }
64 
65                 //够一页了  返回数据
66                 if (pageLeft <= 0)
67                 {
68                     break;
69                 }
70             }
71 
72             return result;
73         }

4.最后用到普通的分页查询逻辑

 1         private List<T> QueryPaged<TKey>(Expression<Func<T, bool>> predicate, int pageIndex, int pageSize, Expression<Func<T, TKey>> orderKeySelector, IMongoCollection<T> collection, bool desc = true)
 2         {
 3             IQueryable<T> queryList = default;
 4             if (predicate == null)
 5             {
 6                 if (orderKeySelector != null)
 7                 {
 8                     queryList = desc ?
 9                         collection.AsQueryable().OrderByDescending(orderKeySelector).Skip(pageSize * (pageIndex - 1)).Take(pageSize) :
10                         collection.AsQueryable().OrderBy(orderKeySelector).Skip(pageSize * (pageIndex - 1)).Take(pageSize);
11                 }
12                 else
13                 {
14                     queryList = collection.AsQueryable().Skip(pageSize * (pageIndex - 1)).Take(pageSize);
15                 }
16             }
17             else
18             {
19                 if (orderKeySelector != null)
20                 {
21                     queryList = desc ?
22                         collection.AsQueryable().Where(predicate).OrderByDescending(orderKeySelector).Skip(pageSize * (pageIndex - 1)).Take(pageSize) :
23                         collection.AsQueryable().Where(predicate).OrderBy(orderKeySelector).Skip(pageSize * (pageIndex - 1)).Take(pageSize);
24                 }
25                 else
26                 {
27                     queryList = collection.AsQueryable().Where(predicate).Skip(pageSize * (pageIndex - 1)).Take(pageSize);
28                 }
29             }
30 
31             return queryList?.ToList();
32         }

其它CURD操作这里不再赘述了,主要还是通过GetCollections方法来确定分表的范围,根据范围得到该操作哪些Collection,之后执行业务代码即可。

标签:MongoDriver,pageIndex,分页,pageSize,Value,collection,startTime,分表,endTime
From: https://www.cnblogs.com/cnwhm/p/17339919.html

相关文章

  • oracle 分页存储过程
     查询oracle时,做存储过程实现分页  createorreplaceprocedurequery_by_page(pagenuminnumber,pagesqlinvarchar2,pagesizeinnumber,pagecountoutnumber,allpagecountoutnumber,v_curoutquerypage.type_cur)asv_sqlvarchar2(3000);v_pbnumber;v_pd......
  • 数据库 分表分库
    一、分表分库1、垂直分区:根据数据库里面数据表的相关性进行拆分。例如,用户表中既有用户的登录信息又有用户的基本信息,可以将用户表拆分成两个单独的表,甚至放到单独的库做分库。简单来说垂直拆分是指数据表列的拆分,把一张列比较多的表拆分为多张表。垂直......
  • react+ts+hook封装一个table分页组件(建议收藏,直接使用)
    前言大家好我是歌谣我是一名坚持写博客四年的博主最好的种树是十年前其次是现在,今天继续对antdesigntable的分页封装进行讲解封装准备(多看官网)jsx风格的api<><Table<User>columns={columns}dataSource={data}/>/*使用JSX风格的API*/<Table<User>data......
  • 用户分页查询展示
    如下图,点击查询按钮,显示用户数据;或者输入用户名、电话号码进行查询。以下标签的使用,查看组件|Element前端页面代码src/views/sys/user.vue:实现了查询、分页功能,未实现新增、编辑、删除功能。<template><div><!--查询栏、(el-card:卡片)--><el-cardclass="box-......
  • mycat单库分表(踩坑安装、配置、程序连接)
    一、简介Mycat是一个彻底开源的,面向企业应用开发的大数据库集群中间件,本身带有非常复杂成熟的功能,完成数据库的集群,实现业务数据的分库分表。本次文章介分享的mycat的单库分表实际应用,包括在实际应用时所要注意的事项。二、下载mycat及环境要求在官网下载按照包:ht......
  • 排序分页结果错乱的问题
     如果orderby有内容,但是不能确定唯一的位置,则sql每次查询的结果可能不一样,如果进行分页的话,就会导致数据错乱。分页必须保证orderby确定每行的位置,如果不能确定,可以使用多列排序方法:orderbya,b......
  • 9.数据库分库分表备份 + 随机点名脚本
    1.数据库分库分表备份  2.随机点名脚本 ......
  • react+ts+hook封装一个table分页组件(建议收藏,直接使用)
    前言大家好我是歌谣我是一名坚持写博客四年的博主最好的种树是十年前其次是现在,今天继续对antdesigntable的分页封装进行讲解封装准备(多看官网)jsx风格的api<><Table<User>columns={columns}dataSource={data}/>/*使用JSX风格的API*/<Table<User>data......
  • mybatisPlus-分页查询
    添加分页插件importcom.baomidou.mybatisplus.annotation.DbType;importcom.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;importcom.baomidou.mybatisplus.extension.plugins.inner.OptimisticLockerInnerInterceptor;importcom.baomidou.mybatisplu......
  • Java MyBatis-Plus(4)MybatisPlus整合Pagehelper实现分页
    序言 /***pageInfo对象中属性含义*privateintpageNum;//当前页码*privateintpageSize;//设置每页多少条数据*privateintsize;//当前页有多少条数据*privateintstartRow;//当前页码第一条数据的*privateintendRow;//......