使用System.nanoTime()

时间:2019-03-09 12:45:29

标签: java random random-seed nanotime current-time

我试图在不使用任何Random()函数的情况下生成无意义的单词。我发现我可以使用当前时钟或鼠标坐标。我选择使用当前时钟。这是我写的代码。

private final char[] charray = {'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z',
    'A','B','C','D','E','F','G','H','I','J','K','L','M','N','O','P','Q','R','S','T','U','V','W','X','Y','Z'};

private char getRandomCharacter(){
    return charray[random(charray.length)];
}

private int random(int value){
    value =(int) System.nanoTime()% 52;
    return value;
}

protected Randomizer(){
    boolean running = true;
    int count = 0;
    int max = 5;
    while(running){
        StringBuilder sb = new StringBuilder();
        int size = random(25) + random(25);
        for (int i = 0; i < size; i++) {
            sb.append(getRandomCharacter());
        }
        System.out.println("Random Line : " + sb.toString());

        if (count++ == max) {
            running = false;
            System.out.println("All of them are random.");
        }
    }
}

public static void main(String[] args) {
    new Randomizer();
}

我期望的输出是:

  

axdlMkjiIfjcmqQopv等。

但是我得到这样的东西:

  

ZNNrrrUUUUxxxxbbbhhhLLLLoooRRRRRvvvYYYYBBBBfffI或   JmmmPPKKKKnnnnRRBeeHHHHlllllOOOOrrrVVVVV

为什么连续太多。 nanoTime太慢了吗?还是在那之前呢?在此之前,我使用currentTimeMillis,情况更糟。我不知道,也找不到任何有关如何使用当前时钟随机化的信息。

3 个答案:

答案 0 :(得分:2)

可以使用时序数据(除其他数据外)来种子随机数生成器,但使用时序数据进行随机性不是简单。可能,但是可能很慢。例如,请参阅我编写的here代码,该代码说明如何使用其他数据为安全的随机实例(H2数据库,MathUtils.generateAlternativeSeed)提供种子。它使用:

  • System.currentTimeMillis()
  • System.nanoTime()
  • 新Object()。hashCode()
  • Runtime.freeMemory(),maxMemory(),totalMemory()
  • System.getProperties()。toString()
  • InetAddress
  • 更多计时数据

这是为了播种安全的伪随机数生成器。这样可以确保即使在没有任何其他运行,不知道当前时间并且没有UI的系统上,您也可以获得足够的熵。

但是,依赖时序数据非常困难,因为它取决于操作系统,方法调用之间的时间,编译器和硬件。

答案 1 :(得分:2)

通过将charray定义为Character[] charray并将其作为列表,可以得到更好的结果:List<Character> chars = Arrays.asList(charray);
getRandomCharacter()方法中使用此列表:

 private char getRandomCharacter(){
     Collections.shuffle(chars); // shuffle before each use 
     return chars.get(random(chars.size()));
 }

当然可以解决random

private int random(int value){
    return (int) System.nanoTime()% value;
}

输出:

  

随机行:tjnBUxTDeTulHfLqnEJBRBLXFqqikUYyrREzzwPwG
随机   线路:MZpzJbOyCaqraRPsQPSK
随机线路:cEzKcsNHTmoVmT
  随机行:CmGXpDHGOsUufSxxStDVQruR
随机行:   XtFKmOAIisnXEdPikhAIcfzD
随机行:   GVxdnwgWLKZvQIGuofCIhiiUbKsEbmAyzVfNNPM

答案 2 :(得分:1)

仅使用时间是有问题的,因为它限制了您可以请求随机编号的频率,并且还非常依赖于实现。

更好的方法是将时间用作种子,然后使用伪随机数生成器,例如linear congruential generator。您在this answer中有更多信息。考虑到这种随机数生成器算法不是secure,而且正如Thomas所指出的那样,如果要在所有系统中使用安全的RNG,仅使用时间作为种子可能​​不够。

因此您的代码可能如下所示:

replace()