写点什么

群主发红包带你深入了解继承和 super、this 关键字

作者:共饮一杯无
  • 2022-10-20
    浙江
  • 本文字数:4964 字

    阅读完需:约 1 分钟

需求

群主发随机红包或者普通红包。某群有多名成员,群主给成员发普通红包。随机红包规则:


  1. 群主的一笔金额,从群主余额中扣除,随机分成 n 等份,让成员领取。

  2. 成员领取红包后,保存到成员余额中。


普通红包的规则:


  1. 群主的一笔金额,从群主余额中扣除,平均分成 n 等份,让成员领取。

  2. 成员领取红包后,保存到成员余额中。

案例分析

案例分析,可以得出如下继承关系:


案例代码实现

定义用户类

/** * 用户类 * @author zjq */public class User {    /**     * 姓名     */    private String name;    /**     * 余额,也就是当前用户拥有的钱数     */    private Integer money; 
public User() { }
public User(String name, Integer money) { this.name = name; this.money = money; }
// 展示一下当前用户有多少钱 public void show() { System.out.println("我是" + name + ",我有多少钱:" + this.fenToYuan(String.valueOf(money))+"元"); }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public Integer getMoney() { return money; }
public void setMoney(Integer money) { this.money = money; }
/** * 分转元 * @param amount * @return */ public String fenToYuan(String amount){ NumberFormat format = NumberFormat.getInstance(); try{ Number number = format.parse(amount); double temp = number.doubleValue() / 100.0; format.setGroupingUsed(false); format.setMaximumFractionDigits(2); amount = format.format(temp); } catch (ParseException e){ e.printStackTrace(); } return amount; }
}
复制代码

定义群主类

