Kotlin 的一些实用小技巧


1.Lazy Loading(懒加载)

延迟加载有几个好处。延迟加载能让程序启动时间更快,因为加载被推迟到访问变量时。 这在使用 Kotlin 的 Android 应用程序而不是服务器应用程序中特别有用。对于 Android 应用,我们自然希望减少应用启动时间,以便用户更快地看到应用内容,而不是等待初始加载屏幕。

懒加载也是更有效率的内存,因为我们只需要调用资源才能将资源加载到内存中。例如:

val gankApi: GankApi by lazy {
    val retrofit: Retrofit = Retrofit.Builder()
            .baseUrl(API_URL)
            .addConverterFactory(MoshiConverterFactory.create())
            .build()
    retrofit.create(GankApi::class.java)
}

如果用户从没有调用 GankApi ,则永远不会加载。因此也不会占用所需资源。

当然懒加载也能较好的用于封装初始化:

val name: String by lazy {
    Log.d(TAG, "executed only first time")
    "Double Thunder"
}

也可以用 lazy 做单例。

//Java
class MyJavaClass {
    class MyItem {}
    MyItem item;
    final MyItem getItem() {
        if (item == null) {
            item = new MyItem ();
        }
        return item;
    }
}

//by Kotlin
class MyKotlinClass {
    val item by lazy { MyItem() }
}

如果你不担心多线程问题或者想提高更多的性能,你也可以使用

lazy(LazyThreadSafeMode.NONE){ ... } 

2. 自定义 Getters/Setters

Kotlin 会自动的使用 getter/setter 模型,但也有一些情况(倒如 Json)我们需要用自定制 getter 和 setter。例如:

@ParseClassName("Book")
class Book : ParseObject() {

    // getString() and put() are methods that come from ParseObject
    var name: String
        get() = getString("name")
        set(value) = put("name", value)

    var author: String
        get() = getString("author")
        set(value) = put("author", value)
}

3. Lambdas

button.setOnClickListener { view ->
    startDetailActivity()
}

toolbar.setOnLongClickListener { 
    showContextMenu()
    true
}

4.Data Classes(数据类)

数据类是一个简单版的 Class,它自动添加了包括 equals(),hashCode(), copy() 和 toString() 方法。将数据与业务逻辑分开。

data class User(val name: String, val age: Int)

如果使用 Gson 解析 Json 的数据类,则可以使用默认值构造函数:

// Example with Gson's @SerializedName annotation
data class User(
    @SerializedName("name") val name: String = "",
    @SerializedName("age") val age: Int = 0
)

5. 集合过滤

val users = api.getUsers()
// we only want to show the active users in one list
val activeUsersNames = items.filter { 
    it.active // the "it" variable is the parameter for single parameter lamdba functions
}
adapter.setUsers(activeUsers)

6. Object Expressions(对象表达式)

Object Expressions 允许定义单例。例如:

package com.savvyapps.example.util

import android.os.Handler
import android.os.Looper

// notice that this is object instead of class
object ThreadUtil {

    fun onMainThread(runnable: Runnable) {
        val mainHandler = Handler(Looper.getMainLooper())
        mainHandler.post(runnable)
    }
}

ThreadUtil 则可以直接调用静态类方法:

ThreadUtil.onMainThread(runnable)

以类似的方式,我们创建对象而不是匿名内部类:

viewPager.addOnPageChangeListener(object : ViewPager.OnPageChangeListener {
    override fun onPageScrollStateChanged(state: Int) {}

    override fun onPageScrolled(position: Int, positionOffset: Float, positionOffsetPixels: Int) {}

    override fun onPageSelected(position: Int) {
        bindUser(position)
    }
});

这两个都基本上是相同的事情 - 创建一个类作为声明对象的单个实例。

7. Companion Object(伴生对象)

Kotlin 是没有静态变量与方法的。相对应的,可以使用伴生对象。伴生对象允许定义的常量和方法,类似于 Java 中的 static。有了它,你可以遵循 newInstance 的片段模式。

class ViewUserActivity : AppCompatActivity() {

    companion object {

        const val KEY_USER = "user"

        fun intent(context: Context, user: User): Intent {
            val intent = Intent(context, ViewUserActivity::class.java)
            intent.putExtra(KEY_USER, user)
            return intent
        }
    }
    
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_cooking)
        
        val user = intent.getParcelableExtra<User>(KEY_USER)
        //...
    }
}

