網(wǎng)站首頁 編程語言 正文
viewBinding
優(yōu)點(diǎn)
當(dāng)一個頁面布局出現(xiàn)多個控件時,使用findViewById去進(jìn)行控件綁定,過于冗長,且存在NULL指針異常風(fēng)險。viewBinding直接創(chuàng)建對視圖的引用,不存在因控件ID不存在而引發(fā)的NULL指針異常。并且在綁定類中對控件添加@NonNull
注解
findViewById | viewBinding |
---|---|
冗長 | 簡短 |
NULL | NULL安全 |
配置
3.6之前的版本在build.gradle文件中聲明如下定義
viewBinding { enabled = true }
4.0以上的版本在build.gradle文件中聲明如下定義
buildFeatures { viewBinding = true }
聲明如上定義之后,點(diǎn)擊同步(Sync Now)按鈕,系統(tǒng)會自動生成viewBinding類,例如MainActivity會生成名為ActivityMainBinding的類,ReceiveActivity會生成名為ActivityReceiveBinding的類,以此類推;
以上viewBinding類會生成在如下路徑文件中
build//generated//data_binding_base_class_source_out//debug//out//com.你的包名//databinding
使用
使用步驟很簡單,需要被調(diào)用的控件聲明id就行,然后聲明viewBinding類對象
private ActivityMainBinding binding;
綁定視圖:
binding = ActivityMainBinding.inflate(getLayoutInflater()); setContentView(binding.getRoot());
控件引用:
binding.postMes.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { startActivity(new Intent(MainActivity.this,ReceiverActivity.class)); } });
源碼解析
如上所示,我們使用了ActivityMainBinding.inflate()方法進(jìn)行視圖綁定和binding.getRoot()方法獲取視圖。
首先我們在外部通過調(diào)用ActivityMainBinding.inflate()方法。
@NonNull public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater) { return inflate(inflater, null, false); }
然后內(nèi)部進(jìn)行重載,添加我們的Avcivity的布局文件,并調(diào)研bind(root)方法
@NonNull public static ActivityMainBinding inflate(@NonNull LayoutInflater inflater, @Nullable ViewGroup parent, boolean attachToParent) { View root = inflater.inflate(R.layout.activity_main, parent, false); if (attachToParent) { parent.addView(root); } return bind(root); }
在bind方法中進(jìn)行控件綁定,通過其findChildViewById()方法
@NonNull public static ActivityMainBinding bind(@NonNull View rootView) { // The body of this method is generated in a way you would not otherwise write. // This is done to optimize the compiled bytecode for size and performance. int id; missingId: { id = R.id.content; TextView content = ViewBindings.findChildViewById(rootView, id); if (content == null) { break missingId; } id = R.id.postMes; Button postMes = ViewBindings.findChildViewById(rootView, id); if (postMes == null) { break missingId; } return new ActivityMainBinding((LinearLayout) rootView, content, postMes); } String missingId = rootView.getResources().getResourceName(id); throw new NullPointerException("Missing required view with ID: ".concat(missingId)); }
然后在findChildViewById()方法中最終也使用到了findViewById()方法,但差距在于跳過視圖本身
/** * Like `findViewById` but skips the view itself. * * @hide */ @Nullable public static <T extends View> T findChildViewById(View rootView, @IdRes int id) { if (!(rootView instanceof ViewGroup)) { return null; } final ViewGroup rootViewGroup = (ViewGroup) rootView; final int childCount = rootViewGroup.getChildCount(); for (int i = 0; i < childCount; i++) { final T view = rootViewGroup.getChildAt(i).findViewById(id); if (view != null) { return view; } } return null; }
最后通過將獲取到的控件定義與內(nèi)部定義的字段進(jìn)行縫合,以暴露給外部使用
@NonNull private final LinearLayout rootView; @NonNull public final TextView content; @NonNull public final Button postMes;
private ActivityMainBinding(@NonNull LinearLayout rootView, @NonNull TextView content, @NonNull Button postMes) { this.rootView = rootView; this.content = content; this.postMes = postMes; }
DataBinding
配置
依舊在build.gradle文件中配置如下定義:
dataBinding { enabled = true }
創(chuàng)建實(shí)體類
實(shí)體類通過繼承BaseObservable
類,而BaseObservable
又實(shí)現(xiàn)了Observable
,從而獲取添加和移除監(jiān)聽的機(jī)制。
在get()方法中使用@Bindable
注解,會自動生成BR類,此類中將添加@Bindable
的字段聲明成常量,然后在set()方法使用notifyPropertyChanged()配合使用,當(dāng)數(shù)據(jù)發(fā)生變化時,dataBinding會自動修改該字段的值。
public class EventMessage extends BaseObservable { public String title; public EventMessage(){ } public EventMessage(String title){ this.title = title; } @Bindable public String getTitle() { return title; } /** * @param title*/ public void setTitle(String title) { this.title = title; notifyPropertyChanged(BR.title); } }
創(chuàng)建布局
創(chuàng)建layout標(biāo)簽布局才會生成ActivityMainBinding(以及布局文件名而定)
EditText通過使用如下定義進(jìn)行綁定,
android:text="@={viewModel.message.title}"
Button通過如下定義進(jìn)行點(diǎn)擊事件監(jiān)聽
android:onClick="@{viewModel.setText}"
以上兩者的存在差距,EditText多了一個=
,而Button沒有,并且Button綁定監(jiān)聽事件,不需要加()
<?xml version="1.0" encoding="utf-8"?> <layout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools"> <data> <variable name="viewModel" type="com.franzliszt.databinding.ViewModel" /> </data> <LinearLayout android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" android:orientation="vertical" android:gravity="center"> <EditText android:id="@+id/inputText" android:layout_width="wrap_content" android:layout_height="wrap_content" android:hint="default" android:text="@={viewModel.message.title}"/> <TextView android:id="@+id/ShowText" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text=""/> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="update" android:onClick="@{viewModel.setText}"/> </LinearLayout> </layout>
創(chuàng)建viewModel
通過監(jiān)聽Button點(diǎn)擊事件,在其中監(jiān)聽EditText輸入事件,并將其輸入的字符串顯示在TextView中
public class ViewModel { private ActivityMainBinding binding; public EventMessage message; public ViewModel(ActivityMainBinding binding, EventMessage message){ this.binding = binding; this.message = message; } public void setText(View view){ String str = message.getTitle(); binding.ShowText.setText(str); } }
dataBinding綁定
private ActivityMainBinding binding;
binding = DataBindingUtil.setContentView( this,R.layout.activity_main ); binding.setViewModel(new ViewModel(binding,new EventMessage()));
原文鏈接:https://blog.csdn.net/News53231323/article/details/125002668
相關(guān)推薦
- 2022-04-27 Python學(xué)習(xí)筆記之字典,元組,布爾類型和讀寫文件_python
- 2022-06-08 Flutter集成高德地圖并添加自定義Maker的實(shí)踐_Android
- 2021-11-21 關(guān)于.NET6?Minimal?API的使用方式詳解_實(shí)用技巧
- 2022-03-28 Python?NumPy實(shí)用函數(shù)筆記之a(chǎn)llclose_python
- 2022-09-16 Oracle查看表空間使用率以及爆滿解決方案詳解_oracle
- 2023-02-14 C#實(shí)現(xiàn)ComboBox變色的示例代碼_C#教程
- 2022-06-25 C#將DataGridView中的數(shù)據(jù)保存到CSV和Excel中_C#教程
- 2023-07-25 node項(xiàng)目使用crypto模塊為用戶密碼加密
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支