Java 与 Kotlin 中各种 findViewById 的方式

Java 中的方式

1.从 Android Support Library 26.0.0 Beta 1 开始 findViewById 将不再需要强转了。


  1. private Button mButton;
  2. ...
  3. mButton = findViewById(R.id.button);

2.使用注解来注入对应类型的 View,最流行的非 ButterKnife 莫属,搭配 Android Butterknife Zelezny 插件自动生成以下的代码,不再需要手写 findViewById 了。需要注意的是重构布局时,如果 id 对应的 View 类型发生了变化而又忘记修改代码,则会在运行时产生异常。


  1. @BindView(R.id.button)
  2. Button mButton;

3.使用 Data Binding Library 自动查找所有 View 并缓存到 binding 实例中以供访问。性能超过手写的 findViewById,因为它只遍历了一遍 XML 布局,而 findViewById 每次都会去遍历 XML 布局;include 布局中的 view 也能同样能访问,并且保留结构。


  1. ActivityMainBinding binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
  2. binding.button.setText("DataBinding 是坠吼的");
  3. binding.include_layout_id.textView.setText("对对对");

Kotlin 中的方式

1.对应 Java 中的方式 1,并将 View 优化为非空类型。lateinit 关键字代表延迟初始化,假如没有初始化就访问会抛异常。


  1. private lateinit var button:Button
  2. ...
  3. button = findViewById(R.id.button)

TODO: lateinit 的用法及注意事项

2.改为使用 lazy 代理延迟初始化,使用 val 限制属性不可重新赋值,并做些许封装。这时长得就比较像 Java 中的方式 2 了。


  1. private val button by bindView<Button>(R.id.button)
  2. // 或者
  3. private val button:Button by bindView(R.id.button)
  4. ...
  5. fun <T : View> Activity.bindView(@IdRes res: Int): Lazy<T> {
  6. return lazy { findViewById<T>(res) }
  7. }

3.使用 Kotlin Android Extensions 直接生成对应的 View 作为属性。不需要 findViewById,不需要定义变量,直接使用。使用时需要注意访问的 View 属于哪个 Layout,因为智能提示的候选项会提供所有布局中的 View 供你选择,然后帮你 import 对应包以便你访问这个 View;假如 import 的多个同一层级的 layout 中具有相同的 id,则这个 id 对应的 View 将无法访问。


  1. import kotlinx.android.synthetic.main.activity_kotlin_main.*
  2. import kotlinx.android.synthetic.main.activity_kotlin_main.view.*
  3. ...
  4. button.setText("Kotlin Android Extensions 我不太喜欢")
  5. inflateView.textView.setText("因为比较混乱,容易出错")

Java 与 Kotlin 中各种 findViewById 的方式

相关推荐