- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
给定一组数字,从 1 到 n,我需要对所有可能对的所有集合建模。
换句话说,一组开奖由所有号码组成。数字是成对的。如果计数是奇数 - 允许一个条目和一个数字(实际上这是必需的)。集合需要是唯一的,即对 [1,2] 与 [2,1] 相同(编辑:和解决方案:[1,2] [3,4] 与 [3,4] [1, 2]).
例如,当n等于5时,可以创建如下集合:
[1,2] [3,4] [5]
[1,2] [3,5] [4]
[1,2] [4,5] [3]
[1,3] [2,4] [5]
[1,3] [2,5] [4]
[1,3] [4,5] [2]
....
我几乎可以对解决方案进行建模,但是唯一性约束对我来说很难实现。
此外 - 我觉得我的解决方案缺乏性能。我现在问题空间很大,但对我来说,即使 n=12 计算也需要 5 分钟以上。
public void compute(HashSet<Number> numbers, ArrayList<Pair> pairs) {
if (numbers.size() <= 1) {
print(pairs);
} else {
for (Number number1 : numbers) {
for (Number number2 : numbers) {
if (number1 != number2) {
Set<Number> possibleNumbers = new HashSet<Number>(numbers);
List<Pair> allPairs = new ArrayList<Pair>(pairs);
possibleNumbers.remove(number1);
possibleNumbers.remove(number2);
allPairs.add(new Pair(number1, number2));
compute(possibleNumbers, allPairs);
}
}
}
}
}
compute(numbers, new ArrayList<Pair>());
对于 n=3,我得到了双倍的解决方案:
[0 1] [2]
[0 2] [1]
[1 0] [2]
[1 2] [0]
[2 0] [1]
[2 1] [0]
那么:我应该如何实现这个问题来摆脱重复。如何改进实现以加快处理速度?
最佳答案
你可以做两件事:
您可以继续当前的方法,但是在添加对时,您需要:
a) 确保每一对总是以某种顺序排列,例如总是[smallerNumber, biggerNumber]
b) 保持 allPairs 列表排序
然后当您完成计算时,删除重复项将变得微不足道。这是一个糟糕的方法,因为它会非常慢。
我将在这里做出 2 个假设:
基本上我们会有一个递归算法(就像你的)。输入将是“数字”-> 有序(升序)列表,输出将是一组成对的列表。让我们再次调用此方法 compute(List numbers)。
-> 输入列表“numbers”有 1 个或 2 个元素时的基本情况。在这种情况下,返回一个包含一个列表的集合,该列表包含一个包含该 1 元素或两者的“对”。 IE。 numbers = [2,3] 或 numbers = [2] 然后你返回一个包含一个列表的集合,其中包含一个“对”(2,3)或(3)。
-> 对于包含第一个元素的列表“数字”中的每一对数字(即在第一级递归中数字 = [1,2,3,4,5] 它将是 [1,2] , [1,3], [1,4], [1,5], [1,6]) 调用 Set<List<Pair> result = compute(numbers.minus(currentPair))
.迭代该集合并在每个元素的开头添加当前对。
例子:
result = empty
numbers = [1,2,3,4,5]
first = (1,2), compute([3,4,5])
first = (3,4), compute([5])
return (5)
result = result + [(3,4)(5)]
first = (3,5) compute([4])
return (4)
result = result + [(3,5),(4)] (a this point result is [ [(3,4)(5)], [(3,5)(4)] ]
add (1,2) to each list in result which gives us [ [(1,2)(3,4)(5)], [(1,2)(3,5)(4)] ]
first = (1,3), compute([2,4,5])
first = (2,4), compute([5])
return (5)
result = result + [(2,4)(5)]
first = (2,5) compute([4])
return (4)
result = result + [(2,5),(4)] (a this point result is [ [(2,4)(5)], [(2,5)(4)] ]
add (1,3) to each list in result which gives us [ [(1,3)(2,4)(5)], [(1,3)(2,5)(4)] ]
at this point we have:
[ [(1,2)(3,4)(5)], [(1,2)(3,5)(4)], [(1,3)(2,4)(5)], [(1,3)(2,5)(4)] ]
对 (1,4)、(1,5) 和 (1,6) 继续此操作,您就完成了。您不必从 (2,3) 开始执行计算 ([1,4,5]),因为这会导致重复。
这个算法应该也适用于偶数集。
还没有测试过,没有证明,但看起来不错,应该很容易编码,如果它有效,那么它应该非常快,因为它只会进行必要的计算。
在代码中(我在这里写了整个代码,所以它完全没有经过测试,可能包含编译和逻辑错误!):
public Set<List<Pair>> compute(List<Integer> numbers) {
if(numbers.size() < 3) {
// Base case
List<Pair> list = new ArrayList<>();
list.add(new Pair(numbers));
Set<List<Pair>> result = new HashSet<>();
result.add(list);
return result;
} else {
Set<List<Pair>> result = new HashSet<ArrayList<>>();
// We take each pair that contains the 1st element
for(int i = 1; i < numbers.size(); i++) {
Pair first = new Pair(numbers.get(0), numbers.get(i));
// This is the input for next level of recursion
// Our numbers list w/o the current pair
List<Integers> nextStep = new ArrayList<>(numbers);
nextStep.remove(i);
nextStep.remove(0);
Set<List<Pair>> intermediate = null;
if(nextStep.size() % 2 == 0) {
intermediate = compute(nextStep);
} else {
intermediate = compute(numbers).addAll( firstElementSingle(numbers) ),compute( nextStep );
}
for(List<Pair> list : intermediate ) {
// We add the current pair at the beginning
list.add(0, first);
}
result.addAll(intermediate);
}
return result;
}
}
如果我错过了什么,我自己也很好奇,所以我期待您的反馈:-)
@编辑:
正如@yusuf 所指出的,当输入是奇数列表时,这会遗漏一些排列。它会错过所有 [1] 是单个元素的结果。
但是如果我在这种情况下没有弄错(奇数个元素),这应该可行:
compute(numbers).addAll( firstElementSingle(numbers) )
firstElementSingle 在哪里:
private Set<List<Integer>> firstElementSingle(List<Integer> numbers) {
Set<List<Integer>> result compute(numbers.subList(1,numbers.size()) );
for(List<Integer> list : result) {
list.add(numbers.get(0));
}
return result;
}
而且仍然只会产生必要的结果。
关于java - 所有可能的对,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19466600/
降本手段一招鲜,增效方法吃遍天; 01 互联网行业里; 降本策略千奇百怪,手段却出奇一致;增效方法五花八门,手段更是花里胡哨; 对于企业来说;
有什么方法可以使用 angularjs 中的部分进行代码分组吗? 原因 --- 我的 Controller 包含太多代码。该 Controller 包含了多个方法和大量功能的代码,降低了代码的可读性。
不幸的是,我的数据库的数据模型必须改变,所以我正在寻找最轻松的方式来迁移我的数据。 此时情况如何: create table cargo{ id serial primary key, per
在 QTextEdit 对象中,假设我想知道字符在鼠标光标下的位置。 我会写... void MyQTextEditObject::mousePressEvent(QMouseEvent* mouse
是否可以在 C++ 中返回一个 return 语句或做一些具有类似功能的事情? 例如,如果代码中有几个函数将指针作为输入,并且每个函数都检查指针是否为 nullptr,这将很方便。如果它是一个 nul
我的 PC 上有一个控制台应用程序,它是 signalR 服务器。 我有一个 html 页面,它是互联网上的 signalR 客户端。但我尝试连接服务器,但我有一个错误的请求 400 错误。如果服务器
我想将应用程序作为后台进程运行。当点击应用程序图标时,它不会显示任何 View ,只会启动后台进程。 最佳答案 对于 iOS 这是不可能的,但是对于 android,react native 有 he
我知道有(昂贵的)框架可以让你在 VS C# 中编写 android 应用程序并将其编译为 android apk。 我也知道,可以在 VS 中编写 Java 应用程序(link)。 是否有可能,甚至
我在做: can :manage, :all if user.role == 'admin' can :approve, Anuncio do |anuncio| anuncio.try(:apr
我是一名优秀的程序员,十分优秀!