还在用findViewById,不来了解下其它方式?

2024-02-10 22:04
文章标签 方式 了解 findviewbyid

本文主要是介绍还在用findViewById,不来了解下其它方式?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

众所周知,都2220年了,findViewById已经是一种非常繁琐的操作,如果要去获取的id数量多,则对开发更加不友好。如果一个页面id过多,经常会有如下场景:

TextView title = findViewById(R.id.*tv_title*);
TextView title2 = findViewById(R.id.tv_title2);
TextView title3 = findViewById(R.id.tv_title3);
TextView title4 = findViewById(R.id.tv_title4);
TextView title5 = findViewById(R.id.tv_title5);
TextView title6 = findViewById(R.id.tv_title6);
TextView title7 = findViewById(R.id.tv_title7);
TextView title8 = findViewById(R.id.tv_title8);
TextView title9 = findViewById(R.id.tv_title28);
TextView title10 = findViewById(R.id.tv_title9);
TextView title11 = findViewById(R.id.tv_title10);
TextView title12 = findViewById(R.id.tv_title11);
TextView title13 = findViewById(R.id.tv_title12);
TextView title14 = findViewById(R.id.tv_title13);
TextView title15 = findViewById(R.id.tv_title14);
TextView title16 = findViewById(R.id.tv_title15);
TextView title17 = findViewById(R.id.tv_title16);
TextView title18 = findViewById(R.id.tv_title17);
TextView title19 = findViewById(R.id.tv_title18);...

数量一多,你会发现,这已经极其不友好。其实不光是不友好有问题,了解findViewById的原理后,你也会发现其内部实现在一定情况下对整体性能有轻微影响。

findViewById() 的原理

findViewById()的流程原理其实非常简单,以activity中的findViewById流程为例,activity要么继承自android.app.Activity,要么继承自androidx.appcompat.app.AppCompatActivity(你要是没适配AndroidX的话那就是support包)。这其中:

1⃣️、android.app.Activity继承类会通过getWindow得到Window对象来调用findViewById();

@Nullable
public <T extends View> T findViewById(@IdRes int id) {return getWindow().findViewById(id);
}

2⃣️、androidx.appcompat.app.AppCompatActivity继承类会通过getDelegate()得到AppCompatDelegate委派类的实例对象后调用其findViewByid(),这个对象实际是AppCompatDelegateImpl对象,创建其时传入了activity.getWindow得到的window对象。

@SuppressWarnings("TypeParameterUnusedInFormals")
@Override
public <T extends View> T findViewById(@IdRes int id) {return getDelegate().findViewById(id);
}

1⃣️和2⃣️最后都会调用Window(getWindow)里的findViewById()。

@Nullable
public <T extends View> T findViewById(@IdRes int id) {return getDecorView().findViewById(id);
}

Window类中通过getDecorView()来得到View对象(实际上是一个ViewGroup对象),

@Nullable
public final <T extends View> T findViewById(@IdRes int id) {if (id == NO_ID) {return null;}return findViewTraversal(id);
}

通过调用findViewById()来调用ViewGroup中重写的findViewTraversal()。下面源码图片是通过在线浏览网站获取到的ViewGroup类中findViewTraversal()的相关实现:

image.gif
可以看到这就是个遍历方法,如果对应界面内子元素是个View,只要id配对上可以直接返回,如果是一个ViewGroup则会调用子ViewGroup或子View的这个方法,依次遍历,直到找到目标id。很明显这是个深度优先搜索,时间复杂度为O(n)。

相关替代方案

1、 ButterKnife

大名鼎鼎的黄油刀,使用方式极为简便,项目地址:

https://github.com/JakeWharton/butterknife

在gradle中依赖:

implementation 'com.jakewharton:butterknife:xxx'
annotationProcessor 'com.jakewharton:butterknife-compiler:xxx'

对应在activity中操作为(具体一键生成方式这里不表):

public class MainActivity extends AppCompatActivity {@BindView(R.id.tv_title1)TextView tvTitle1;@BindView(R.id.tv_title2)TextView tvTitle1;@BindView(R.id.tv_title3)TextView tvTitle3;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);//绑定处理ButterKnife.bind(this);}
......
}

