如何使用 RxJS 从嵌套的类型化对象数组中查找最大值和最小值?

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

how to find max min from nested typed object array using RxJS?

问题

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

  1. type DataTableType = {
  2. name: string;
  3. series: {
  4. name: string;
  5. value: number;
  6. }[];
  7. }[];
  8. let datatable: DataTableType = [
  9. {
  10. name: 'Row 1',
  11. series: [
  12. {
  13. name: 'Series 1',
  14. value: 7110
  15. },
  16. {
  17. name: 'Series 2',
  18. value: 8240
  19. },
  20. {
  21. name: 'Series 3',
  22. value: 2111
  23. },
  24. {
  25. name: 'Series 4',
  26. value: 3120
  27. }
  28. ]
  29. },
  30. // ... (其他数据行)
  31. ];
  32. function findMax(column: DataTableType) {
  33. let maxN = column[0].series[0].value;
  34. of(...column)
  35. .pipe(
  36. mergeMap(
  37. (item: DataTableType[number], i: number) => from(
  38. [...item.series]
  39. )
  40. .pipe(
  41. max((a, b) => a.value < b.value ? b.value : a.value)
  42. )
  43. )
  44. )
  45. .subscribe(
  46. (category: { value: number; }) => {
  47. maxN = category.value;
  48. }
  49. )
  50. ;
  51. return maxN;
  52. }
  53. function findMin(column: DataTableType) {
  54. let minN = column[0].series[0].value;
  55. of(...column)
  56. .pipe(
  57. mergeMap(
  58. (item: DataTableType[number], i: number) => from(
  59. [...item.series]
  60. )
  61. .pipe(
  62. min((a, b) => a.value > b.value ? b.value : a.value)
  63. )
  64. )
  65. )
  66. .subscribe(
  67. (category: { value: number; }) => {
  68. minN = category.value;
  69. }
  70. )
  71. ;
  72. return minN;
  73. }

请注意,这些函数可能不会立即返回最大和最小值,因为它们使用RxJS进行异步操作。如果您在这些函数之后执行其他操作,应该等待它们的异步操作完成,或者可以将操作移到subscribe回调函数内部以确保在获取最大和最小值之后再进行操作。

英文:

given I have this

  1. type DataTableType = {
  2. name: string;
  3. series: {
  4. name: string;
  5. value: number;
  6. }[];
  7. }[];
  8. let datatable: DataTableType = [
  9. {
  10. name: &#39;Row 1&#39;,
  11. series: [
  12. {
  13. name:&#39;Series 1&#39;,
  14. value:7110
  15. },
  16. {
  17. name:&#39;Series 2&#39;,
  18. value:8240
  19. },
  20. {
  21. name:&#39;Series 3&#39;,
  22. value:2111
  23. },
  24. {
  25. name:&#39;Series 4&#39;,
  26. value:3120
  27. }
  28. ]
  29. },
  30. {
  31. name: &#39;Row 2&#39;,
  32. series: [
  33. {
  34. name:&#39;Series 1&#39;,
  35. value:6500
  36. },
  37. {
  38. name:&#39;Series 2&#39;,
  39. value:9120
  40. },
  41. {
  42. name:&#39;Series 3&#39;,
  43. value:5123
  44. },
  45. {
  46. name:&#39;Series 4&#39;,
  47. value:8200
  48. }
  49. ]
  50. },
  51. {
  52. name: &#39;Row 3&#39;,
  53. series: [
  54. {
  55. name:&#39;Series 1&#39;,
  56. value:-6453
  57. },
  58. {
  59. name:&#39;Series 2&#39;,
  60. value:-9000
  61. },
  62. {
  63. name:&#39;Series 3&#39;,
  64. value:5009
  65. },
  66. {
  67. name:&#39;Series 4&#39;,
  68. value:8900
  69. }
  70. ]
  71. },
  72. {
  73. name: &#39;Row 4&#39;,
  74. series: [
  75. {
  76. name:&#39;Series 1&#39;,
  77. value:8567
  78. },
  79. {
  80. name:&#39;Series 2&#39;,
  81. value:4332
  82. },
  83. {
  84. name:&#39;Series 3&#39;,
  85. value:-3111
  86. },
  87. {
  88. name:&#39;Series 4&#39;,
  89. value:-3222
  90. }
  91. ]
  92. },
  93. {
  94. name: &#39;Row 5&#39;,
  95. series: [
  96. {
  97. name:&#39;Series 1&#39;,
  98. value:8333
  99. },
  100. {
  101. name:&#39;Series 2&#39;,
  102. value:3234
  103. },
  104. {
  105. name:&#39;Series 3&#39;,
  106. value:-7323
  107. },
  108. {
  109. name:&#39;Series 4&#39;,
  110. value:1544
  111. }
  112. ]
  113. },
  114. {
  115. name: &#39;Row 6&#39;,
  116. series: [
  117. {
  118. name:&#39;Series 1&#39;,
  119. value:2112
  120. },
  121. {
  122. name:&#39;Series 2&#39;,
  123. value:3232
  124. },
  125. {
  126. name:&#39;Series 3&#39;,
  127. value:-8231
  128. },
  129. {
  130. name:&#39;Series 4&#39;,
  131. value:1111
  132. }
  133. ]
  134. },
  135. {
  136. name: &#39;Row 7&#39;,
  137. series: [
  138. {
  139. name:&#39;Series 1&#39;,
  140. value:-2112
  141. },
  142. {
  143. name:&#39;Series 2&#39;,
  144. value:3232
  145. },
  146. {
  147. name:&#39;Series 3&#39;,
  148. value:8231
  149. },
  150. {
  151. name:&#39;Series 4&#39;,
  152. value:-9111
  153. }
  154. ]
  155. },
  156. {
  157. name: &#39;Row 8&#39;,
  158. series: [
  159. {
  160. name:&#39;Series 1&#39;,
  161. value:-2112
  162. },
  163. {
  164. name:&#39;Series 2&#39;,
  165. value:100
  166. },
  167. {
  168. name:&#39;Series 3&#39;,
  169. value:-3
  170. },
  171. {
  172. name:&#39;Series 4&#39;,
  173. value:0
  174. }
  175. ]
  176. }
  177. ]

I want to find the Max and Min from the datatable with using RxJS, I have done this below to find the Max and Min

  1. function findMax(column: DataTableType) {
  2. let maxN = column[0].series[0].value;
  3. of(...column)
  4. .pipe(
  5. mergeMap(
  6. (item: DataTableType[number], i: number) =&gt; from(
  7. [...item.series]
  8. )
  9. .pipe(
  10. max((a, b) =&gt; a.value &lt; b.value? b.value : a.value)
  11. )
  12. )
  13. )
  14. .subscribe(
  15. (category: { value: number; }) =&gt; {
  16. maxN = category.value;
  17. }
  18. )
  19. ;
  20. return maxN;
  21. }
  22. function findMin(column: DataTableType) {
  23. let minN = column[0].series[0].value;
  24. of(...column)
  25. .pipe(
  26. mergeMap(
  27. (item: DataTableType[number], i: number) =&gt; from(
  28. [...item.series]
  29. )
  30. .pipe(
  31. min((a, b) =&gt; a.value &gt; b.value? b.value : a.value)
  32. )
  33. )
  34. )
  35. .subscribe(
  36. (category: { value: number; }) =&gt; {
  37. minN = category.value;
  38. }
  39. )
  40. ;
  41. return minN;
  42. }

But whenever I run the functions findMax(datatable) and findMin(datatable) I get max = 0 and min = -2112 of the last element of the object array which is Row 8.

My expected answer should be max = 9120 and min = -9111

Please I need help.

答案1

得分: 1

