简单的随机立减——JAVA实现概率计算(数字不同范围按照不同几率产生随机数)-记录

JAVA实现概率计算(数字不同范围按照不同几率产生随机数)

转载于JAVA实现概率计算(数字不同范围按照不同几率产生随机数)

原文是int 转成了double-记录

maven

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

import org.apache.commons.lang3.RandomUtils;

import java.math.BigDecimal;
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<Double> 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必须等于百分比个数");
        }
        BigDecimal bigDecimal = new BigDecimal(0);
        for(Double p:percents){
            if(p<0 || p>100){
                throw  new IllegalArgumentException("百分比必须在[0,100]之间");
            }
            bigDecimal = bigDecimal.add(new BigDecimal(Double.toString(p)));
        }
        if(bigDecimal.doubleValue() != 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>();
        double scopeMax = 0.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;
        double randomInt = RandomUtils.nextDouble(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 double percent; //百分比

        public double percentScopeMin; //百分比转换为[1,100]的数字的最小值
        public double percentScopeMax; //百分比转换为[1,100]的数字的最大值
    }

    public static void main(String[] args) {
        List<Double> separates = new ArrayList<Double>();
        separates.add(1.0);
        separates.add(10.0);
        separates.add(20.0);
        List<Double> percents = new ArrayList<Double>();
        percents.add(90.0);
        percents.add(5.0);
        percents.add(4.0);
        percents.add(1.0);
        int a = 0;
        int b = 0;
        int c = 0;
        int d = 0;
        for(int i=0;i<10000;i++) {
            double number = produceRateRandomNumber(0, 10000, separates, percents);
            if (number < 1) {
                a++;
            }
            if (1.0 < number && number < 10.0) {
                b++;
            }
            if (10.0 < number && number < 20.0) {
                c++;
            }
            if (20.0 < number) {
                d++;
            }
            System.out.println(String.format("%.2f",number));
        }
        System.out.println(" a = " + a);
        System.out.println(" b = " + b);
        System.out.println(" c = " + c);
        System.out.println(" d = " + d);
//        System.out.println(" number = " + number);
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: jmu-java-随机数-使用蒙特卡罗法计算圆周率的值 蒙特卡罗法是一种通过随机抽样来解决问题的方法。在计算圆周率的问题中,我们可以通过随机生成点的方式来模拟落在圆内的点和落在正方形内但不在圆内的点的数量,从而得到圆的面积和正方形的面积,进而计算出圆周率的值。 在使用Java编写程序时,我们可以利用Java中的随机数生成器来生成随机点的坐标,然后通过判断点是否在圆内来计算圆内和圆外的点的数量。最后,根据圆的面积和正方形的面积的比例,计算出圆周率的值。 这种方法虽然简单,但是需要生成大量的随机点才能得到较为准确的结果。因此,在实际应用中,需要根据具体问题的要求来确定生成随机点的数量。 ### 回答2: 蒙特卡罗法是一种基于随机数的数值解法,其大致思想是利用概率统计原理构造随机过程,并利用随机过程解决计算问题。在计算圆周率时,可以使用蒙特卡罗法来估算其值。 具体来说,我们可以通过生成随机数的方式来模拟在一个正方形内部随机投点,将正方形视作一个包含在其中的圆的外接正方形,那么正方形内部的点有多少个落在圆的内部,就可以用这个数量来近似出圆的面积。根据圆的公式,可以通过这个近似面积来计算出圆的半径和周长,从而得出圆周率的值。 在实现上,我们可以使用Java中的Math类中的随机数函数来生成一堆随机点,然后统计其中有多少个点落在圆的内部。最终得到的比率就是圆的面积和正方形面积的比例,用这个比例乘以4就可以得到圆周率的值了。 然而,需要注意的是,这种随机投点的方法是一种概率性的计算方法,随着投点数量的增加,估算出的圆周率值会越来越接近实际值。因此,我们需要在程序中设置投点数量,以达到较为准确的计算结果。 总的来说,使用蒙特卡罗法计算圆周率的值是一种在计算中有很大应用价值的技术,特别适用于大量的模拟计算和参数估算。在Java中,通过使用Math类中的随机数函数,可以快速便捷地实现这种方法。 ### 回答3: JMU-Java是一种集成了Java语言与数学算法的库,通过使用随机数生成器可以使用蒙特卡罗法计算圆周率的值。这种方法利用了概率统计的思想,在一个较大的正方形内随机生成若干个点,并统计落在圆内的点的数量,然后通过数学公式计算圆周率的值。这个方法的精确度可以通过增加正方形内的点数来提高,因此随着正方形中点数的增大,计算出的圆周率会越来越接近真实值。 使用JMU-Java库进行计算时,可以通过以下步骤实现: 首先需要导入库文件,然后设置随机数发生器并选择生成点的数量和正方形的边长。然后,利用循环语句和if语句进行判断,统计出正方形内等分的子正方形和落在圆内的点的数量,然后通过公式计算出圆的面积和圆周率的值,并将结果输出。 使用蒙特卡罗法计算圆周率的优点是计算简单易懂,并且可以通过增加点的数量提高计算精确度。但是,它也存在一些缺点,如计算误差较大,需要大量计算,因此会消耗较多的时间和计算资源。另外,在使用时需要注意数据的范围和运算精度,以免影响计算结果的准确性。 总之,蒙特卡罗法计算圆周率的方法是一种简单、直观、易于实现的方法。通过使用JMU-Java库,可以方便地实现该方法,并且在学习编程和数学等方面都具有一定的帮助作用。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

yizhi-w

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值