其实ButterKnife只是通过注解对findViewById的进行的一个取代,增加代码的可读性,而findViewById的各种缺点依然存在。当然,就这个开源框架而言,功能绝不仅仅是替代findViewById()。自从kt语言出来后,黄油刀的功效捉襟见肘,非Java版的老工程,不推荐。

2、kotlin-android-extensions

如果你项目可以使用kotlin,则可以使用kotlin-android-extensions。

在module的gradle中加入:

plugins {id 'kotlin-android-extensions'
}

可直接在对应类中通过其id的形式控制其控件,相当于已经获取了一遍控件id。

import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import com.test.demo.main.activity_main.*class MainActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)setContentView(R.layout.activity_main)tv_hello.setOnClickListener {}}
}

这种方法其本质上依旧还是通过findViewById去实现,有兴趣的小伙伴可以反编译看看。虽然现在此法已不被官方推荐,但其便利性还是首屈一指。

3、ViewBinding

一曲新人笑,几度旧人哭。此法一出,kotlin-android-extensions已不被官方推荐。

image.png
(注意,此法只能在AndroidStudio3.6及更高版本上可用)

使用方式:在build.gradle中依赖:

android {...buildFeatures {viewBinding true}
}

reload后,系统会为每个layout目录下 XML 布局文件生成一个绑定类。每个绑定类均包含对根视图以及具有 ID 的所有视图的引用。系统会通过以下方式生成绑定类的名称:将 XML 文件的名称转换为驼峰式大小写,并在末尾添加“Binding”一词。
例如:某个布局命名为activity_login,其所生成的绑定类的名称就为LoginActivityBinding,这个绑定类就会完成findViewById的工作。

不同布局会生成不同绑定类,他们所生成的路径在:在build/generated/data_binding_base_class_source_out/debug/out/com/xxx/yyy/databinding/目录下。

当然,如果不想xml文件生成 Binding 类,可以在 xml 布局文件中根 view 写入此属性:

tools:viewBindingIgnore="true"

其在代码中使用方式如下:(ViewBinding在Java和kotlin类中都可使用,这里仅是拿kotlin类举例)

· Activity:

@Override
protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);ActivityMainBinding binding = ActivityMainBinding.inflate(getLayoutInflater());setContentView(binding.getRoot());binding.tvTest.setText("This is ViewBinding");
}

可见,setContentView()中的参数改为了XXXbing.getroot()。调用布局中的某控件,只需要XXXbing.viewID(驼峰原则)可直接拿到实例对象(上述代码中的binding.tvTest控件在xml中的id为tv_test)。例如:xml中TextView控件id为tv_demo,则在activity中对应实例为XXXbing.tvDemo。

· Fragment中:

public class MyFragment extends Fragment {private FragmentMyBinding binding;private Context context;@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {binding = FragmentMyBinding.inflate(getLayoutInflater(), container, false);return binding.getRoot();}@Overridepublic void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {super.onViewCreated(view, savedInstanceState);binding.tvTitle.setText("Hello ViewBinding");}@Overridepublic void onDestroyView() {super.onDestroyView();binding = null;}
}

可以看出,跟Activity的引用方式区别不大,这里需要稍微注意Fragment 的存在时间比其视图长。在 Fragment对应onDestroyView()时要清除对绑定类实例的所有引用。

·RecyclerView adapter中:

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> {
、、、、、、@NonNull@Overridepublic ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {ItemLayoutBinding itemBinding = ItemLayoutBinding.inflate(inflater, parent, false);return new ViewHolder(itemBinding);}@Overridepublic void onBindViewHolder(@NonNull ViewHolder holder, int position) {holder.textView.setText(mData.get(position));}static class ViewHolder extends RecyclerView.ViewHolder {TextView textView;public ViewHolder(@NonNull ItemLayoutBinding itemBinding) {super(itemBinding.getRoot());textView = itemBinding.textView;}}
}

可见,应用方式无大致区别。

