Kotlin —  最佳实践

使用表达式

不建议:

fun getDefaultSex(name:String):User{
    if(name=="mike"||name=="jack"){
         return male; 
    }
    if(name="marry"||name="jane"){
         return female
    }
    return male
}

建议:

fun getDefaultSex(name:String)=when(name){
    "mike","jack"->male
     "marry","jane"->female
      else->male
}

每次使用if的时候都考虑一下,表达式是否可以用更简洁的方式代替。

try-catch也是一个表达式:

var string=""
string=try{
    return "Ok"
}catch(e:Exception){
    return "Error"
}

扩展函数

在Java中,我们经常在Util类中创建静态Util方法。这种模式翻译成Kotlin将如下所示:

不建议:

object StringUtils{
    fun countNum(num:Int):Int{
        return num*2
  }
}
StringUtils.countNum(3)

Kotlin允许移除不必要的包装类,并使用扩展函数。通常我们还可以利用扩展函数来挺高可读性。

建议:

 fun Int.countNum():Int{
        return plus(2)
  }
  3.countNum()

输出结果:

5

变量名代替Setter

在Java中,我们常为变量添加getter/setter方法,这样就会有大量的模板代码。

不建议:

val params=User()
.setName("android coder")
.setAge(25)
.setSex("male")

在Kotlin中,命名和默认参数可以直接赋值:

建议:

val params=User(name="android coder",age=25,sex="male")

apply()为Grouping Object初始化

不建议:

val user=User()
user.Name="android code"
user.age=25
user.sex=male
user.isDream=true

扩展函数apply()可以对对象的初始化代码进行分组和集中。这样,我们就不必一遍一遍的重复变量名称了。

建议:

val user=User().apply{
    name="android coder"
    age=25
    sex="male"
    isDream=true  
}

当处理Java类时apply()往往会很有用

不要重载默认构造函数

不要重载方法和构造函数来实现默认参数

不建议:

fun user(name:String){
    user(name,true)
}
fun find(name:String,isDream:Boolean){
}

建议:

fun user(name:String,isDream:Boolean=true){
}

事实上,默认参数几乎取代了方法和构造函数重载的写法,因为,重载主要用于创建默认函数。

清晰简单处理NULL

避免if-null检查

不建议:

if(user==null||user.name==null||user.age==null){
    throw IllegalArgumentException("Invalid User")
}
val name=user.name

每次我们都要写一个if-null检查。Kotlin提供了更好的方法来处理空值。使用as??:来检查类型,如果不是我们需要的,则抛出一个异常

建议:

user as? VIPUser ?:throw IllegalArgumentException("NO VIP")
user.getVIP()

避免not-null断言!!

不建议:

user!!.vip!!.name!!.address!!

双重感叹号看起来会让你很烦。这是故意这么设计的,Kotlin的设计者正试着让你朝着一个更好的方向发展,而这个解决方案并不涉及编译器无法验证的断言。来源于:“Kotlin in Action”--->Dmitry Jemerov和Svetlana Isakova。

考虑使用let()

有时候,let()可以是if的简洁版。但是,你必须以合理的方式来使用它,以免出现问题。

val user:User?=findUser()
if(user!=null){
    println(user.name)
}

当使用let()时,不需要额外的变量

class findUser{
  val name:String="
}
findUser()?.let{println(it.name)}
//或者
findUser()?.customer?.let(::print)

简单的单表达函数映射

不建议:

fun mapToDTO(entity: SnippetEntity): SnippetDTO {
    val dto = SnippetDTO(
            code = entity.code,
            date = entity.date,
            author = "${entity.author.firstName} ${entity.author.lastName}"
    )
    return dto
}

使用单个表达式函数和命名参数,我们可以在对象之间编写简单,简洁和可读的映射。

建议:

fun mapToDTO(entity: SnippetEntity) = SnippetDTO(
        code = entity.code,
        date = entity.date,
        author = "${entity.author.firstName} ${entity.author.lastName}"
)
val dto = mapToDTO(entity)

如果你对扩展功能熟悉,你可以在这里使用它们来使函数定义和使用更短,更易读。同时,我们不会用映射逻辑来破坏value对象

fun SnippetEntity.toDTO() = SnippetDTO(
        code = code,
        date = date,
        author = "${author.firstName} ${author.lastName}"
)
val dto = entity.toDTO()

在Property Initializers时参考构造函数参数

不建议:

class UsersClient(baseUrl: String, appName: String) {
    private val usersUrl: String
    private val httpClient: HttpClient
    init {
        usersUrl = "$baseUrl/users"
        val builder = HttpClientBuilder.create()
        builder.setUserAgent(appName)
        builder.setConnectionTimeToLive(10, TimeUnit.SECONDS)
        httpClient = builder.build()
    }
    fun getUsers(){
        //call service using httpClient and usersUrl
    }
}

我们可以在属性初始化时引用主构造函数(而不是在init块中)。apply()可以帮助初始化代码分组。

class UsersClient(baseUrl: String, appName: String) {
    private val usersUrl = "$baseUrl/users"
    private val httpClient = HttpClientBuilder.create().apply {
        setUserAgent(appName)
        setConnectionTimeToLive(10, TimeUnit.SECONDS)
    }.build()
    fun getUsers(){
        //call service using httpClient and usersUrl
    }
}

解构

一方面,解构对于从一个函数返回多个值是很有用的。我们可以定义一个自己的数据类,或者使用Pair

data class User(name:String,age:Int)
fun createUser:User{
    return User("android coder",25)
}
//使用解构
val (name,age)=createUser()

另一方面,解构可以用于map映射

val map=mapOf("android coder" to name,25 to age)
for((name,age) in map){
    //do something
}

创建特殊的结构

listOf、mapOf、和infix函数可以用来创建JSON。

val user=mapOf(
        "name" to "android coder",
        "age"    to  25,
        "language" to listOf("chinese","english")
        "address"  to mapOf(
                "city"  to  "china"
                "pronvice" to "beijing"
                "street"   to "chaoyang"
      )
)

通常我们应该使用数据类和对象映射来创建JSON。但有时(例如在测试中)这是非常有用的。

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

推荐阅读更多精彩内容