package com.zjq.javabase.base09.demo14;
import org.apache.commons.lang3.RandomUtils;import java.util.ArrayList;
/** * 群主的类 * @author zjq */public class Manager extends User {
/** * 收到单个红包最大值 */ private static final int MAX_AMOUNT = 20000;
public Manager() { }
public Manager(String name, int money) { // 通过super 调用父类构造方法 super(name, money); }

/** * 发红包 * @param totalMoney 红包总金额(单位分) * @param count 发包个数 * @param type 发包类型(0、随机红包,1、定额红包) * @return 红包集合 * @throws Exception */ public ArrayList<Integer> send(Integer totalMoney, int count,int type) throws Exception { // 首先需要一个集合,用来存储若干个红包的金额 ArrayList<Integer> redList = new ArrayList<>(count);
// 首先看一下群主自己有多少钱 Integer leftMoney = super.getMoney(); // 群主当前余额 if (totalMoney > leftMoney) { System.out.println("余额不足"); return redList; // 返回空集合 }
// 扣钱,其实就是重新设置余额 super.setMoney(leftMoney - totalMoney);

if (count == 1) { redList.add(totalMoney); return redList; }
switch (type) { case 0: // 默认分配1分至每一位 for (int i = 0; i < count; i++) { redList.add(1); } int surplus_currency = totalMoney - redList.size(),// 剩余金额数 surplus_number = redList.size();// 剩余需追加的数量
for (int i = 0; i < redList.size(); i++) {
// 没值可以追加了 if (new Integer(0).equals(surplus_currency)) { break; }
// (总数-(总包-i)*最小值) / (总包 - i) 随机安全值算法 int safe_total = (int)Math.floor((totalMoney - (count - i)) / (count - i)); if (new Integer(0).equals(safe_total)) {// 随机值不能为0 safe_total = 1; } // 该次随机值 int randomint = surplus_currency >= safe_total - 1 ? safe_total : surplus_currency + 1; // 下次可能最大能剩余值 int nextMax_currency = (MAX_AMOUNT - 1) * (surplus_number - 1); // 最小的随机数 剩余金额-剩余最大随机的总数(不含这一次) int minRandom = surplus_currency - nextMax_currency; if (minRandom < 0) { minRandom = 0; }
// 规避一些特殊情况,每个接近2000或1时会发生 boolean must = (surplus_currency - count * MAX_AMOUNT <= 2 && surplus_currency - count * MAX_AMOUNT >= 0) /*|| surplus_currency < packet_number * 2*/; // 控制安全随机值 随机安全值不能大于最大限制,并且不能小于最小限 制 if (safe_total < minRandom || safe_total > MAX_AMOUNT || must) { safe_total = MAX_AMOUNT; // 该次随机值 randomint = surplus_currency >= safe_total - 1 ? safe_total : surplus_currency + 1; // 下次可能最大能剩余值 nextMax_currency = (randomint - 1) * (surplus_number - 1); // 最小的随机数 剩余金额-剩余最大随机的总数(不含这一次) minRandom = surplus_currency - nextMax_currency; if (minRandom < 0) { minRandom = 0; } }
// 下一次最大的随机值 int nextMaxRandomInt = nextMax_currency - (surplus_currency - (randomint - 1)); Integer maxRandom = nextMaxRandomInt <= 0 ? nextMaxRandomInt + randomint: null; // 能随机 剩余的金额 - 最大随机数 > 最大随机数 * 剩余数量 boolean canRandom = surplus_currency - (randomint - 1) > nextMax_currency || nextMaxRandomInt > (randomint - 1) || !new Integer(0).equals(minRandom);
int addNumber; // 追加的金额 if (canRandom && !new Integer(randomint).equals(minRandom+1) && !(new Integer(randomint).equals(minRandom) && new Integer(safe_total).equals(minRandom)) ) { addNumber = myRandom(minRandom, maxRandom == null ? randomint : maxRandom- 1); }else { addNumber = randomint - 1; } redList.set(i,redList.get(i) + addNumber); surplus_currency -= addNumber; surplus_number--; } break; case 1: // 定额红包校验 redList = new ArrayList<>(count); for (int i = 0; i <count; i++) { //定额红包要是不能整除会有问题,正常实现应该是输入单个红包金额和总数直接就能计算 redList.add(totalMoney/count); } break; default: throw new Exception("类型错误!"); }
System.out.println("我是" + this.getName() + "我发了"+fenToYuan(String.valueOf(totalMoney))+"元红包"+",我现在有多少钱:" + fenToYuan(String.valueOf(this.getMoney()))+"元"); return redList; }
/** * 生成随机金额 * @param min * @param randomint * @return */ public static int myRandom(int min,int randomint) { if (min == 0) { return RandomUtils.nextInt(0,randomint); }else { int nextInt = RandomUtils.nextInt(min,randomint - min); return nextInt + min; } }}
复制代码

定义成员类

/** * 普通成员 * @author zjq */public class Member extends User {
public Member() { }
public Member(String name, Integer money) { super(name, money); }
public void receive(ArrayList<Integer> list) { // 从多个红包当中随便抽取一个,给我自己。 // 随机获取一个集合当中的索引编号 int index = new Random().nextInt(list.size()); // 根据索引,从集合当中删除,并且得到被删除的红包,给我自己 Integer delta = list.remove(index); // 当前成员自己本来有多少钱: Integer money = super.getMoney(); // 加法,并且重新设置回去 super.setMoney(money + delta); System.out.println("我是" + this.getName() + ",我抢到了"+fenToYuan(String.valueOf(delta))+"元红包"+",我现在有多少钱:" + fenToYuan(String.valueOf(this.getMoney()))+"元"); }}
复制代码

定义测试类

public class MainRedPacket {
public static void main(String[] args) throws Exception { Manager manager = new Manager("群主", 10000); Member member1 = new Member("张三", 1000); Member member2 = new Member("李四", 1000); Member member3 = new Member("王五", 1000); Member member4 = new Member("赵六", 1000); Member member5 = new Member("孙七", 1000); Member member6 = new Member("小詹", 1000); Member member7 = new Member("小明", 1000); Member member8 = new Member("小红", 1000);
manager.show(); // 100 member1.show(); // 10 member2.show(); // 10 member3.show(); // 10 member4.show(); // 10 member5.show(); // 10 member6.show(); // 10 member7.show(); // 10 member8.show(); // 10 System.out.println("============================");
// 群主总共发20块钱,分成8个红包 ArrayList<Integer> redList = manager.send(2000, 8,0); // 八个普通成员收红包 member1.receive(redList); member2.receive(redList); member3.receive(redList); member4.receive(redList); member5.receive(redList); member6.receive(redList); member7.receive(redList); member8.receive(redList); }
}
复制代码


结果输出如下:



本文内容到此结束了,

如有收获欢迎点赞👍收藏💖关注✔️,您的鼓励是我最大的动力

如有错误❌疑问💬欢迎各位指出。

主页共饮一杯无的博客汇总👨‍💻

保持热爱,奔赴下一场山海。🏃🏃🏃

发布于: 刚刚阅读数: 4
用户头像

鲜衣怒马意气风发,愿你归来仍是少年。 2018-10-19 加入

全栈开发者,CSDN博客专家,51CTO 专家博主,阿里云专家博主,华为云享专家,持续输出干货,欢迎关注。

评论

发布
暂无评论
群主发红包带你深入了解继承和super、this关键字_Java_共饮一杯无_InfoQ写作社区