总结

1、findViewById兼容性好,适用所有场景,且灵活;

2、findViewById性能略差,底层就是个深度优先搜索,且id过多情况下容易造成可读性极差的情况,从上述的原理流程中不难看出,在Activity中调用findViewById,实际上是调用Window中的findViewById,但是Fragment中并没有单独的Window,Fragment中调用findViewById的效果和Activity中调用的效果一模一样。所以如果一个Activity中有多个Fragment,Fragment中的控件名称又有重复的,直接使用findViewById会爆错。

3、ButterKnife可一键生成,方便至极,但缺点跟findViewById一样。如果不是老工程,此法已不推荐使用。

4、Google官方表示,与使用 findViewById 相比,ViewBinding具有一些很显著的优点:

· 空指针安全:由于视图绑定(ViewBinding)会创建对视图的直接引用,因此不存在因视图 ID 无效而引发 Null 指针异常的风险。此外,如果视图仅出现在布局的某些配置中,则绑定类中包含其引用的字段会使用 @Nullable 标记。(说白了就是让你丫代码少爆空指针)

· 类型安全:每个绑定类中的字段均具有与它们在 XML 文件中引用的视图相匹配的类型。这意味着不存在发生类转换异常的风险。

这些差异意味着布局和代码之间的不兼容将会导致构建在编译时(而非运行时)失败。

下篇预告:第四种方式:DataBinding。

这篇关于还在用findViewById,不来了解下其它方式?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/698148

相关文章

gradle第三方Jar包依赖统一管理方式

《gradle第三方Jar包依赖统一管理方式》:本文主要介绍gradle第三方Jar包依赖统一管理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录背景实现1.顶层模块build.gradle添加依赖管理插件2.顶层模块build.gradle添加所有管理依赖包

Linux之systemV共享内存方式

《Linux之systemV共享内存方式》:本文主要介绍Linux之systemV共享内存方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、工作原理二、系统调用接口1、申请共享内存(一)key的获取(二)共享内存的申请2、将共享内存段连接到进程地址空间3、将

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

C#使用StackExchange.Redis实现分布式锁的两种方式介绍

《C#使用StackExchange.Redis实现分布式锁的两种方式介绍》分布式锁在集群的架构中发挥着重要的作用,:本文主要介绍C#使用StackExchange.Redis实现分布式锁的... 目录自定义分布式锁获取锁释放锁自动续期StackExchange.Redis分布式锁获取锁释放锁自动续期分布式

Java对象转换的实现方式汇总

《Java对象转换的实现方式汇总》:本文主要介绍Java对象转换的多种实现方式,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java对象转换的多种实现方式1. 手动映射(Manual Mapping)2. Builder模式3. 工具类辅助映

Spring Boot读取配置文件的五种方式小结

《SpringBoot读取配置文件的五种方式小结》SpringBoot提供了灵活多样的方式来读取配置文件,这篇文章为大家介绍了5种常见的读取方式,文中的示例代码简洁易懂,大家可以根据自己的需要进... 目录1. 配置文件位置与加载顺序2. 读取配置文件的方式汇总方式一:使用 @Value 注解读取配置方式二

一文详解Java异常处理你都了解哪些知识

《一文详解Java异常处理你都了解哪些知识》:本文主要介绍Java异常处理的相关资料,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现,文中通过代码介绍的非常详细,需要的朋... 目录前言一、什么是异常二、异常的分类2.1 受检异常2.2 非受检异常三、异常处理的语法3.1 try-

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

C# foreach 循环中获取索引的实现方式

《C#foreach循环中获取索引的实现方式》:本文主要介绍C#foreach循环中获取索引的实现方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、手动维护索引变量二、LINQ Select + 元组解构三、扩展方法封装索引四、使用 for 循环替代

将Java程序打包成EXE文件的实现方式

《将Java程序打包成EXE文件的实现方式》:本文主要介绍将Java程序打包成EXE文件的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录如何将Java程序编程打包成EXE文件1.准备Java程序2.生成JAR包3.选择并安装打包工具4.配置Launch4