使用.NET Core实现饿了吗拆红包功能

需求说明

以前很讨厌点外卖的我,最近中午经常点外卖,因为确实很方便,提前点好餐,算准时间,就可以在下班的时候吃上饭,然后省下的那些时间就可以在中午的时候多休息一下了。

点餐结束后,会有一个好友分享红包功能,虽说这个红包不能提现,但却可以抵扣点餐费用,对于经常点餐的人来说,直接用于抵扣现金确实是很大的诱惑,在点餐之后所获得的那个红包,必须要分享出去才能拆。

那么如果自己也想实现以下抢红包功能,需要说明的是,本文所描述的红包功能更多的关注与随机红包的生成,至于高并发、数据一致性等问题,本文暂未涉及,以下是本文所讨论的两个技术点:

不同的消费金额获取的红包总额不同,消费金额越大,红包总额就越大,红包总数也就越多;假设有一天,有一种需求是,需要保证参与抢红包的人获得的红包金额在平均数附近波动,也就是尽量的服从正态分布;

功能实现

本文描述的场景,所涉及到的金额以分为单位,目的是为了更好的处理随机数。总体的示意图如下:

消费后红包的初始化

需求重点,用户分享出去的红包总额跟消费总额成正比,可以分拆的子红包个数也与消费总额成正比。

比如:

10-20元的消费金额,可以分享的单个红包金额为10元,可以供5个人抢20-40元的消费金额,可以分享的单个红包金额为20元,可以供8个人抢40-60元的消费金额,可以分享的单个红包金额为30元,可以供10个人抢60-100元的消费金额,可以分享的单个红包金额为40元,可以供10个人抢100元以上的消费金额,可以分享的单个红包金额为50元,可以供10个人抢

那么我们设计出来一个实体,用于表示红包信息,以方便的配置及调整红包规则

public class RedPacketsInfo
{
/// <summary>
/// 最大消费金额
/// </summary>
public int MaxAmount { get; set; }
/// <summary>
 /// 最小消费金额
  /// </summary>
  public int MinAmount { get; set; }
  /// <summary>
  /// 红包金额
  /// </summary>
  public int TotalAmount { get; set; }
  /// <summary>
  /// 红包可被分割的数量
  /// </summary>
  public int RedPacketQuantity { get; set; }
}

红包初始化信息

private static List<RedPacketsInfo> GetRedPackets()
{
  return new List<RedPacketsInfo>()
  {
    new RedPacketsInfo
    {
      MinAmount = 1000,
      MaxAmount = 2000,
      RedPacketQuantity = 5,
      TotalAmount=1000
    },
    new RedPacketsInfo
    {
      MinAmount = 2000,
      MaxAmount = 3000,
      RedPacketQuantity = 5,
      TotalAmount=1000
    },
    new RedPacketsInfo
    {
      MinAmount = 4000,
      MaxAmount = 6000,
      RedPacketQuantity = 5,
      TotalAmount=1000
    },
    new RedPacketsInfo
    {
      MinAmount = 6000,
      MaxAmount = 8000,
      RedPacketQuantity = 5,
      TotalAmount=1000
    },
    new RedPacketsInfo
    {
      MinAmount = 10000,
      MaxAmount = int.MaxValue,
      RedPacketQuantity = 5,
      TotalAmount=1000
    }
  };
}

接下来我们就可以通过消费金额获取相应的红包信息了。

随机红包的生成时机及处理

随机红包的生成可以在抢之前生成也可以在抢的过程中确定,一般而言,很多时候红包会在抢的过程中动态的实际分配,不过在本文中,红包在用户分享成功后会预先生成,主要原因是为了更好地处理处理数据,以使得数据能够服从正态分布。

以下是其流程图,其中有一段逻辑是回调功能,可能会有圈友会问,如何保证有回调以及回调是成功的,这个地方有很多种处理,比如MQ、任务调度等,此处也不做讨论

那么我们需要设计一个新的实体,以表示分享出去的红包及其生成的随机红包:

public class SharedRedPacket
{
  /// <summary>
  /// 分享人UserId
  /// </summary>
  public int SenderUserId { get; set; }
  /// <summary>
  /// 分享时间
  /// </summary>
  public DateTime SendTime { get; set; }
  public List<RobbedRedPacket> RobbedRedPackets { get; set; }
}
public class RobbedRedPacket
{
  /// <summary>
  /// 抢到红包的人的UserId
  /// </summary>
  public int UserId { get; set; }
  /// <summary>
  /// 抢到的红包金额
  /// </summary>
  public int Amount { get; set; }
  /// <summary>
  /// 抢到时间
  /// </summary>
  public DateTime RobbedTime { get; set; }
}

在实现过程中,根据用户消费金额获取相应红包,然后通过随机数,生成n-1个原始的随机数据,最后一个数据用总和减去n-1个数据的和获取到

//红包随机拆分
Random ran = new Random();
List<double> randoms = new List<double>(redPacketsList.Count);
for (int i = 0; i < redPacketsInfo.RedPacketQuantity - 1; i++)
{
  int max = (totalAmount - (redPacketsInfo.RedPacketQuantity - i)) * 1;
  int result = ran.Next(1, max);
  randoms.Add(result);
  totalAmount -= result;
}
randoms.Add(totalAmount);

然后通过设置好系数,以处理数据达到服从正太分布的目的:

//正太分布处理
for (int i = 0; i < redPacketsInfo.RedPacketQuantity; i++)
{
  double a = Math.Sqrt(Math.Abs(2 * Math.Log(randoms[i], Math.E)));
  double b = Math.Cos(2 * Math.PI * randoms[i]);
  randoms[i] = a * b * 0.3 + 1;
}

经过第二次处理后,得到的数据与原始数据有偏差,那么我们通过等比例方式再次处理,以确保拆分后的红包总额等于红包原始总额:

//生成最终的红包数据
double d = originalTotal / randoms.Sum();
SharedRedPacket sharedRedPacket = new SharedRedPacket();
sharedRedPacket.RobbedRedPackets = new List<RobbedRedPacket>(redPacketsList.Count);
for (int i = 0; i < redPacketsInfo.RedPacketQuantity - 1; i++)
{
  sharedRedPacket.RobbedRedPackets.Add(new RobbedRedPacket
  {
    Amount = (int)Math.Round(randoms[i] * d, 0)
  });
}
sharedRedPacket.RobbedRedPackets.Add(new RobbedRedPacket
{
  Amount = originalTotal - sharedRedPacket.RobbedRedPackets.Sum(p => p.Amount)
});

测试

测试效果图如下:

部分代码如下,

Console.WriteLine("是否分享输入Y分享成功,输入N退出");
 string result = Console.ReadLine();
 if (result == "Y")
 {
   var leftRedPacket = sharedRedPacket.RobbedRedPackets.Where(p => p.UserId <= 0).ToList();
   var robbedRedPacket = leftRedPacket[new Random().Next(1, leftRedPacket.Count + 1)];
   Console.WriteLine("抢到的到红包金额是:" + robbedRedPacket.Amount);
   Console.WriteLine("-------------------------------------------------------");
 }

总结

以上所述是小编给大家介绍的使用.NET Core实现饿了吗拆红包功能,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!

(0)

