为什么这个未使用的流会影响结果?

huangapple go评论75阅读模式
英文:

Why does this unused stream have an impact on the result?

问题

以下是翻译好的代码部分:

  1. private List<Var> getAllDefinedVars(List<Var> allVars) {
  2. List<Var> collect = allVars.stream().filter(v -> false).collect(Collectors.toList());
  3. return collect;
  4. }

请注意,此行代码实际上没有对结果产生影响,因为它只是使用流过滤了一个始终为假的条件,从而创建了一个空的列表。删除这行代码不应该对最终结果产生影响。

英文:

Why does the following effectivly unused line (in the method: getAllDefinedVars) have an impact on the end result:
List<Var> collect = allVars.stream().filter(v -> false).collect(Collectors.toList());

If I remove the whole method and the one line of code, which is calling this method (first line in generateOneSetOfBools), I get an other result in the end.
I would expect such a behavior if...

  1. the mentioned line had an impact on the List allVars or any other variable
  2. the result of the stream would be used

As far as I can see, none of this happens. Therefore a removal of this whole method should have no impact on the result.

To convince yourself you can run the main the first time with the method containing the stream and the second time without this method and then compare the output.

  1. public class PairwiseMain {
  2. private static PairwisePermutator pp = new PairwisePermutator();
  3. public static void main(String[] args) {
  4. for(int i = 0; i &lt; 20; i++) {
  5. pp.permutate(i);
  6. System.out.println(&quot;----------------&quot;);
  7. }
  8. }
  9. }
  1. import java.util.ArrayList;
  2. import java.util.Arrays;
  3. import java.util.Collections;
  4. import java.util.HashMap;
  5. import java.util.HashSet;
  6. import java.util.LinkedHashSet;
  7. import java.util.List;
  8. import java.util.Map;
  9. import java.util.Set;
  10. import java.util.stream.Collectors;
  11. import java.util.stream.IntStream;
  12. import java.util.stream.Stream;
  13. public class PairwisePermutator {
  14. private int n;
  15. public Stream&lt;boolean[]&gt; permutate(int n) {
  16. this.n = n;
  17. List&lt;Var&gt; vars = generateRequiredVars(n);
  18. List&lt;TupleSet&gt; table = generateTable(vars);
  19. generateCombinations(table, vars);
  20. return null;
  21. }
  22. private void generateCombinations(List&lt;TupleSet&gt; table, List&lt;Var&gt; vars) {
  23. TupleSet start = findStartTupleSet(table);
  24. if(start == null) {
  25. return;
  26. }else {
  27. Tuple t = start.tuples.get(0);
  28. start.setVars(t);
  29. generateOneSetOfBools(table, vars, new HashSet&lt;&gt;(Arrays.asList(start)));
  30. }
  31. System.out.println(Arrays.toString(vars.toArray()));
  32. resetAllVars(vars);
  33. generateCombinations(table, vars);
  34. }
  35. private void resetAllVars(List&lt;Var&gt; vars) {
  36. vars.stream().forEach(v -&gt; v.value = null);
  37. }
  38. private void generateOneSetOfBools(List&lt;TupleSet&gt; table, List&lt;Var&gt; allVars, Set&lt;TupleSet&gt; start) {
  39. List&lt;Var&gt; alreadyDefinedVars = getAllDefinedVars(allVars); //REMOVAL OF THIS LINE SHOULD HAVE NO IMPACT
  40. Map&lt;TupleSet, Integer&gt; relevant = findRelevantTuplesInOtherTupleSets(table, start);
  41. boolean changes = false;
  42. boolean existsMultipleOptions = false;
  43. TupleSet minimalMultipleOptions = null;
  44. int minimalMultipleOptionsNumber = Integer.MAX_VALUE;
  45. for(Map.Entry&lt;TupleSet, Integer&gt; entry : relevant.entrySet()) {
  46. if(entry.getValue() == -1) {
  47. removeTuple(entry.getKey());
  48. start.add(entry.getKey());
  49. changes = true;
  50. }else if(entry.getValue() == 1) {
  51. changes = true;
  52. removeTuple(entry.getKey(), table, allVars);
  53. start.add(entry.getKey());
  54. }else if(entry.getValue() &gt;= 2) {
  55. existsMultipleOptions = true;
  56. if(entry.getValue() &lt; minimalMultipleOptionsNumber) {
  57. minimalMultipleOptionsNumber = entry.getValue();
  58. minimalMultipleOptions = entry.getKey();
  59. }
  60. }
  61. }
  62. if(!changes &amp;&amp; existsMultipleOptions) {
  63. removeRandomTuple(minimalMultipleOptions, start);
  64. }else if(!changes) {
  65. setVars(table);
  66. }
  67. if(areAllVarsDefined(allVars)) {
  68. removeMatchingTuples(table);
  69. return;
  70. }
  71. generateOneSetOfBools(table, allVars, start);
  72. }
  73. private List&lt;Var&gt; getAllDefinedVars(List&lt;Var&gt; allVars) {
  74. List&lt;Var&gt; collect = allVars.stream().filter(v -&gt; false).collect(Collectors.toList());
  75. return collect;
  76. }
  77. private void removeMatchingTuples(List&lt;TupleSet&gt; table) {
  78. for(TupleSet ts : table) {
  79. boolean v1 = ts.x.value;
  80. boolean v2 = ts.y.value;
  81. Tuple toRemove = null;
  82. for(Tuple t : ts.tuples) {
  83. if(t.a == v1 &amp;&amp; t.b == v2) {
  84. toRemove = t;
  85. }
  86. }
  87. if(toRemove != null) {
  88. ts.setVars(toRemove);
  89. }
  90. }
  91. }
  92. private boolean areAllVarsDefined(List&lt;Var&gt; allVars) {
  93. return allVars.stream().filter(v -&gt; v.value != null).count() == n;
  94. }
  95. private void removeRandomTuple(TupleSet minimalMultipleOptions, Set&lt;TupleSet&gt; start) {
  96. Tuple toRemove = minimalMultipleOptions.tuples.get(0);
  97. minimalMultipleOptions.setVars(toRemove);
  98. start.add(minimalMultipleOptions);
  99. }
  100. private void setVars(List&lt;TupleSet&gt; table) {
  101. boolean foundOne = false;
  102. for(TupleSet ts : table) {
  103. if(ts.x.value == null &amp;&amp; ts.y.value == null) {
  104. foundOne = setVars(ts);
  105. }
  106. }
  107. if(!foundOne) {
  108. for(TupleSet ts : table) {
  109. if(ts.x.value == null || ts.y.value == null) {
  110. if(ts.tuples.isEmpty()){
  111. ts.x.value = true;
  112. ts.y.value = false;
  113. }else {
  114. ts.setVars(ts.tuples.get(0));
  115. }
  116. }
  117. }
  118. }
  119. }
  120. private boolean setVars(TupleSet ts) {
  121. boolean foundOne;
  122. if(ts.tuples.isEmpty()){
  123. ts.x.value = true;
  124. ts.y.value = false;
  125. foundOne = true;
  126. }else {
  127. ts.setVars(ts.tuples.get(0));
  128. foundOne = true;
  129. }
  130. return foundOne;
  131. }
  132. private void removeTuple(TupleSet ts, List&lt;TupleSet&gt; table, List&lt;Var&gt; vars) {
  133. Tuple toRemove = null;
  134. if(ts.x.value != null) {
  135. for(Tuple t : ts.tuples) {
  136. if(t.a == ts.x.value) {
  137. toRemove = t;
  138. }
  139. }
  140. }else if(ts.y.value != null) {
  141. for(Tuple t : ts.tuples) {
  142. if(t.b == ts.y.value) {
  143. toRemove = t;
  144. }
  145. }
  146. }
  147. if(toRemove != null)
  148. ts.setVars(toRemove);
  149. }
  150. private void removeTuple(TupleSet ts) {
  151. boolean v1 = ts.x.value;
  152. boolean v2 = ts.y.value;
  153. Tuple toRemove = null;
  154. for(Tuple t : ts.tuples) {
  155. if(t.a == v1 &amp;&amp; t.b == v2) {
  156. toRemove = t;
  157. }
  158. }
  159. if(toRemove != null) {
  160. ts.setVars(toRemove);
  161. }
  162. }
  163. private Map&lt;TupleSet, Integer&gt; findRelevantTuplesInOtherTupleSets(List&lt;TupleSet&gt; table, Set&lt;TupleSet&gt; alreadyVisited) {
  164. Map&lt;TupleSet, Integer&gt; freedomMap = new HashMap&lt;&gt;();
  165. for(TupleSet ts : table) {
  166. if(!alreadyVisited.contains(ts)) {
  167. int degreeOfFreedom = calculateDegreeOfFreedom(ts);
  168. freedomMap.put(ts, degreeOfFreedom); }
  169. }
  170. return freedomMap;
  171. }
  172. private int calculateDegreeOfFreedom(TupleSet ts) {
  173. List&lt;Var&gt; alreadyDefinedVars = new ArrayList&lt;&gt;();
  174. if(ts.x.value != null) {
  175. alreadyDefinedVars.add(ts.x);
  176. }
  177. if(ts.y.value != null) {
  178. alreadyDefinedVars.add(ts.y);
  179. }
  180. int degreeOfFreedom = 0;
  181. if(areBothDefinied(alreadyDefinedVars)) {
  182. degreeOfFreedom = -1;
  183. }else if(isExactlyOneDefinied(alreadyDefinedVars)) {
  184. Var defined = getDefinedVar(alreadyDefinedVars);
  185. if(defined == ts.x) {
  186. degreeOfFreedom = (int) ts.tuples.stream().filter(t -&gt; t.a == ts.x.value).count();
  187. }else if(defined == ts.y) {
  188. degreeOfFreedom = (int) ts.tuples.stream().filter(t -&gt; t.b == ts.y.value).count();
  189. }
  190. }else if(alreadyDefinedVars.isEmpty()) {
  191. degreeOfFreedom = ts.tuples.size();
  192. }
  193. return degreeOfFreedom;
  194. }
  195. private Var getDefinedVar(List&lt;Var&gt; alreadyDefinedVars) {
  196. return alreadyDefinedVars.get(0);
  197. }
  198. private boolean isExactlyOneDefinied(List&lt;Var&gt; alreadyDefinedVars) {
  199. return alreadyDefinedVars.size() == 1;
  200. }
  201. private boolean areBothDefinied(List&lt;Var&gt; alreadyDefinedVars) {
  202. return alreadyDefinedVars.size() == 2;
  203. }
  204. private TupleSet findStartTupleSet(List&lt;TupleSet&gt; table) {
  205. TupleSet startingTupleSet = null;
  206. for(TupleSet ts : table) {
  207. if(!ts.tuples.isEmpty()) {
  208. startingTupleSet = ts;
  209. }
  210. }
  211. return startingTupleSet;
  212. }
  213. private List&lt;Var&gt; generateRequiredVars(int n) {
  214. return IntStream.range(0, n).mapToObj(number -&gt; new Var()).collect(Collectors.toList());
  215. }
  216. private List&lt;TupleSet&gt; generateTable(List&lt;Var&gt; vars) {
  217. List&lt;TupleSet&gt; tupleSets = new ArrayList&lt;&gt;();
  218. int kStart = 1;
  219. for(int i = 0; i &lt; vars.size(); i++) {
  220. for(int k = kStart; k &lt; vars.size(); k++) {
  221. tupleSets.add(getInitSet(vars.get(i), vars.get(k)));
  222. }
  223. kStart++;
  224. }
  225. return tupleSets;
  226. }
  227. private TupleSet getInitSet(Var x, Var y){
  228. return new TupleSet(x, y, (new ArrayList&lt;&gt;(Arrays.asList( new Tuple(true, true),
  229. new Tuple(true, false),
  230. new Tuple(false, true),
  231. new Tuple(false, false)))));
  232. }
  233. private static class TupleSet{
  234. Var x;
  235. Var y;
  236. ArrayList&lt;Tuple&gt; tuples;
  237. TupleSet(Var x, Var y, ArrayList&lt;Tuple&gt; tuples){
  238. this.x = x;
  239. this.y = y;
  240. this.tuples = tuples;
  241. }
  242. public void setVars(Tuple t) {
  243. x.value = t.a;
  244. y.value = t.b;
  245. tuples.remove(t);
  246. }
  247. @Override
  248. public String toString() {
  249. return &quot;[&quot;+x+&quot;,&quot;+y+&quot;]&quot;+tuples;
  250. }
  251. }
  252. private static class Var{
  253. public Boolean value;
  254. @Override
  255. public String toString() {
  256. return value == null ? &quot;null&quot; : value.toString();
  257. }
  258. }
  259. private static class Tuple{
  260. public Boolean a;
  261. public Boolean b;
  262. public Tuple(Boolean a, Boolean b) {
  263. this.a = a;
  264. this.b = b;
  265. }
  266. @Override
  267. public String toString() {
  268. return &quot;(&quot;+a+&quot;,&quot;+b+&quot;)&quot;;
  269. }
  270. }
  271. }

