直接引用布局 ID

都知道,在 Kotlin 中可以不再使用繁琐的 findViewById(),而是可以直接调用布局 ID,例如:

<TextView
    android:id=@+id/text_download
    android:layout_width=match_parent
    android:layout_height=wrap_content
    android:gravity=center_horizontal
    android:text=Download />

在 Java 中你需要……

TextView t_d = (TextView)findViewById(R.id.text_download);
t_d.setText(daolnwoD);

但是,在 Kotlin 中可以直接:

text_download.text = daolnwoD

但是这个地方有一个坑,就是不能在 onCreateView() 返回之前使用,不过这只有 Fragment 需要注意。

打个比方,这是你的某一个 Fragment:

class DeviceFragment : Fragment() {

    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val root = inflater.inflate(R.layout.fragment_device, container, false)
        return root
    }
}

加入这个时候,你要随着布局一起执行方法 setInfo(),你可能会这么做:

(布局文件同上)

class DeviceFragment : Fragment() {

    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val root = inflater.inflate(R.layout.fragment_device, container, false)
        setInfo()
        return root
    }

    fun setInfo(){
        text_download.text = daolnwoD
    }
}

看起来没有问题,但是运行时很可能会触发空指针

java.lang.NullPointerException: Attempt to invoke virtual method \’……\’ on a null object reference

后来在 Rikka 的开发者群里问了才知道,Kotlinx 的 View 必须在 Fragment 的 onCreateView() 返回之后才能用

也就是说,你需要这么改:

class DeviceFragment : Fragment() {

    override fun onCreateView(
        inflater: LayoutInflater,
        container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        val root = inflater.inflate(R.layout.fragment_device, container, false)
        return root
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        setInfo()
    }

    fun setInfo(){
        text_download.text = daolnwoD
    }
}

即,将方法放进 onViewCreated() 中执行。