第4章 集合操作工具库

1.非 FP(函数编程) 集合处理方法

forEach(), some(), every()处理集合的方法,看起来像函数编程中的方法,实际上却不是的;

forEach() 迭代调用回调函数会产生副作用。some(),every()不可撤销的将集合变为一个 true/false 结果,所以下面将跳过这3个方法。

2.Map

手动实现map()

function map(arr, mapperFn) {
    var newList = [];
    
    for (let idx = 0; idx < arr.length; idx++) {
        newList.push(
            // 为了和内置的map()函数签名一致,传入3个参数
            mapperFn(arr[idx], idx, arr)
        );
    }

    return newList;
}
// 例子
map(["1", "2", "3"], unary(parseInt)); // [1,2,3]
// unary()是前面章节中的一个函数
var unary = 
        fn =>
            arg =>
                fn(arg);

原生map(),我们可以先将functions集合和另一个函数组合,然后再执行:

var increment = v => ++v;
var decrement = v => --v;
var square = v => v * v;

var double = v => v * 2;

[increment, decrement, square]
    .map(fn => compose(fn, double))
    .map(fn => fn(3)); // [7, 5, 36]

3.Filter

filter这个单词在JS中有2层意思:

  1. 过滤不想要的,留下想要的(filtering out)
  2. 过滤想要的,留下不想要的(filtering in)

filter()内的回调函数返回布尔值,我们把这样函数称之为 predicate functions.

手动实现:

function filter(arr, predicateFn) {
    var newList = [];
    
    for (let idx = 0; idx < arr.length; idx++) {
        if (predicateFn(arr[idx], idx, arr)) {
            newList.push(arr[idx]);
        }
    }
    
    return newList;
}

4.Reduce

Reduce是函数编程中最重要的工具,像多合一瑞士军刀一样。

它有2种形式,一种提供initialValue,一种不提供。

fig11.png
fig12.png

1.单独实现Reduce()

function reduce(arr, reducerFn, initialValue) {
    var acc, startIdx;

    // 提供initialValue的情况
    if (arguments.length === 3) {
        acc = initialValue;
        startIdx = 0;
    } else if (arr.length > 0) { // 不提供initialValue
        arr = arr[0];
        startIdx = 1;
    } else {
        throw new Error("Must provide at least one value");
    }

    for (let idx = startIdx; idx < arr.length; idx++) {
        // 注意这里是4个参数
        acc = reducerFn(acc, arr[idx], idx, arr);
    }
    
    return acc;
}

2.使用 reduce 模拟 map 的功能

这里面的技巧是,reduce的初始值可以为一个空的数组[].

var double = v => v * 2;

[1, 2, 3].map(double); // [2, 4, 6]

// list的初始值为[]
[1, 2, 3].reduce(
    (list, v) => (
        list.push(double(v)), // 注意
        list
    ), []
);
// [2, 4, 6]

上面标记注意的地方,其写法为:

(list, v) => (
    list.push(double(v));
    return list;
)

3.使用 reduce 模拟 filter 的功能

其原理和模拟map类似

var isOdd = v => v % 2 === 1;

[1, 2, 3, 4, 5].filter(isOdd); // [1, 3, 5]

[1, 2, 3, 4, 5].reduce(
    (list, v) => (
        isOdd(v) ? list.push(v) : undefined,
        list
    ), []
); // [1, 3, 5]

5.高级集合操作

下面介绍许多库中常用的一些函数, unique(), flatten(), zip(), merge()...

1.去重 unique()

去除数组中重复的items有几种实现方法。

1.使用 filter + indexOf

var unique = 
    arr =>
        arr.filter(
            (v, idx) =>
                arr.indexOf(v) === idx 
                // 利用值的位置和索引位置对比
        );

2.瑞士军刀 reduce + indexOf

var unique =
    arr =>
        arr.reduce(
            (list, v) =>
                list.indexOf(v) === -1 ?
                    (list.push(v), list) : list
        , []);

3.ES6新集合类型 Set

var unique =
    arr =>
        [...new Set(arr)];

unique([1, 2, 3, 3, 4, 4, 1])
// [1, 2, 3, 4]

2.将嵌套的数组打平 Flatten

比如将:

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

// 转换为
[1, 2, 3, 4, 5, 6, 7]

// 或者
[1, 2, 3, 4, 5, [6, 7]]

1.使用 reduce + 递归 直接全部打平

var flatten =
    arr =>
        arr.reduce(
            (list, v) => (
                // 判断内部item是否为数组
                // 如果是则递归调用flatten
                // 不是则直接添加到list数组中
                list.concat(Array.isArray(v) ? flatten(v) : v)
            ),
            []
        );

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]] );
// [0,1,2,3,4,5,6,7,8,9,10,11,12,13]

