函数式编程之Map&Reduce

一.内容概述

  • 在上一篇中介绍了什么是函数式编程, 以及介绍了最常用的一个函数式编程函数Filter
  • 详情请查看 Swift中的函数式编程
  • 这篇文章中介绍函数式编程中另外两个非常常用和重要的函数: Map&Reduce
  • 理解了这两个东西基本对函数式编程就会有一定的理解和认识.

二. Map的使用

1. Map的介绍

  • Map在此处并非地图的意思, 它的含义是映射
  • 将一个元素映射成另外一种元素(类似于字典中的Key/Value映射)
  • 其实Swift系统本身是有映射的函数, 可以将一个集合映射成另外一个集合

2. 示例分析

  • 比如:
    • 将一个String类型的数组, 映射成一个Int类型的数组
    • 将一个String类型的数组, 映射成一个Emoji数组
  • 如何使用呢?
// 1.定义字符串数组
let animals = ["DRAGON", "CAT", "TIGER"]

// 2.使用函数映射
let animalEmojis = animals.map { (str : String) -> Character in
    let tmp = "\\N{\(str)}"
    let emojiStr =
    tmp.stringByApplyingTransform(
        NSStringTransformToUnicodeName,
        reverse: true)

    return emojiStr!.characters.first!
}

// 3.打印结果
print(animalEmojis)
  • 结果如下:
  • 分析map后面的大括号{}
    • 其实{}本身也可以是一个函数
    • 例如我单独将函数抽取出来,并且传入.
    • 代码如下:
// 0.定义字符串数组
let animals = ["DRAGON", "CAT", "TIGER"]

// 1.抽取函数
func strToEmoji(str : String) -> Character {
    let tmp = "\\N{\(str)}"
    let emojiStr =
    tmp.stringByApplyingTransform(
        NSStringTransformToUnicodeName,
        reverse: true)

    return emojiStr!.characters.first!
}

// 2.将函数传入
let animalEmojis = animals.map(strToEmoji)

// 3.打印结果
print(animalEmojis)
  • 代码分析:
    • 通过上述代码我们知道, 其实map函数是接收一个函数作为参数.
    • 而函数本身是告诉map函数, 在映射过程中究竟是进行怎样的转化

3.定义自己的Map函数

  • 我们可以给Array扩展一个函数, 函数接收一个闭包参数
// 1.定义自己的map函数
extension Array {
    func myMap<SrcType, DestType>(transform: (SrcType) -> (DestType)) -> [DestType] {
            var output: [DestType] = []

            for item in self {
                let transformed =
                transform(item as! SrcType)
                output.append(transformed)
            }

            return output
    }
}

// 2.使用自己的map函数
let animals = ["DRAGON", "CAT", "TIGER"]

let animalEmojis = animals.myMap { (str : String) -> (Character) in
    let tmp = "\\N{\(str)}"
    let emojiStr =
    tmp.stringByApplyingTransform(
        NSStringTransformToUnicodeName,
        reverse: true)

    return emojiStr!.characters.first!
}

print(animalEmojis)
  • 代码分析
    • Map有两个泛型类型, 一个是原始类型, 一个是目标类型
    • 在自己的Map函数中, 使用外界传入的函数, 进行转化, 并且生成新的类型
    • 放入数组中, 进行返回即可

二. Reduce的使用

1.Reduce的介绍

  • Reduce也是函数式编程中很常见的一个函数
  • 它的作用是用于将集合中的内容合并起来, 获取到合并后的结果
  • 合并的规则可以由调用者来指定, 可以是:相加/相乘/字符串拼接.
  • 总之, 你告诉函数如何合并, 函数会给你一个合并后的结果

2.Reduce的使用

  • 我们来看一个Reduce的示例
// 1.定义数组
let numbers = [1, 2, 3, 4, 5]

// 2.合并数组中所有的元素
// 2.1.普通做法
var total = 0
for num in numbers {
    total += num
}

// 2.2.函数式编程中Reduce
var total1 = numbers.reduce(0) { (num1 : Int, num2 : Int) -> Int in
    return num1 + num2
}
var total2 = numbers.reduce(0) { $0 + $1 }
  • 代码解析
    • 有一个Int类型数组, 想要将数组中元素合并
    • 普通做法是直接遍历, 但是如果我们不是相加, 有其他合并需求, 则需要更改for循环中的需求
    • 但是通过我们系统提供的reduce, 让我们直接传入闭包. 并且告知函数规则, 就可以进行合并
    • reduce函数接收两个参数:
      • 参数一: 第一次合并之前的初始化值
      • 参数二: 闭包参数, 闭包中的参数一是每次合并后的值, 第二个参数是后面要合并的元素
    • { $0 + $1 } --> 闭包的一种简便写法

3. 定义自己的Reduce函数

  • 我们还是给系统的Array扩充一个自己的Reduce方法
// 1.定义自己的Reduce函数
extension Array {
    func myReduce<C, T>(initial: C, combine: (C, T) -> C) -> C {
            var seed = initial

            for item in self {
                seed = combine(seed, item as! T)
            }

            return seed
    }
}


// 2.定义数组
let numbers = [1, 2, 3, 4, 5]

// 3.使用自己的Reduce函数
let result = numbers.myReduce(1) { (num1 : Int, num2 : Int) -> Int in
    return num1 * num2
}

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

推荐阅读更多精彩内容