Why Coding like This?—— Reduce 函数揭秘

3.Reduce 函数揭秘

Topic 3:

请用Reduce函数对Int类型数组内所有元素求和,例如数组[1,2,3,4]的4个元素和为10.

Example:

//例一:
let intArray = [1,2,3,4]

var sum = intArray.reduce(0){
  result, x in
  result + x
}

why coding like this?

1.开篇

命题一:假设让你写一个函数来实现对Int类型数组内所有元素求和,例如数组[1,2,3,4]的求和结果为10。
思路:与类map,filter函数思路不同,前者是对数组内每一个元素单独做处理,而reduce恰恰相反,是将所有数组元素整合到一起。因此关于求和函数思路很明确,遍历整个数组元素,逐一求和后作为结果值返回,大致为 0 + 1 + 2 + 3 +4。其中0是初始值。
代码:

//例二:
func sum(xs:[Int])->Int{
  var result:Int = 0
  for x in xs{
    result += x
  }
  return result
}
//不妨测试下
sum(intArray)//print 10

可以看到对于sum函数传入Int类型数组,首先我们声明一个Int类型变量用于保存最后的求和结果值,初始值为0;接着使用for-in语句遍历传入的数组,使用result += x逐一求和累加;最后返回结果值result。

2.过渡

命题二:基础入门之后,按照惯例对条件进行改变,假如把所有元素求和更改为对所有元素求乘,如[1,2,3,4] 返回1 * 2 * 3 * 4 = 24。
思路:显然对于前面的代码稍许改动即可,主要是替换result += x(累加运算) 更为result = result * x(阶乘运算)。当然有一点切记不要遗忘,就是初始result = 1
代码:

//例三:
func multiplicator(xs:[Int])->Int{
    var result:Int = 1
    for x in xs{
        result = x * result
    }
    return result
}
multiplicator(intArray)//输出24

命题三: 说完Int类型数组,再来说说String类型,将一个文字片段数组组成一个完整的句子,例如["hello"," ","world","!","say"," ","by"," ","optionalswift"],最后整合成"hello world!say by optionalswift"
思路: 思路大概是遍历数组,然后将所有字符串元素拼接在一起,关键这个拼接用什么?例如有str1str2,swift中其实只需要简单的newStr = str1 + str2 即可。
代码:

//例四:
func jointStringArray(xs:[String])->String{
  var result :String = ""
  
  for x in xs{
    result += x
  }
  return result
}
let helloworld = ["hello"," ","world","!","say"," ","by"," ","optionalswift"]
jointStringArray(helloworld)//输出"hello world!say by optionalswift"  

你已经急着上手想写泛型函数了吗?等等,我们貌似还忽略了一些复杂的链接情况。比如["hello","world","say","By","optional"],我们需要在每一个单词之间插入"-"连字符,别问我为什么?因为是我出题!

//例五:hyphen 意思为连字符`-`
func jointStringArrayByHyphen(xs:[String])->String{
  var result : String = "整合后的字符串为:"
  for x in xs{
    result = result + "-" + x  //注意右侧是一个类似combine(result,x)函数进行处理 得到整合后的结果给result
  }
  return result
}
let newArray = ["hello","world","say","By","optional"]
jointStringArrayByHyphen(newArray)//输出"整合后的字符串为:-hello-world-say-By-optional"

3.高潮

在开始写我们的泛型reduce函数之前,分析函数的输入输出以及如何实现:首先该函数将传入一个泛型类数组,暂且定为[T];此外函数可自己设定初始值给result,暂且定为U;最后是重中之重:传入一个闭包作为连接result和数组元素的处理函数,combine(result,x),不难分析result的类型为T,x是数组xs中的元素,类型为U,返回嘛自然是和result一致的类型喽。因此combine闭包类型为(U,T)->U。因此代码这么写:

//例六:
func myReduce<T,U>(arr:[T],initialValue:U,combine:(U,T)->U)->U{
  var result = initialValue //赋值初始值 类型为U 并且是作为结果值返回的
  for elem in arr{
    result = combine(result,elem)   //注意右侧是传入的闭包 该闭包类型为(U,T)->U,即把上一次的结果值依次和数组元素做拼接处理,该处理可在闭包中实现,取决于你
  }
  return result
}
myReduce(newArray, "整合后的字符串为:"){
  result , elem in //注意result ,x 于combine(result,elem)相对应
  return result + "-" + x //注意这里return 其实是可以省略的!
}

不得不说,这个函数还是有点料的,需要细细品味一番。你以为这就结束了吗,还有落幕呢?

4.落幕

对sum函数进行改写,使用前面自定义的myReduce函数封装

func sumUsingReduce(xs:[Int])->Int{
    return myReduce(xs,0){result, x in result + x}
}

注意到省略了return 因为swift会帮你推算要返回什么。简化的感觉不够彻底。

func sumUsingReduce(xs:[Int])->Int{
    return myReduce(xs,0,+)
}

闭包仅仅传入了一个+号,swift推算过程是首先combine闭包有两个传入参数result和elem,除此之外别无其他,因此+只能对这两个参数求和,得到一个结果值x,由于combine函数还需要返回一个结果值,但是思来想去貌似除了x没有其他可用,因此把x作为闭包结果值返回和result相加。

类似的你可是使用return myReduce(xs,1,*)。

你以为这就结束了吗? 现在用reduce来改写map函数 以及filter函数

func mapUsingReduce<T,U>(xs:[T],f:T->U)->[U]{
    return xs.reduce([]){result,x in result + [f(x)]}//使用了系统API 尝试用自定义的
}

首先注意到函数传入的两个参数以及返回结果值和早前map函数是一模一样的,关键是在主体的实现上!

再来看filter的实现:

filterUsingReduce<T>(xs:[T],check:T->Bool)->[T]{
    return xs.reduce([]){
        return check(x) ? result + [x] : result //使用了系统API 尝试用自定义的
    }
}

map,filter,reduce 小节至此结束! 希望对大家有收获!

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

推荐阅读更多精彩内容