Kotlin高阶函数的应用

2020-09-29  本文已影响0人  code希必地

高阶函数非常适用于简化各种API的调用,一些API原有用法在使用高阶函数进行简化后其可读性和易用性方面会有很大的提高。

1、简化SharedPreferences的使用

在简化前先来回顾下SharedPreferences的使用:

val sp = getPreferences(Context.MODE_PRIVATE)
val editor = sp.edit()
editor.apply {
    putString("name", name)
    putInt("age", age)
    apply()
}

虽然API的使用虽然已经很简单但是我们依然可以使用高阶函数对其进行简化,首先创建一个SharedPreferences.kt文件,然后在文件中定义save()方法

inline fun SharedPreferences.save(block: SharedPreferences.Editor.() -> Unit) {
    val editor = edit()
    editor.block()
    editor.apply()
}

下面解释一下这个高阶函数:

 getSharedPreferences("LoginActivity", Context.MODE_PRIVATE).save {
            putString("name", "阿三")
            putInt("age",19)
        }

这里我们没有调用apply()commit(),高阶函数内部已经调用了,是不是很简单啊。
最后不得不提一下Google中提供了KTX扩展库中包含了SharedPreferences的简化用法,我们在module的gradle中导入implementation 'androidx.core:core-ktx:1.3.1'即可使用KTX扩展库了。下面看下KTX扩展库中对SharedPreferences的简化使用

 getSharedPreferences("LoginActivity", Context.MODE_PRIVATE).edit {
            putString("name", "阿三")
            putInt("age", 19)
        }

感兴趣的可以看下edit的定义,和我们刚才定义的高阶函数基本是一样的。

2、简化ContentValue的使用

ContentValues的基本用法如下:

val contentValues = ContentValues()
contentValues. put("autor", "曹雪芹")
contentValues. put("price", 400)
contentValues.  put("pages", 4000)   

在真正开始简化之前,我们先来看下mapOf()用法。

val map= mapOf(1 to "Apple",
               2 to "Banana",
               3 to "Orange")

可以看到使用1 to "Apple"创建键值对,在Kotlin中使用A to B语法结构会创建一个Pair对象,有了这个知识后我们创建一个ContentValues.kt文件在文件中创建方法build()方法,用于生成ContentValues对象。

fun build(vararg pairs: Pair<String, Any?>): ContentValues {}
fun build(vararg pair: Pair<String, Any?>): ContentValues {
    val contentValues = ContentValues()
    for (item in pair) {
        val key = item.first
        val value = item.second
        when (value) {
            is String -> contentValues.put(key, value)
            is Long -> contentValues.put(key, value)
            is Int -> contentValues.put(key, value)
            is Double -> contentValues.put(key, value)
            is ByteArray -> contentValues.put(key, value)
            is Boolean -> contentValues.put(key, value)
            is Float -> contentValues.put(key, value)
            is Short -> contentValues.put(key, value)
            is Byte -> contentValues.put(key, value)
            null -> contentValues.putNull(key)
        }
    }
    return contentValues
}

有了build方法后,使用ContentValues就变得很简单了

val contentValues = build(
            "autor" to "曹雪芹",
            "price" to 400,
            "pages" to 4000
        )

这样我们就能通过类似mapOf()方式来构建ContentValues了,但是这和高阶函数没有任何关系,下面我们结合apply{}来简化一下:

fun build(vararg pair: Pair<String, Any?>)=ContentValues().apply {
    for (item in pair) {
        val key = item.first
        val value = item.second
        when (value) {
            is String -> put(key, value)
            is Long -> put(key, value)
            is Int -> put(key, value)
            is Double -> put(key, value)
            is ByteArray -> put(key, value)
            is Boolean -> put(key, value)
            is Float -> put(key, value)
            is Short -> put(key, value)
            is Byte -> put(key, value)
            null -> putNull(key)
        }
    }
}

如果你很熟悉apply{}的话,代码也很好理解,我们使用ContentValues()对象调用applyContentValues()对象传入Lambda表达式中并作为其上下文,而且apply的返回值就是调用者本身。其实KTX扩展库中也提供了类似方法contentValuesOf来实现这个功能,感兴趣的可以自行实现。

上一篇下一篇

猜你喜欢

热点阅读