以下是您要翻译的内容:

The functions given to min and max should behave like the comparator function passed to Array.sort():
>The return value should be a number whose sign indicates the relative order of the two elements: negative if a is less than b, positive if a is greater than b, and zero if they are equal.

Unfortunately, the documentation of RxJS provides this only through examples, i.e. for min() it gives the example:
min((a, b) =&gt; a.age &lt; b.age ? -1 : 1)
Note that min and max work with the same comparison, you don't have to adjust it depending on whether you want min or max. So in your case, it is (a,b) =&gt; a.value - b.value for both.

A second problem in your functions is that you apply the min/max to the steam you generate inside the mergeMap(), so it will be evaluated for each group, instead of the flattened stream. So you will get a result for every Row item, and your variable will be set to each of them, until it is returned with the value from the last iteration.

Instead, apply min/max after the merge, so that you get the smallest/largest value from the whole stream.

Together with the fixed comparator, this gives you:

  1. let maxN = column[0].series[0].value;
  2. from(column)
  3. .pipe(mergeMap(item =&gt; item.series))
  4. .pipe(max((a, b) =&gt; a.value - b.value))
  5. .subscribe(
  6. (category) =&gt; {
  7. maxN = category.value;
  8. }
  9. )
  10. ;
  11. return maxN;
  12. }```
  13. Alternatively, you can also have the `mergeMap` extract the values directly, then you don&#39;t have to use a comparator:
  14. ```function findMax(column) {
  15. let maxN = column[0].series[0].value;
  16. from(column)
  17. .pipe(mergeMap(item =&gt; from(item.series).pipe(map(s =&gt; s.value))))
  18. .pipe(max())
  19. .subscribe(
  20. (category) =&gt; {
  21. maxN = category;
  22. }
  23. )
  24. ;
  25. return maxN;
  26. }```
  27. <details>
  28. <summary>英文:</summary>
  29. The functions given to min and max should behave like the comparator function passed to [Array.sort()](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#syntax):
  30. &gt;The return value should be a number whose sign indicates the relative order of the two elements: negative if `a` is less than `b`, positive if `a` is greater than `b`, and zero if they are equal.
  31. Unfortunately, the documentation of RxJS provides this only through examples, i.e. for [`min()`](https://rxjs.dev/api/index/function/min) it gives the example:

min((a, b) => a.age < b.age ? -1 : 1)

  1. Note that `min` and `max` work with the same comparison, you don&#39;t have to adjust it depending on whether you want min or max. So in your case, it is `(a,b) =&gt; a.value - b.value` for both.
  2. ----------
  3. A second problem in your functions is that you apply the `min`/`max` to the steam you generate inside the `mergeMap()`, so it will be evaluated for each group, instead of the flattened stream. So you will get a result for every `Row` item, and your variable will be set to each of them, until it is returned with the value from the last iteration.
  4. Instead, apply `min`/`max` after the merge, so that you get the smallest/largest value from the whole stream.
  5. Together with the fixed comparator, this gives you:

function findMax(column) {
let maxN = column[0].series[0].value;
from(column)
.pipe(mergeMap(item => item.series))
.pipe(max((a, b) => a.value - b.value))
.subscribe(
(category) => {
maxN = category.value;
}
)
;
return maxN;
}

  1. ----------
  2. Alternatively, you can also have the `mergeMap` extract the values directly, then you don&#39;t have to use a comparator:

function findMax(column) {
let maxN = column[0].series[0].value;
from(column)
.pipe(mergeMap(item => from(item.series).pipe(map(s => s.value))))
.pipe(max())
.subscribe(
(category) => {
maxN = category;
}
)
;
return maxN;
}

  1. </details>

huangapple
  • 本文由 发表于 2023年7月3日 22:05:51
  • 转载请务必保留本文链接:https://go.coder-hub.com/76605535.html
匿名

发表评论

匿名网友

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

确定