导言
最近搞到了个签到管理,其中的业务逻辑感觉有点复杂(可能是我的方向不对),虽然是实现了,不过代码和逻辑很多,也有些乱,想趁着还记得逻辑来记录梳理一下,看看自己以后有没有更好的思路,或者有大佬有思路也可以在评论指导一下,非常感谢( ̄▽ ̄)!
业务
签到可以有时间段限制,一天可以有多段可签到时间(如9:00到12:00,15:00到18:00...),签退不能超过最大签到时间,也不能跨时间段签到)。
逻辑
由于可签到时间段我是按一天来想,所以格式是从一天的零点算起到当天的24:结束,所以下面的一些逻辑和实现也参照这个来。
签到:
这个比较简单,可以拿个bool变量IsInSignInTime 开始为假,判断当前签到时间是否在签到时间段里,直接拿到存数据库的时间段列表来循环判断就行,如果当前时间既大于可签到时间段开始时间又小于结束时间的话,就说明在可签到时间段里反之直接返回前端错误。
签退:
签退除了前端传的数据有误之外不管是 超过最大签到时间 还是 跨时间段 都是可以直接签退成功的,就是有上面两种情况的就要把本次签到的时间记为零。
1.跨时间段
由于上面在创建签到时做了限制,如果有可签到时间段,那最大签到时间不能超过24小时(如果超24小时的话肯定会跨时间段的)。所以签退只需考虑签到时间为一天以内的情况,有以下两种状态:
1)签退跨时间段,但是没跨天
这样直接和签到的逻辑差不多,拿个bool变量,如果跨了时间段就把其变真,后面做个置零判断。
2)签退跨天,但是没跨可签到时间段
如果有签到时间段:
这样还要在1)的循环中作判断,看有没有这样从今天晚上12点之前到明天凌晨0点的时间段,如果有的话就额外判断用户的签到时间段在不在这里面。在的话就不能判为跨时间。
C#代码实现
说明:
签到记录的时间是按分钟来记的,时间段的比较是用了C#的TimeOnly来比。里面有点注释掉的代码,不影响实现。
#region 签到记录接口/// <summary>/// 签到定义/// </summary>/// <returns></returns>[HttpPost][Authorization]public IActionResult Create([FromBody] SignInOrSignOutDto parm){var signIn = _signInService.GetId(parm.SignInID);if (_signInService.Any(m => m.SignInNo == signIn.SignInNo && m.Enable == false)){return toResponse(StatusCodeType.Error, $"该签到未启用,不能签到!");}//签到启用//没有同一条签到记录//在签到时间段//存在同一签到记录且状态为1的签到记录if (_signInRecordService.Any(m => m.SignInID == parm.SignInID && m.UserID == parm.UserID && m.Status == 1)){return toResponse(StatusCodeType.Error, $"不能重复签到!");}//不在签到时间段//从中拿到签到时间段// 将JSON字符串反序列化为List<SignInPeriods>对象列表var signInPeriodsList = JsonConvert.DeserializeObject<List<SignInPeriodsDto>>(signIn.SignInPeriods);TimeOnly currentTime = TimeOnly.FromDateTime(DateTime.Now);//拿个变量在循环里,如果当前时间在签到时间段就为真,反之为假bool IsInSignInTime = false;//循环比较判断foreach(SignInPeriodsDto signInPeriod in signInPeriodsList){if(signInPeriod.StartTime < currentTime&& signInPeriod.EndTime > currentTime){//在里面变真IsInSignInTime = true;}}//为假就是不在签到时间段,返回错误if (!IsInSignInTime){return toResponse(StatusCodeType.Error, $"不在签到时间段!");}try{var signInRecord = parm.Adapt<Base_SignInRecord>().ToCreate(_tokenManager.GetSessionInfo());signInRecord.Status = 1;signInRecord.StartTime = DateTime.Now;_unitOfWork.BeginTran();//新增签到记录表var response = _signInRecordService.Add(signInRecord);//改签到表对应签到的当前签到人数_signInService.Update(m => m.ID == parm.SignInID, m => new Base_SignIn(){CurrentSignInNum = signIn.CurrentSignInNum + 1,});//改用户-签到表对应用户状态为在线/签到_signInUsersService.Update(m => m.SignInNo == signIn.SignInNo && m.UserID == parm.UserID, m => new Base_SignIn_Users(){IsOnline = true});_unitOfWork.CommitTran();return toResponse(response);}catch (Exception){_unitOfWork.RollbackTran();throw;}}/// <summary>/// 签退定义/// </summary>/// <returns></returns>[HttpPost][Authorization]public IActionResult Delete([FromBody] SignInOrSignOutDto parm){//拿到数据var signIn = _signInService.GetId(parm.SignInID);var signInRecord = _signInRecordService.GetFirst(m => m.SignInID == signIn.ID && m.UserID == parm.UserID && m.Status == 1);if (_signInService.Any(m => m.SignInNo == signIn.SignInNo && m.Enable == false)){return toResponse(StatusCodeType.Error, $"该签到未启用,不能签退!");}//不做数的 体现在Time字段上//超出最大签到时间的//横跨签到时间段的//算出签到时间TimeDateTime currentTimeForDateTime = DateTime.Now;var TimePeriod = currentTimeForDateTime - signInRecord.StartTime;var currentTime = TimeOnly.FromDateTime(currentTimeForDateTime);测试 可删算出签到时间Time//DateTime currentTimeForDateTime = new DateTime(2024, 9, 24, 07, 49, 15);//var TimePeriod = currentTimeForDateTime - signInRecord.StartTime;var currentTime = TimeOnly.FromDateTime(currentTimeForDateTime);//TimeOnly currentTime = TimeOnly.Parse("07:49:15");int Time = (int)TimePeriod.TotalMinutes;//签到提示信息,如果签到没问题就直接输成功var remindMessager = "成功";//从中拿到签到时间段// 将JSON字符串反序列化为List<SignInPeriods>对象列表var signInPeriodsList = JsonConvert.DeserializeObject<List<SignInPeriodsDto>>(signIn.SignInPeriods);//看是否超了最大时间if (Time > signIn.MaxSignInTime && signIn.MaxSignInTime != 0){Time = 0;remindMessager = "签到时间超出最大签到时间,本次签到时间置零!";}//如果是开了可签到时间段if (!string.IsNullOrEmpty(signIn.SignInPeriods)){直接拿开始时间和现在时间做判断,看其是这个时间段是否在某一时间段里面,其他不满足的都是跨时间段的//if (Time < 1440)//没隔天 一天的分钟计数//{//把签到记录表的开始签到时间转成TimeOnlyvar StartTime = TimeOnly.FromDateTime(signInRecord.StartTime);//拿个变量在循环里,如果时间段没跨签到时间段就为真,反之为假bool IsInSignInTime = false;//看有没有开始时间为0:00的时间段和结束时间为24:00的时间段SignInPeriodsDto todayPeriod = null; //结束时间为24:00的时间段SignInPeriodsDto nextDayPeriod = null;//开始时间为0:00的时间段//循环比较判断foreach (SignInPeriodsDto signInPeriod in signInPeriodsList){if (signInPeriod.StartTime < StartTime && signInPeriod.EndTime > currentTime){//在里面变真IsInSignInTime = true;}if (signInPeriod.StartTime == TimeOnly.MinValue) //看其开始时间段是否为为0:00{nextDayPeriod = signInPeriod;}if (signInPeriod.EndTime == new TimeOnly(23, 59, 59)) //看其结束时间段是否为为24:00{todayPeriod = signInPeriod;}}//有的话看开始签到时间是否在大于结束时间为24:00的时间段里 及 签退时间是否在在开始时间为0:00的时间段里//不满足条件的直接判跨时间段if (todayPeriod != null && nextDayPeriod != null){ //可以考虑将跨天取消掉,分别判是否在签到时间段里和单独看是否在跨天时间段里就行//可以在添加签到时加个判断,如过加了时间段,那最大时间就不能出现超过24小时的情况,//可以免除上面签到和签退时间隔天了还在同一签到时间段的情况if(todayPeriod.StartTime< StartTime&& nextDayPeriod.EndTime >currentTime){IsInSignInTime = true;}}//如果没设置最大签到时间但是设置了时间段//只判断其有没有超时间段,在关键判断那看其签到时间是不是超了一天// Time > 1440由于上面已经验证了有没有时间段,所以到这的都是有时间段的,不用额外判断了if (!IsInSignInTime|| Time > 1440)//跨时间段了{Time = 0;remindMessager = "横跨时间段,本次签到时间置零!";}//}}//至此留下了符合条件的try{var userSession = _tokenManager.GetSessionInfo();_unitOfWork.BeginTran();//改签到记录表状态为 签退,补上签到时间Timevar response = _signInRecordService.Update(m => m.SignInID == signIn.ID && m.UserID == parm.UserID && m.Status == 1, m => new Base_SignInRecord(){Status = 0,EndTime = currentTimeForDateTime,Time = Time,});//改签到表在线签到人数-1_signInService.Update(m => m.ID == parm.SignInID, m => new Base_SignIn(){CurrentSignInNum = signIn.CurrentSignInNum - 1,UpdateID = userSession.UserID,UpdateName = userSession.UserName,UpdateTime = DateTime.Now});//改 用户-签到表对应用户状态为离线/签退_signInUsersService.Update(m => m.SignInNo == signIn.SignInNo && m.UserID == parm.UserID,m=> new Base_SignIn_Users(){IsOnline = false});_unitOfWork.CommitTran();return toResponse(StatusCodeType.Success, remindMessager);}catch (Exception){_unitOfWork.RollbackTran();throw;}}#endregion
结语
这个搞了有些久,还是怕有些情况没考虑,不过接口是考虑到的情况都测试通过了的。想着还有没有更优雅,更便捷的解决方法,不管是从业务开始考虑,还是从我这方法开始优化,那就看以后的我有没有时间看了,或者各位大佬指点一下,我会很开心的!