An example of an impact would be running pp.permutate(5) with the following output:
With the stream: run the code as it was posted

  1. [true, true, true, true, true]
  2. [false, false, false, true, false]
  3. [false, false, false, false, true]
  4. [true, true, true, false, false]
  5. [true, false, false, true, false]
  6. [false, false, true, true, false]
  7. [true, true, true, false, false]

Without the stream: only delete the first line in generateOneSetOfBool()

  1. [true, true, true, true, true]
  2. [false, false, false, true, false]
  3. [false, false, false, false, true]
  4. [true, true, true, false, false]
  5. [false, true, true, true, false]
  6. [true, false, true, true, false]
  7. [true, true, false, false, false]

Both outputs differ, for example the last line

答案1

得分: 2

你得到的结果差异与这行代码 List collect = allVars.stream().filter(v -&gt; false).collect(Collectors.toList()); 无关。问题在于你的算法是非确定性的。我已经拿到了你的代码并为相同的输入多次运行它:

  1. for(int i = 0; i &lt; 20; i++) {
  2. pp.permutate(5);
  3. System.out.println(&quot;----------------&quot;);
  4. }

无论是否有你提到的那行代码,我都得到了你提到的两种输出(只出现这两种变化):

  1. [true, true, true, true, true]
  2. [false, false, false, true, false]
  3. [false, false, false, false, true]
  4. [true, true, true, false, false]
  5. [true, false, false, true, false]
  6. [false, false, true, true, false]
  7. [true, true, true, false, false]
  8. ----------------
  9. [true, true, true, true, true]
  10. [false, false, false, true, false]
  11. [false, false, false, false, true]
  12. [true, true, true, false, false]
  13. [false, true, true, true, false]
  14. [true, false, true, true, false]
  15. [true, true, false, false, false]

