對於一般需求,uniq()
即可勝任,但若需自行提供特殊比較方式,則要使用 uniqWith()
,自行傳入 Callback。
Version
macOS Mojave 10.14.5
VS Code 1.33.1
Quokka 1.0.212
Ramda 0.26.1
uniqBy()
import { uniqBy } from 'ramda';
let data = [1, 2, 3, -1];
uniqBy(x => Math.abs(x))(data); // ?
data
中包含 1
與 -1
,若1
與 -1
均視為相同而不想重複顯示,也就是我們希望結果只顯示 [1, 2, 3]
。
uniqBy()
(a -> b) -> [a] -> [a]
將 array 經過 supply function 轉換,回傳 element 不重複 array
uniqWith()
import { uniqWith, eqBy } from 'ramda';
let data = [1, 2, 3, -1];
let pred = (x, y) => Math.abs(x) === Math.abs(y);
uniqWith(pred)(data); // ?
我們也可以使用 uniqWith()
完成。
uniqWith()
((a, a) → Boolean) → [a] → [a]
自行提供 predicate 決定比較方式,回傳 element 不重複 array
((a, a) → Boolean)
:predicate function,自行決定比較方式
[a]
:data 為 array
[a]
:回傳 element 不重複 array
由 Ramda source code 得知,
uniqWith()
其實是兩層for
loop,第一層為第一個 argument,第二層為第二個 argument,也就是第一個 argument 代表 array 中每個 element,第二個 argument 代表所比較 array 中每個 element
Point-free
import { uniqWith, eqBy } from 'ramda';
let data = [1, 2, 3, -1];
let pred = eqBy(Math.abs);
uniqWith(pred)(data); // ?
(x, y) => Math.abs(x) === y
可進一步由 eqBy(Math.abs)
化簡成 point-free。
eqBy()
(a → b) → a → a → Boolean
若兩 value 經過 function 轉換後結果相同則回傳true
,否則false
Object
import { uniqWith } from 'ramda';
let data = [
{ title: 'FP in JavaScript', price: 100 },
{ title: 'RxJS in Action', price: 200 },
{ title: 'Speaking JavaScript', price: 300 },
{ title: 'FP in JavaScript', price: 400 },
{ title: 'Exploring ReasonML', price: 200 },
];
let pred = (x, y) => x.title === y.title || x.price === y.price;
console.dir(uniqWith(pred)(data));
若 element 為 object,只要 title
或 price
相等就視為相同,也就是第四筆與第一筆相同,第五筆與第二筆相同,我們希望結果只顯示不重複的前三筆。
此時就不能再使用 uniqBy()
,因為條件太特殊,只能使用 uniqWith()
。
Point-free
import { uniqWith, anyPass, eqProps } from 'ramda';
let data = [
{ title: 'FP in JavaScript', price: 100 },
{ title: 'RxJS in Action', price: 200 },
{ title: 'Speaking JavaScript', price: 300 },
{ title: 'FP in JavaScript', price: 400 },
{ title: 'Exploring ReasonML', price: 200 },
];
let pred = anyPass([eqProps('title'), eqProps('price')]);
console.dir(uniqWith(pred)(data));
(x, y) => x.title === y.title || x.price === y.price
也可由 anyPass()
與 eqProps()
產生,使其 point-free。
anyPass()
[(*… → Boolean)] → (*… → Boolean)
將眾多 predicate 組合成單一 predicate,只要任何 predicate 成立即可
Conclusion
- Unique series 三兄弟:
uniq()
、uniqBy()
與uniqWith()
,其中uniqWith()
算客製化程度最高的 function,可自行決定比較方式 - Ramda 若很難從官網說明明白其意義,直接從其 source code 了解也是不錯方式,
uniqWith()
就是一例
Reference
Ramda, uniqBy()
Ramda, uniqWith()
Ramda, eqBy()
Ramda, anyPass()
Ramda, eqProps()