程序中经常遇到随机送红包之类的情景,这个随机还得指定概率,比如10%的机率可以得到红包。那么java怎么实现一个简单的概率计算了,见如下例子:

int randomInt =  RandomUtils.nextInt(1,101);
if(randomInt <= 10){ //100里面1个数,小于等于10的概率就是10%
//do something
}

RandomUtils工具类是commons-lang3包里面的

<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
<version>3.7</version>
</dependency>

如果要在某个数字区间产生一个随机数,区间内部在不同的片段几率不同如何实现呢?经常有这样的场景,比如,随机赠送红包,范围0.1元-100元,0.1-1元的概率是90%,1元-10元的概率是9%,10元-100元的概率是1%,也就是说数额越大得到的几率越小!实现的原理如下图:

原理就是,将范围分割成一个个子范围(片段),具体采用哪个范围,再用机率判断。片段机率可以依次排好序,映射成[1,100]之间的数字。然后随机一个[1,100]之间的数,该数落在哪个区间,就采用哪个片段产生随机数。具体源代码如下:

package com.hdwang;

import org.apache.commons.lang3.RandomUtils;

import java.util.ArrayList;
import java.util.List; /**
* 按几率产生随机数
* 例如,产生0.1-100的随机数,0.1-1的几率是90%,1-10的几率是9%,10-100的几率是1%
*/
public class RateRandomNumber { /**
* 产生随机数
* @param min 最小值
* @param max 最大值
* @return 随机结果
*/
public static double produceRandomNumber(double min,double max){
return RandomUtils.nextDouble(min,max); //[min,max]
} /**
* 按比率产生随机数
* @param min 最小值
* @param max 最大值
* @param separates 分割值(中间插入数)
* @param percents 每段数值的占比(几率)
* @return 按比率随机结果
*/
public static double produceRateRandomNumber(double min,double max,List<Double> separates,List<Integer> percents){
if(min > max){
throw new IllegalArgumentException("min值必须小于max值");
}
if(separates == null || percents==null || separates.size()==0){
return produceRandomNumber(min,max);
}
if(separates.size() +1 != percents.size()){
throw new IllegalArgumentException("分割数字的个数加1必须等于百分比个数");
}
int totalPercent = 0;
for(Integer p:percents){
if(p<0 || p>100){
throw new IllegalArgumentException("百分比必须在[0,100]之间");
}
totalPercent += p;
}
if(totalPercent != 100){
throw new IllegalArgumentException("百分比之和必须为100");
}
for(double s:separates){
if(s <= min || s >= max){
throw new IllegalArgumentException("分割数值必须在(min,max)之间");
}
}
int rangeCount = separates.size()+1; //例如:3个插值,可以将一个数值范围分割成4段
//构造分割的n段范围
List<Range> ranges = new ArrayList<Range>();
int scopeMax = 0;
for(int i=0;i<rangeCount;i++){
Range range = new Range();
range.min = (i==0 ? min:separates.get(i-1));
range.max = (i== rangeCount-1 ?max:separates.get(i));
range.percent = percents.get(i); //片段占比,转换为[1,100]区间的数字
range.percentScopeMin = scopeMax +1;
range.percentScopeMax = range.percentScopeMin + (range.percent-1);
scopeMax = range.percentScopeMax; ranges.add(range);
}
//结果赋初值
double r = min;
int randomInt = RandomUtils.nextInt(1,101); //[1,100]
for(int i=0;i<ranges.size();i++){
Range range = ranges.get(i);
//判断使用哪个range产生最终的随机数
if(range.percentScopeMin <= randomInt && randomInt <= range.percentScopeMax){
r = produceRandomNumber(range.min,range.max);
break;
}
}
return r;
} public static class Range{
public double min;
public double max;
public int percent; //百分比 public int percentScopeMin; //百分比转换为[1,100]的数字的最小值
public int percentScopeMax; //百分比转换为[1,100]的数字的最大值
} public static void main(String[] args) {
List<Double> separates = new ArrayList<Double>();
separates.add(1.0);
separates.add(10.0);
List<Integer> percents = new ArrayList<Integer>();
percents.add(90);
percents.add(9);
percents.add(1);
for(int i=0;i<100;i++) {
double number = produceRateRandomNumber(0.1, 100, separates, percents);
System.out.println(String.format("%.2f",number));
}
}
}