我没有逐行查看你的代码,所以我不确定,但我猜想你的一些集合不保证元素的顺序。

然而,有趣的是,当我多次运行 main 时,不同版本的输出出现的顺序总是相同的(或者至少在我尝试了5次中如此)。更有趣的是,当删除你提到的那行代码时,那个顺序会发生变化,但在调用 main 之间保持不变。考虑到在不同的机器上表现不同,我的结论可能是与程序在内存中的放置方式有关。

英文:

The difference in results you're getting has nothing to do with the line List collect = allVars.stream().filter(v -&gt; false).collect(Collectors.toList());. The problem is that your algorithm is non-deterministic. I've taken your code and run it multiple time for the same input:

  1. for(int i = 0; i &lt; 20; i++) {
  2. pp.permutate(5);
  3. System.out.println(&quot;----------------&quot;);
  4. }

Doesn't matter if there was the line you're talking about or not - I was getting both outputs you're mentioning (only those two variations appear tho):

  1. [true, true, true, true, true]
  2. [false, false, false, true, false]
  3. [false, false, false, false, true]
  4. [true, true, true, false, false]
  5. [true, false, false, true, false]
  6. [false, false, true, true, false]
  7. [true, true, true, false, false]
  8. ----------------
  9. [true, true, true, true, true]
  10. [false, false, false, true, false]
  11. [false, false, false, false, true]
  12. [true, true, true, false, false]
  13. [false, true, true, true, false]
  14. [true, false, true, true, false]
  15. [true, true, false, false, false]

I didn't go line by line through your code so I'm not sure but I'd guess that some of your collections doesn't guarantee anything about the order of elements.

However, what's interesting is that when I run main multiple time the order in which different versions of the output appear will always be the same (or at least in 5 times I tried). What is more, when that one line you mentioned is removed then that order changes - but stays the same between calls of main. When we add to that the fact that on different machine it behaves differently my conclusion would be that it might have something to with how the program is placed in memory.

huangapple
  • 本文由 发表于 2020年8月6日 21:49:10
  • 转载请务必保留本文链接:https://go.coder-hub.com/63285034.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定