相关推荐

  • 使用.NET Core实现饿了吗拆红包功能

    需求说明 以前很讨厌点外卖的我,最近中午经常点外卖,因为确实很方便,提前点好餐,算准时间,就可以在下班的时候吃上饭,然后省下的那些时间就可以在中午的时候多休息一下了. 点餐结束后,会有一个好友分享红包功能,虽说这个红包不能提现,但却可以抵扣点餐费用,对于经常点餐的人来说,直接用于抵扣现金确实是很大的诱惑,在点餐之后所获得的那个红包,必须要分享出去才能拆. 那么如果自己也想实现以下抢红包功能,需要说明的是,本文所描述的红包功能更多的关注与随机红包的生成,至于高并发.数据一致性等问题,本文暂未涉及,

  • .Net Core实现图片文件上传下载功能

    当下.Net Core项目可是如雨后春笋一般发展起来,作为.Net大军中的一员,我热忱地拥抱了.Net Core并且积极使用其进行业务的开发,我们先介绍下.Net Core项目下实现文件上传下载接口. 一.开发环境 毋庸置疑,宇宙第一IDE VisualStudio 2017 二.项目结构 FilesController 文件上传下载控制器 PictureController 图片上传下载控制器 Return_Helper_DG 返回值帮助类 三.关键代码 1.首先我们来看Startup.cs

  • 基于jQuery实现的双11天猫拆红包抽奖效果

    本文实例讲述了jQuery实现的双11天猫拆红包抽奖效果代码,是一款基于jquery+css3实现的鼠标点击红包摇一摇抽奖代码,具有点击抽奖后红包摇动并弹出抽奖结果信息的功能,分享给大家供大家参考.具体如下: 运行效果截图如下: 具体代码如下: HTML部分: <div class="opacity" style="display: none;"></div> <div class="red"><img

  • ASP.NET Core中使用MialKit实现邮件发送功能

    具体代码如下所示: # 导包 首先我们需要导入 MailKit NuGet包,NuGet安装包命令在下方拓展介绍中. # 引用命名空间 using MailKit.Net.Smtp; using MimeKit; # 邮件发送帮助类 /// <summary> /// 发送邮件 /// </summary> /// <param name="Name">发件人名字</param> /// <param name="rece

  • ASP.NET Core扩展库之Http请求模拟功能的使用

    如今,完全独立的业务应用几乎不存在,不管是在企业内部微服务之间的调用,还是与外部第三方服务的调用,Http的API交互是常见的场景,这些实际情况给我们的开发带来了比较大的挑战,一是第三方服务可能会牵制我们的开发进度,特别是在多团队开发的情况下,由于依赖于其他团队的服务,有时候需要等待其他团队的进度,导致自己团队的无效等待.有时因为其他团队的延期,导致团队的被动延期.二是第三方服务的质量问题或开发过程中的频繁更新导致的部署问题,将严重拖累自己团队的开发进度,同时让你无法专心的开发自己的服务.三是单

  • asp.net core集成kindeditor实现图片上传功能

    本文为大家分享了asp.net core 如何集成kindeditor并实现图片上传功能的具体方法,供大家参考,具体内容如下 准备工作 1.visual studio 2015 update3开发环境 2.net core 1.0.1 及以上版本 目录 新建asp.net core web项目 下载kindeditor 增加图片上传控制器 配置kindeditor参数 代码下载 新建asp.net core web项目 新建一个asp.net core项目,这里命名为kindeditor 选中w

  • AntDesign Pro + .NET Core 实现基于JWT的登录认证功能

    很多同学说AgileConfig的UI实在是太丑了.我想想也是的,本来这个项目是我自己使用的,一开始甚至连UI都没有,全靠手动在数据库里修改数据.后来加上了UI也是使用了老掉牙的bootstrap3做为基础样式.前台框架也是使用了angularjs,同样是老掉牙的东西.过年期间终于下决心翻新AgileConfig的前端UI.最后选择的前端UI框架为AntDesign Pro + React.至于为啥选Ant-Design Pro是因为他好看,而且流行,选择React是因为VUE跟Angular我

  • Asp.Net Core 使用Monaco Editor 实现代码编辑器功能

    在项目中经常有代码在线编辑的需求,比如修改基于Xml的配置文件,编辑Json格式的测试数据等.我们可以使用微软开源的在线代码编辑器Monaco Editor实现这些功能.Monaco Editor是著名的VSCode的前身,项目地址:https://microsoft.github.io/monaco-editor/.本文介绍在Asp.Net Core项目中使用Monaco Editor实现代码编辑器功能. 安装 可以使用npm下载moaco-editor: npm install monaco

  • ASP.NET Core中的Razor页面实现路由功能

    在服务器端 Web 应用程序框架中,其中非常重要的设计是开发人员如何将URL与服务器上的资源进行匹配,以便正确的处理请求.最简单的方法是将 URL 映射到磁盘上的物理文件,在 Razor 页面框架中,ASP.NET团队就是这样实现的. 关于 Razor 页面框架如何将 URL 与文件相匹配,有一些规则您必须了解,以及如何根据需要自定义规则改变输出的结果.如果您将 Razor 页面与 Web Form 框架进行比较,您还需要了解取代的 Ur l参数以及在URL中传递数据的机制. 规则一,Razor

  • .Net Core创建Api进行文件上传功能

    1.Net Core创建api接口,用于接收外部请求,进行文件的上传 2.添加控制器类,添加如下代码: [HttpPost] [Route("PostFile")] public String PostFile([FromForm] IFormCollection formCollection) { String result = "Fail"; if (formCollection.ContainsKey("user")) { var user

随机推荐