2.对打平的层次添加一个 depth

 // 默认打开层次为无限, 即直接全部打开
 var flatten =
    (arr, depth = Infinity) =>
        arr.reduce(
            (list, v) =>
                list.concat(
                    depth > 0 ?
                        (depth > 1 && Array.isArray(v) ?
                            flatten(v, depth - 1) :
                            v
                        ) :
                        [v]
                )
        , []);

实例:

// 0 表示不打开
flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 0 );
// [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]]

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 1 );
// [0,1,2,3,4,[5,6,7],[8,[9,[10,[11,12],13]]]]

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 2 );
// [0,1,2,3,4,5,6,7,8,[9,[10,[11,12],13]]]

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 3 );
// [0,1,2,3,4,5,6,7,8,9,[10,[11,12],13]]

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 4 );
// [0,1,2,3,4,5,6,7,8,9,10,[11,12],13]

flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]], 5 );
// [0,1,2,3,4,5,6,7,8,9,10,11,12,13]

// 不添加depth, 默认为都打开
flatten( [[0,1],2,3,[4,[5,6,7],[8,[9,[10,[11,12],13]]]]] );
// [0,1,2,3,4,5,6,7,8,9,10,11,12,13] 

3.flatMap() 通常使用方法

通常flatten结合map使用对实际数据进行操作:

var firstNames = [
    { name: "Jonathan", variations: [ "John", "Jon", "Jonny" ] },
    { name: "Stephanie", variations: [ "Steph", "Stephy" ] },
    { name: "Frederick", variations: [ "Fred", "Freddy" ] }
];
firstNames
    .map(entry => [entry.name].concat(entry.variations))
// [ ["Jonathan","John","Jon","Jonny"], ["Stephanie","Steph","Stephy"],
//   ["Frederick","Fred","Freddy"] ]

// 然后利用flatten得到所有的names

flatten(
firstNames
    .map( entry => [entry.name].concat( entry.variations ) )
);
// ["Jonathan","John","Jon","Jonny","Stephanie","Steph","Stephy","Frederick",
//  "Fred","Freddy"]

flatMap()实现:

var flatMap =
    (arr, mapperFn) =>
        arr.reduce(
            (list, v) =>
                list.concat(mapperFn(v))
        , []);

3.交替取出2个数组中的元素 zip

有时候我们需要交替取出2个数组中的元素组成一个新的item, 然后返回一个新的数组。

比如:

zip([1, 2, 3, 4], [5, 6, 7, 8, 9]);
// [ [1,5], [2,6], [3,7], [4,8] ]

上面实例可以看出,以长度小的那个数组长度作为返回数组长度

function zip(list1, list2) {
    var zipped = [];
    list1 = list1.slice();
    list2 = list2.slice();
    
    while(list1.length > 0 && list2.length > 0) {
        zipped.push([list1.shift(), list2.shift()]);
    }
    
    return zipped;
}

4.交替合并2个数组merge

交替添加到一个新的数组中

mergeLists( [1,3,5,7,9], [2,4,6,8,10] );
// [1,2,3,4,5,6,7,8,9,10]

可以看出上面的实例可以理解为 zip() -> flatten(), 但是这有个缺点就是必须以长度短的为基准,下面实现:先交替添加,剩下的直接添加到数组里面。

function merge(list1, list2) {
    var merged = [];
    list1 = list1.slice();
    list2 = list2.slice();
    
    while (list1.length > 0 || list2.length > 0) {
        if (list1.length > 0) {
            merged.push( list1.shift() );
        }
        if (list2.length > 0) {
            merged.push( list2.shift() );
        }
    }
    return merged;
}

4.融合 Fusion

像这样的操作

..
.filter(..) 
.filter(..)
.map(..)
.map(..)
.map(..)
.reduce(..);

这种声明式的好处就是思路十分清晰,确定就是每次操作,都要对数组遍历一遍,这要势必会影响性能

融合可以处理相连的操作符,减少遍历的次数,下面对相连的map()进行融合

var removeInvalidChars = str => str.replace(/[^\w]*/g, "");
var upper = text => text.toUpperCase();
var elide = str =>
        str.length > 10?
            str.slice(0, 7) + "..." :
            str;
words;
// ["Mr.","Jones","isn't","responsible","for","this","disaster!"]

words
    .map( removeInvalidChars )
    .map( upper )
    .map( elide );
// ["MR","JONES","ISNT","RESPONS...","FOR","THIS","DISASTER"]

由于上面的map都是一元的,可以想到使用前面的学过的 compose 或者 pipe

words.map(
    compose(elide, upper, removeInvalidChars)
);
// 或者
words.map(
    pipe(removeInvaildChars, upper, elide)
);

总结

通过本章学习了:

  • map, filter, reduce的手工实现
  • unique() 函数
  • flatten() 函数
  • zip()
  • merge()
  • 以及融合对相连的map进行合并
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,527评论 5 470
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,314评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,535评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,006评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,961评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,220评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,664评论 3 392
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,351评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,481评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,397评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,443评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,123评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,713评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,801评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,010评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,494评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,075评论 2 341

推荐阅读更多精彩内容