首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >为什么播种这个随机对象会导致意外的比较器行为?

为什么播种这个随机对象会导致意外的比较器行为?
EN

Stack Overflow用户
提问于 2014-03-23 04:14:01
回答 3查看 82关注 0票数 0

我是一个学生,教自己如何比较,我遇到了奇怪的行为,我有困难的理解。我试图使用Arrays.sort和一个随机对象对数组进行洗牌,但是当我为随机对象注入种子时,洗牌要么使数组倒转,要么什么也不做。如果不播种随机对象,就会像预期的那样对其进行洗牌。

无种子:

代码语言:javascript
运行
复制
Integer[] integers = new Integer[]{1, 2, 3, 4, 5, 6};
Arrays.sort(integers, new Comparator<T>() {
        @Override
        public int compare(T o1, T o2) {
            return new Random().nextInt();
        }
    });

>>> [4, 3, 5, 2, 1, 6] 

种子:

代码语言:javascript
运行
复制
Integer[] integers = new Integer[]{1, 2, 3, 4, 5, 6};
Arrays.sort(integers, new Comparator<T>() {
        @Override
        public int compare(T o1, T o2) {
            return new Random(System.currentTimeMillis()).nextInt();
        }
    });

>>> [1, 2, 3, 4, 5, 6] or [6, 5, 4, 3, 2, 1]

有人能帮我理解这种行为吗?

我明白这是个糟糕的密码。这些只是玩具的例子,帮助我熟悉比较器的机制。不管怎么说,我对由此产生的行为更感兴趣。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2014-03-23 04:18:56

种子数控制您生成的随机数流。如果你用同样的种子播种,你每次都会得到相同的序列。您每次在这里播种相同的数字(因为执行排序甚至不需要1ms ),所以nextInt()总是返回相同的值。因此,它总是返回相同的类型。由于您的数据是排序进来的,这将导致它保持相同的顺序。

如果您想要正确地种子,创建一次随机对象,并调用它,而不是每次用相同的种子重新创建。

票数 3
EN

Stack Overflow用户

发布于 2014-03-23 04:18:35

由于代码运行速度快,创建的许多随机对象将获得相同的种子。

例如,您可以在比较器中添加一个简短的Thread.sleep(10)。或者你可以跑:

代码语言:javascript
运行
复制
  for (int i = 0; i < 7; i++) {
     System.out.println("" + new Random(System.currentTimeMillis()).nextInt());
  }

你明白为什么这是很糟糕的代码,对吧?您只会创建一个随机对象一次,然后在您的比较器中使用相同的对象,对吗?

票数 2
EN

Stack Overflow用户

发布于 2015-02-24 18:40:09

即使只使用单个随机对象,使用随机比较器仍然会得到意想不到的行为。来自比较国提供的文件

实现者必须确保sgn(比较(x,y)) == -sgn(比较( y,x))对所有x和y。

如果不遵循该实现要求,您将得到未定义的行为。您可以在这里看到由此产生的洗牌的可视化结果:http://bost.ocks.org/mike/shuffle/compare.html

(注意:这里使用的是Javascript,而不是Java,但这个问题也适用于Java。)

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22586997

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档