我们熟悉的使用:

val intent = ViewUserActivity.intent(context, user)
startActivity(intent)

8.Global Constants(全局常量)

Kotlin 允许跨越整个应用的全局常量。通常,常量应尽可能减少其范围,但是全局都需要这个常量时,这是一个很好的方式。

const val PRESENTATION_MODE_PRESENTING = "presenting"
const val PRESENTATION_MODE_EDITING = "editing"

9.Optional Parameters(可选参数)

可选参数使得方法调用更加灵活,而不必传递 null 或默认值。 例如:这在定义动画时:

fun View.fadeOut(duration: Long = 500): ViewPropertyAnimator {
    return animate()
            .alpha(0.0f)
            .setDuration(duration)
}
icon.fadeOut() // fade out with default time (500)
icon.fadeOut(1000) // fade out with custom time

10. Extensions(扩展属性)

例如:在 Activity 调用键盘的隐藏

fun Activity.hideKeyboard(): Boolean {
    val view = currentFocus
    view?.let {
        val inputMethodManager = getSystemService(Context.INPUT_METHOD_SERVICE) 
                as InputMethodManager
        return inputMethodManager.hideSoftInputFromWindow(view.windowToken,
                InputMethodManager.HIDE_NOT_ALWAYS)
    }
    return false
}

推荐一个收集 Extensions 的网站 。 kotlinextensions.com

11. lateinit

对于 Null 的检查是 Kotlin 的特点之一,所以在数据定义时,初始化数据。但有一些在 Android 中某些属性需要在 onCreate() 方法中初始化。

private lateinit var mAdapter: RecyclerAdapter<Transaction>

override fun onCreate(savedInstanceState: Bundle?) {
   super.onCreate(savedInstanceState)
   mAdapter = RecyclerAdapter(R.layout.item_transaction)
}

如果是基础数据类型:

var count: Int by Delegates.notNull<Int>()
var name:String by Delegate()

如果使用 Butter Knife:

@BindView(R.id.toolbar) lateinit var toolbar: Toolbar
override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        ButterKnife.bind(this)
        // you can now reference toolbar with no problems!
        toolbar.setTitle("Hello There")
}

12. Safe Typecasting(安全转换)

在 Android 中需要安全类型转换。当您首先在 Kotlin 中进行类型转换时,您可以这样实现:

var feedFragment: FeedFragment? = supportFragmentManager
    .findFragmentByTag(TAG_FEED_FRAGMENT) as FeedFragment

但实际上这样只能导致崩溃。当调用『as』时,它将进行对象转换,但如果转换的对象为『null』时,则会报错。正确的使用方式应该是用『as?』:

var feedFragment: FeedFragment? = supportFragmentManager
    .findFragmentByTag(TAG_FEED_FRAGMENT) as? FeedFragment
if (feedFragment == null) {
    feedFragment = FeedFragment.newInstance()
    supportFragmentManager.beginTransaction()
            .replace(R.id.root_fragment, feedFragment, TAG_FEED_FRAGMENT)
            .commit()
}

13. let 操作符

『let』操作符:如果对象的值不为空,则允许执行这个方法。

//Java
if (currentUser != null) {
    text.setText(currentUser.name)
}

//instead Kotlin
user?.let {
    println(it.name)
}

14. isNullOrEmpty | isNullOrBlank

我们需要在开发 Android 应用程序时多次验证。 如果你没有使用 Kotlin 处理这个问题,你可能已经在 Android 中发现了 TextUtils 类。

if (TextUtils.isEmpty(name)) {
    // alert the user!
}
public static boolean isEmpty(@Nullable CharSequence str) {
    return str == null || str.length() == 0;
}

如果 name 都是空格,则 TextUtils.isEmpty 不满足使用。则 isNullorBlank 可用。

public inline fun CharSequence?.isNullOrEmpty(): Boolean = this == null || this.length == 0

public inline fun CharSequence?.isNullOrBlank(): Boolean = this == null || this.isBlank()

// If we do not care about the possibility of only spaces...
if (number.isNullOrEmpty()) {
    // alert the user to fill in their number!
}

