java random by weight

使用反射实现了一个通用的random by weight,只要被随机的结构中带有getWeight函数即可

public static <T extends Object>
T getRandomObjByWeight(List<T> objs, Class<?> clazz) {
	try {
		Method getWeightFunc = clazz.getMethod("getWeight");

		Double totalWeight = 0.0;
		for(Object obj : objs) {
			Integer weight = (Integer)getWeightFunc.invoke(obj);
			totalWeight += Double.valueOf(weight);
		}

		double random = Math.random() * totalWeight;
		for(T obj : objs) {
			Integer weight = (Integer)getWeightFunc.invoke(obj);
			random -= Double.valueOf(weight);
			if (random <= 0.0) {
				return obj;
			}
		}
	} catch (Exception e) {
		LogUtil.LOG_BASE.error("getRandomObjByWeight异常",  e.getMessage(),  e);
	}

	return null;
}

测试代码

public class Item {
	private Integer id;
	private Integer weight;

	public Item(Integer id, Integer weight) {
		this.id = id;
		this.weight = weight;
	}

	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	public Integer getWeight() {
		return weight;
	}

	public void setWeight(Integer weight) {
		this.weight = weight;
	}
}

public static void main(String[] args) {
	List<Item> items = new ArrayList<>();

	items.add(new Item(1, 50));
	items.add(new Item(2, 50));
	items.add(new Item(3, 50));

	Map<Integer, Integer> counts = new HashMap<>();

	for (int i = 0; i < 1000; ++i) {
		Item item = GameUtil.getRandomObjByWeight(items, Item.class);

		Integer count = counts.get(item.id);
		if (count == null) {
			counts.put(item.id, 1);
		} else {
			counts.put(item.id, count+1);
		}
	}

	Integer total = 0;
	for (Map.Entry<Integer, Integer> pair : counts.entrySet()) {
		System.out.println("id=" + pair.getKey() + " count=" + pair.getValue());

		total += pair.getValue();
	}

	System.out.println("total=" + total);
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值