原文标题:Converting Plaid to Kotlin: Lessons learned (Part 2)

原文链接:http://antonioleiva.com/plaid-kotlin-2/

原文作者:Antonio Leiva(http://antonioleiva.com/about/

原文发布:2015-11-17

我们在第一部分中所见的各种显著地改进,要归功于在Activity中使用了Kotlin语言。但是,由于主要是重载方法做些事情,仍然免不了一些公式化代码,所以这种类型的类还不能很好的展示其效果。

我持续用Kotlin语言改写该APP(大家可以在repo看到所有改变),并遇到一些有趣的事情。今天,我着重谈论DataManager类的转换。透露一下,该类的大小已从422行减少到177行。我认为这很容易理解。

When

纵观该类时,首先看到最多的就是在loadSource类中有大量的if/else语句。在第一个实例中,可以用switch语句提升可读性,但是总还是有点不易理解。在Kotlin语言中,可以用when expression(表达式),它非常类似Java语言中的switch语句,但是功能更强。条件可以依据需要编写,且可以很好的替代if/else语句。这里使用其最简单版本,也已经可以使这个方法更容易阅读:

 when (source.key) {
SourceManager.SOURCE_DESIGNER_NEWS_POPULAR -> loadDesignerNewsTopStories(page)
SourceManager.SOURCE_DESIGNER_NEWS_RECENT -> loadDesignerNewsRecent(page)
SourceManager.SOURCE_DRIBBBLE_POPULAR -> loadDribbblePopular(page)
SourceManager.SOURCE_DRIBBBLE_FOLLOWING -> loadDribbbleFollowing(page)
SourceManager.SOURCE_DRIBBBLE_USER_LIKES -> loadDribbbleUserLikes(page)
SourceManager.SOURCE_DRIBBBLE_USER_SHOTS -> loadDribbbleUserShots(page)
SourceManager.SOURCE_DRIBBBLE_RECENT -> loadDribbbleRecent(page)
SourceManager.SOURCE_DRIBBBLE_DEBUTS -> loadDribbbleDebuts(page)
SourceManager.SOURCE_DRIBBBLE_ANIMATED -> loadDribbbleAnimated(page)
SourceManager.SOURCE_PRODUCT_HUNT -> loadProductHunt(page)
else -> when (source) {
is Source.DribbbleSearchSource -> loadDribbbleSearch(source, page)
is Source.DesignerNewsSearchSource -> loadDesignerNewsSearch(source, page) }
}

然而,这不是case,是when表达式,所以可以返回一个值,这非常有用。

映射处理

虽然,这个例子并没有减少多少代码(行),但是可以有趣的看到,在Kotlin中怎样处理映射表。在Java语言中,getNextPageIndex如下:

 private int getNextPageIndex(String dataSource) {
int nextPage = 1; // default to one – i.e. for newly added sources
if (pageIndexes.containsKey(dataSource)) {
nextPage = pageIndexes.get(dataSource) + 1;
}
pageIndexes.put(dataSource, nextPage);
return nextPage;
}

那在Kotlin语言中是怎样的?

 private fun getNextPageIndex(dataSource: String): Int {
val nextPage = 1 + pageIndexes.getOrElse(dataSource) { 0 }
pageIndexes += dataSource to nextPage
return nextPage
}

这里有些有趣的事,首先是getOrElse函数,如果在映射表中,没有找到元素,则允许返回默认值:

 val nextPage = 1 + pageIndexes.getOrElse(dataSource) { 0 }

多亏这一点,我们可以节省一些条件检查代码。而另一件事更有趣的事是,如何在映射表中添加新的项。Kotlin语言的映射表实现了“+”操作符,这样添加新项就可以如此:map = map + Pair,还可以 map += Pair。即:

 pageIndexes += Pair(dataSource, nextPage)

但是,我之前可能说过,可以用to(中缀函数)返回Pair。这样前一行就如同:

 pageIndexes += dataSource to nextPage

将callback(回调函数)转换到Lambda表达式

这如同变魔术一样。在装载类中有许多重复的代码。它们大多数有复制来的相同结构,而其它则是调用API。首先,创建Retrofit回调函数。如果请求成功,无论需要什么都可以从结果获取必要的信息。最后,调用“数据装载”侦听器。在任何情况(无论成功或失败)下,loadingCount都更新了。

仅举一例:

 private void loadDesignerNewsTopStories(final int page) {
getDesignerNewsApi().getTopStories(page, new Callback<StoriesResponse>() {
@Override
public void success(StoriesResponse storiesResponse, Response response) {
if (storiesResponse != null
&& sourceIsEnabled(SourceManager.SOURCE_DESIGNER_NEWS_POPULAR)) {
setPage(storiesResponse.stories, page);
setDataSource(storiesResponse.stories,
SourceManager.SOURCE_DESIGNER_NEWS_POPULAR);
onDataLoaded(storiesResponse.stories);
}
loadingCount.decrementAndGet();
} @Override
public void failure(RetrofitError error) {
loadingCount.decrementAndGet();
}
});
}

如果不分析,这段代码理解起来相当困难。我们可以创建一个回调函数,它创建retrofit回调函数,实现前面回调函数的结构。所不同的是从结果中获取必要的信息:

 private inline fun <T> callback(page: Int, sourceKey: String,
crossinline extract: (T) -> List<PlaidItem>)
= retrofitCallback<T> { result, response ->
if (sourceIsEnabled(sourceKey)) {
val items = extract(result)
setPage(items, page)
setDataSource(items, sourceKey)
onDataLoaded(items)
}
}

这十分相似:检查是否启用源,如果启用,就调用setPagesetDataSourceonDataLoaded从结果中获取相关项。retrofitCallback的实现可使前面的函数更简单,而它可以省略了:

 private inline fun <T> retrofitCallback(crossinline code: (T, Response) -> Unit): Callback<T>
= object : Callback<T> {
override fun success(t: T?, response: Response) {
t?.let { code(it, response) }
loadingCount.decrementAndGet()
} override fun failure(error: RetrofitError) {
loadingCount.decrementAndGet()
}
}

inline修饰符是优化函数的方法,该函数的参数包含其它函数。当函数包含lambda表达式时,其转换等同于对象包含了函数的实现代码。然而,如果用inline,lambda表达式将在被调用时由它的代码来替换。如果lambda表达式返回一个值,就要用crossinline。更多资料请阅读Kotlin参考资料

两个函数是泛型,所以可以接受任何要求的类型。这样,前面的请求就可以如下:

 private fun loadDesignerNewsTopStories(page: Int) {
designerNewsApi.getTopStories(page,
callback(page, SourceManager.SOURCE_DESIGNER_NEWS_POPULAR) { it.stories })
}

该函数调用getTopStories,创建接收page和源key的回调函数,函数从结果中获取stories。类似的结构为调用的其它部分运行,但是无论需要什么结果都可以做。例如,另一个响应需要修改包含的user:

 private fun loadDribbbleUserShots(page: Int) = dribbbleLogged {
val user = dribbblePrefs.user
dribbbleApi.getUserShots(page, DribbbleService.PER_PAGE_DEFAULT,
callback(page, SourceManager.SOURCE_DRIBBBLE_USER_SHOTS) {
// this api call doesn't populate the shot user field but we need it
it.apply { forEach { it.user = user } }
})
}

如你所见,前者还要求记录到dribbble中。dribbbleLogged函数负责检查,如果不是就做其它事。

 private inline fun dribbbleLogged(code: () -> Unit) {
if (dribbblePrefs.isLoggedIn) {
code()
} else {
loadingCount.decrementAndGet()
}
}

总结

这部分展示了lambda表达式能力和作为“第一类公民(first class citizen)”函数的使用。代码的减少是巨大的(减少238%),但还不是最重要的改进。现在代码更易阅读,错误更少。记得阅读我在Github的Plaid分支中最后的提交。

前一篇:http://www.cnblogs.com/figozhg/p/5041855.html

用Kotlin语言重新编写Plaid APP:经验教训(II)的更多相关文章

  1. 用Kotlin语言重新编写Plaid APP:经验教训(I)

    原文标题:Converting Plaid to Kotlin: Lessons learned (Part 1) 原文链接:http://antonioleiva.com/plaid-kotlin- ...

  2. 用React Native编写跨平台APP

    用React Native编写跨平台APP React Native 是一个编写iOS与Android平台实时.原生组件渲染的应用程序的框架.它基于React,Facebook的JavaScript的 ...

  3. 释放Android的函数式能量(I):Kotlin语言的Lambda表达式

    原文标题:Unleash functional power on Android (I): Kotlin lambdas 原文链接:http://antonioleiva.com/operator-o ...

  4. Kotlin 语言高级安卓开发入门

    过去一年,使用 Kotlin 来为安卓开发的人越来越多.即使那些现在还没有使用这个语言的开发者,也会对这个语言的精髓产生共鸣,它给现在 Java 开发增加了简单并且强大的范式.Jake Wharton ...

  5. 用Xamarin和Visual Studio编写iOS App

    一说开发 iOS app,你立马就会想到苹果的开发语言 Objective C/Swift 和 Xcode.但是,这并不是唯一的选择,我们完全可以使用别的语言和框架. 一种主流的替换方案是 Xamar ...

  6. 认识一下Kotlin语言,Android平台的Swift

    今天在CSDN首页偶然看到一个贴子JetBrains正式公布Kotlin 1.0:JVM和Android上更好用的语言 看完后,感觉Kotlin语法非常简洁,有一系列动态语言的特点,Lambda表达式 ...

  7. Hybrid App经验解读 一

    郑昀编纂 关键词:Hybrid,Zepto,Fastclick,Backbone,sui,SPA,pushState,跨域,CORS click 事件还是 tap 事件? Zepto 的 show/h ...

  8. Kotlin语言编程技巧集

    空语句 Kotlin 语言中的空语句有 {} Unit when (x) { 1 -> ... 2 -> ... else -> {} // else -> Unit } Wh ...

  9. Kotlin语言学习笔记(2)

    类(classes) // 类声明 class Invoice { } // 空的类 class Empty // 主体构造器(primary constructor) class Person co ...

随机推荐

  1. sublime text 3 + python配置,完整搭建及常用插件安装

    四年的时间,一直使用EmEditor编辑器进行Python开发,之前是做面向过程,只需要将一个单独的py文件维护好即可,用着也挺顺手,但是最近在做面向对象的开发,不同的py文件中相互关联较多,感觉单纯 ...

  2. 如何利用mono把.net windows service程序迁移到linux上

    How to migrate a .NET Windows Service application to Linux using mono? 写在最前:之所以用要把windows程序迁移到Linux上 ...

  3. Handlebars 模板引擎之前后端用法

    前言 不知不觉间,居然已经这么久没有写博客了,坚持还真是世界上最难的事情啊. 不过我最近也没闲着,辞工换工.恋爱失恋.深圳北京都经历了一番,这有起有落的生活实在是太刺激了,就如拿着两把菜刀剁洋葱一样, ...

  4. mvc4 自定义HtmlHelper

    好久没写博客了,最近只看博客不写的习惯很不好啊. 好了,最近的项目中大量的用到了表单,很多表单有特殊的编写,但是在该项目中又有很多重复的地方,这个时候若能封装成htmlhelper将大大降低工作量的. ...

  5. 基于Quick-cocos2d-x的资源更新方案 二

    写在前面 又是12点半了,对于一个程序员来说,这是一个黄金时间,精力旺盛,我想,是最适合整理和分享一些思路的时候了. 自从上次写了 基于Quick-cocos2d-x的资源更新方案 同样可见quick ...

  6. 如何利用 Visual Studio 自带工具提高开发效率

    Visual Stuido 是一款强大的Windows 平台集成开发工具,你是否好好地利用了它呢? 显示行号 有些时候(比如错误定位)的时候,显示行号将有利于我们进行快速定位. 如何显示 1. 工具 ...

  7. Parallel并行之乱用

    关于Parallel我也不细说了,一则微软封装的很好用,二来介绍这个的遍地都是. 我要说的是,要想成为一个优秀的标题党,一定要把重点放到别的地方,为了节省大家阅读时间,我先把结论说了,然后再慢慢从头说 ...

  8. JS作用域面试题总结

    关于JS作用域问题,是面试的时候面试官乐此不疲的面试题,有时候确实是令人抓狂,今天看到一个讲解这个问题的视频,明白了那些所谓的“原理”顿时有种豁然开朗的感觉~~~ 1.js作用域(全局变量,局部变量) ...

  9. 计算机程序的思维逻辑 (47) - 堆和PriorityQueue的应用

    45节介绍了堆的概念和算法,上节介绍了Java中堆的实现类PriorityQueue,PriorityQueue除了用作优先级队列,还可以用来解决一些别的问题,45节提到了如下两个应用: 求前K个最大 ...

  10. Java 计算N阶乘末尾0的个数-LeetCode 172 Factorial Trailing Zeroes

    题目 Given an integer n, return the number of trailing zeroes in n!. Note: Your solution should be in ...