// when we need to block the user from inputting only spaces
if (name.isNullOrBlank()) {
    // alert the user to fill in their name!
}

15. 避免 Kotlin 类的抽象方法

也是尽可能的使用 lambdas 。这样可以实现更简洁直观的代码。例如在 Java 中的点击监听为:

public interface OnClickListener {
    void onClick(View v);
}

在 Java 中使用:

view.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
        // do something
    }
});

而在 Kotlin 中:

view.setOnClickListener { view ->
    // do something
}

//同时也可以为
view.setOnClickListener {
    // do something
}

view.setOnClickListener() {
    // do something
}

如果在 Kotlin 是使用单抽象方法的话:

view.setOnClickListener(object : OnClickListener {
    override fun onClick(v: View?) {
        // do things
    }
})

下面是另一种方法:

private var onClickListener: ((View) -> Unit)? = null
fun setOnClickListener(listener: (view: View) -> Unit) {
    onClickListener = listener
}

// later, to invoke
onClickListener?.invoke(this)

16. with 函数

with 是一个非常有用的函数,它包含在 Kotlin 的标准库中。它接收一个对象和一个扩展函数作为它的参数,然后使这个对象扩展这个函数。这表示所有我们在括号中编写的代码都是作为对象(第一个参数) 的一个扩展函数,我们可以就像作为 this 一样使用所有它的 public 方法和属性。当我们针对同一个对象做很多操作的时候这个非常有利于简化代码。

with(helloWorldTextView) {
    text = "Hello World!"
    visibility = View.VISIBLE
}

17. Static Layout Import

Android 中最常用的代码之一是使用 findViewById() 来获取对应 View。

有一些解决方案,如 Butterknife 库,可以节省很多代码,但是 Kotlin 采取另一个步骤,允许您从一个导入的布局导入对视图的所有引用。

例如,这个 XML 布局:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    >

    <TextView
        android:id="@+id/tvHelloWorld"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>
</RelativeLayout>

在 Activity 中:

//导入对应的 xml
import kotlinx.android.synthetic.main.activity_main.*

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        //直接使用
        tvHelloWorld.text = "Hello World!"
    }
}

18. 用 Kotlin 实现 POJO 类

在 Java 中

public class User {
   private String firstName;
   private String lastName;

   public String getFirstName() {
       return firstName;
   }
   public void setFirstName(String firstName) {
       this.firstName = firstName;
   }

   public String getLastName() {
       return lastName;
   }
   public void setLastName(String lastName) {
       this.lastName = lastName;
   }
}

而在 Kotlin 中可以简化成:

class User {
   var firstName: String? = null
   var lastName: String? = null
}

19. 减少 AsyncTash 的使用

搭配 Anko lib 使用。后台和主线程的切换特别直观和简单。uiThread 在主线程上运行,并且我们不需要关心 Activity 的生命周期(pause 与 stop), 所以也不会出错了。

doAsync {
    var result = expensiveCalculation()
    uiThread {
        toast(result)
    }
}

20. apply 函数

它看起来于 with 很相似,但是是有点不同之处。apply 可以避免创建 builder 的方式来使用,因为对象调用的函数可以根据自己的需要来初始化自己,然后 apply 函数会返回它同一个对象:

user = User().apply {
    firstName = Double
    lastName = Thunder
}

21. is 操作符

我们可以在运⾏时通过使⽤ is 操作符或其否定形式 !is 来检查对象是否符合给定类型:
在许多情况下,不需要在 Kotlin 中使⽤显式转换操作符,因为编译器跟踪不可变值的 is 检查,并在需要时⾃动插⼊(安全的)智能转换:

fun display(myView: View) {
    //判断 myView 是否为ImageView,并自动转换
    if (myView is ImageView) {
        myView.setImageResource(R.drawable.image)
    } else if (myView is TextView) {
        myView.setText("Double Thunder")
    }
}
//上面也可以简化为:
fun display2(myView: View) {
    when (myView) {
        is ImageView -> myView.imageAlpha = 10
        is TextView -> myView.text = "Double Thunder"
    }
}

22.filterNotNull 过滤操作符。

过滤所有元素中不是 null 的元素。


参考文章:

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

推荐阅读更多精彩内容