JAVA实现概率计算(数字不同范围按照不同几率产生随机数)的更多相关文章

  1. java根据概率生成数字

    /** * JAVA 返回随机数,并根据概率.比率 * @author zhanglei * */ public class MathRandom { /** * 0出现的概率为%50 */ publ ...

  2. Java中精确的数字计算类BigDecimal

    在日常开放当中需要我们计算数字,利率.通常Java的做法是使用Math相关的API.但是,这样做是不够精确的,由于float和double不能进行计算,如果强行进行计算会使得计算不准确.造成难以挽回的 ...

  3. HMM的概率计算问题和预测问题的java实现

    HMM(hidden markov model)可以用于模式识别,李开复老师就是采用了HMM完成了语音识别. 一下的例子来自于<统计学习方法> 一个HMM由初始概率分布,状态转移概率分布, ...

  4. 算法笔记_155:算法提高 概率计算(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 生成n个∈[a,b]的随机整数,输出它们的和为x的概率. 输入格式 一行输入四个整数依次为n,a,b,x,用空格分隔. 输出格式 输出一行 ...

  5. Java实现 蓝桥杯 算法提高 概率计算

    算法提高 概率计算 时间限制:1.0s 内存限制:256.0MB 问题描述 生成n个∈[a,b]的随机整数,输出它们的和为x的概率. 输入格式 一行输入四个整数依次为n,a,b,x,用空格分隔. 输出 ...

  6. Java完成简单猜数字游戏v2.0

    猜数字游戏v2.0 优化了获取随机数.输入数据超出边界值的代码,并增加了异常处理,能够在玩家输入错误数据错误时给出可靠指引,希望对和我一样的新人有帮助, 最后希望有大神愿意帮我解决代码优化的问题,谢谢 ...

  7. MT2018笔试题之计算数字位数

    一.计算数字位数 1.题目 给定一个数字T,计算从1到T的所有正整数的位数和.比如T=13,则12345678910111213有17位数字. 输入描述 3 13 4 5 输出 17 4 5 2.思路 ...

  8. 计算数字出现的次数 Exercise07_03

    import java.util.Scanner; /** * @author 冰樱梦 * 时间:2018年下半年 * 题目:计算数字出现的次数 * */ public class Exercise0 ...

  9. Java 高精度浮点数计算工具

    说起编程中的高精度数值,我第一反应就是double类型了.的确,double阶码11位,尾数52位,几乎能应对任何苛刻的要求......然而,当我天真地尝试用double来算泰勒展开式的函数值,离散代 ...

随机推荐

  1. 2017-2018-2 20155333 《网络对抗技术》 Exp1 PC平台逆向破解

    2017-2018-2 20155333 <网络对抗技术> Exp1 PC平台逆向破解 1. 逆向及Bof基础实践说明 1.1 实践目标 本次实践的对象是一个名为pwn1的linux可执行 ...

  2. 20155337《网络对抗》Exp5 MSF基础应用

    20155337<网络对抗>Exp5 MSF基础应用 实践目标 本实践目标是掌握metasploit的基本应用方式,重点常用的三种攻击方式的思路.具体需要完成: 1.1一个主动攻击实践,如 ...

  3. 【转载】C++引用详解

    原文:http://www.cnblogs.com/gw811/archive/2012/10/20/2732687.html 引用的概念 引用:就是某一变量(目标)的一个别名,对引用的操作与对变量直 ...

  4. banner 跟随鼠标呈现视差效果

    参考 Element 官网,利用 js / jq 和 css3, 实现某图片随着鼠标移动呈现的视差效果. <!DOCTYPE html> <html> <head> ...

  5. C#用Infragistics 导入导出Excel(一)

    最近项目中有数据的导入导出Excel的需求,这里做简单整理. 公司用的是Infragistics的产品,付费,不需要本地安装Office. 有需要的朋友可以下载 Infragistics.2013.2 ...

  6. 10、Dockerfile实战-PHP

    一.镜像制作步骤 安装编译依赖包 编译安装 配置 二.编写Dockerfile FROM centos:7 MAINTAINER QUNXUE RUN yum install -y gcc gcc-c ...

  7. win10+anaconda3+python3.6+opencv3.1.0

    最近用windows系统比较多,就想在win10下搞一下深度学习这一方面的研究,那么就需要配置好环境巴拉巴拉的一堆东西.默默记个笔记,正所谓“好记性不如烂笔头”. 1.安装Anaconda 这个是一个 ...

  8. 阿里云ECS 固定带宽变为按量付费的方式

    阿里云ECS 固定带宽变为按量付费的方式 阿里云控制台 2.升降配置-降低配置-降低至最低配置 3.为按量带宽设置一个峰值,例如100M. 4.过几分钟,就自动变为按量付费的带宽了.

  9. PAT-1010 Radix

    1010 Radix (25 分) Given a pair of positive integers, for example, 6 and 110, can this equation 6 = 1 ...

  10. 《Linux内核设计与实现》第五周读书笔记——第十一章

    <Linux内核设计与实现>第五周读书笔记——第十一章 20135301张忻 估算学习时间:共2.5小时 读书:2.0 代码:0 作业:0 博客:0.5 实际学习时间:共3.0小时 读书: ...