通过点击事件监听 setOnClickListener 彻底理解回调-Android

2023-05-16

前言

老司机们对于回调肯定熟悉得不能再熟悉了,但是新司机可能还是一脸懵逼的,我比较笨,当年懵逼了好久,看夏安明的这一篇博客地址,虽然下边的留言都是,写得好!懂了懂了!但是我当时看了三遍还是不懂好吗 - -,现在我站在我的角度,用我理解的方式给大家讲解回调,我这么笨都理解了,聪明的新司机们肯定也是可以的

setOnClickListener 分析

setOnCLickLinstener,只要写过 Android 的同学应该都见过,大家都知道是点击事件监听,但是是怎么实现的呢?对,你没有猜错,就是回调
你在 onClick(View view)中写的方法,就是一个回调方法,你仔细想一想,这个方法是在你传的参数 new View.OnClickListener()中的方法,你再仔细的想一想,为什么你传入了 new View.OnClickListener()这个参数,Android Studio 就会自动补全,让你去实现 onClick(View view)这个方法呢? 一切都在你想象之中,OnClickListener 就是一个接口,new 出一个接口,你就得实现他里边的抽象方法,在 Android 中,大多数回调都是靠接口来进行的
并且,你实现了 onClick(View view)方法后,这个方法并没有在我们的 Activity 或者 Fragment 中调用,那为什么他生效了呢?这就是回调,你实现了他,而他却是在另一个地方调用的
那是在什么地方调用的呢?

我们点进 setOnClickListener 方法中一探虚实

于是我们跳到了 View.java,原来这个方法是写在 View 中的,这时你想到,第一行代码中说了,我们的控件都继承于 View,原来如此

    public void setOnClickListener(@Nullable OnClickListener l) {
        if (!isClickable()) {
            setClickable(true);
        }
        getListenerInfo().mOnClickListener = l;
    }

setOnClickListener 方法就如同我们调用时的那样,传入一个 OnClickListener 对象作为参数,那我们来看一看 OnClickListener 是个啥子

    public interface OnClickListener {
        /**
         * Called when a view has been clicked.
         *
         * @param v The view that was clicked.
         */
        void onClick(View v);
    }

果然不出你所料,就是个 interface

然后注意这一行

getListenerInfo().mOnClickListener = l;

把我们传入的 OnClickListener 对象赋值给了 getListenerInfo().mOnClickListener,记住我们传入的 OnClickListener 对象就相当于携带了我们实现的 onClick(View view)方法,进到 View 里边来了

记好了哦!

我们来看看 getListenerInfo()方法

 ListenerInfo getListenerInfo() {
        if (mListenerInfo != null) {
            return mListenerInfo;
        }
        mListenerInfo = new ListenerInfo();
        return mListenerInfo;
    }

getListenerInfo()返回一个 ListenerInfo,如果 mListenerInfo 已经存在,就返回,如果不存在,就 new 一个返回,也许你已经知道,或许不久后你就知道,这叫单例模式,保证只有一个 ListenerInfo 对象

然后我们来看看 ListenerInfo 又是个啥子

static class ListenerInfo {
        protected OnFocusChangeListener mOnFocusChangeListener;
        private ArrayList<OnLayoutChangeListener> mOnLayoutChangeListeners;
        protected OnScrollChangeListener mOnScrollChangeListener;
        private CopyOnWriteArrayList<OnAttachStateChangeListener> mOnAttachStateChangeListeners;
        public OnClickListener mOnClickListener;
         protected OnLongClickListener mOnLongClickListener;
        protected OnContextClickListener mOnContextClickListener;
        protected OnCreateContextMenuListener mOnCreateContextMenuListener;
        private OnKeyListener mOnKeyListener;
        private OnTouchListener mOnTouchListener;
        private OnHoverListener mOnHoverListener;
        private OnGenericMotionListener mOnGenericMotionListener;
        private OnDragListener mOnDragListener;
        private OnSystemUiVisibilityChangeListener mOnSystemUiVisibilityChangeListener;
        OnApplyWindowInsetsListener mOnApplyWindowInsetsListener;
    }

原来是一个内部静态类,成员包括各种事件的监听接口,其中包括

public OnClickListener mOnClickListener;

诶哟,和我们传入的一样的一个 OnClickListener 接口引用,于是绕了这么一大圈(我们先不管为啥绕),我们传入的持有我们实现的 onClick(View view)方法的 OnClickListener 接口对象(还记得吗?),被赋值到了 View 中的 mListenerInfo 中的 mOnClickListener 对象,也就是,我们实现的 onCLick(View view) 方法,被 mListenerInfo.mOnClickListener 持有了

这时,你应该想到了,我们实现的 onClick(View view)应该就是在 View 中被调用了,bingo !

    public boolean performClick() {
        final boolean result;
        final ListenerInfo li = mListenerInfo;
        if (li != null && li.mOnClickListener != null) {
            playSoundEffect(SoundEffectConstants.CLICK);
            li.mOnClickListener.onClick(this);
            result = true;
        } else {
            result = false;
        }

        sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
        return result;
    }

从字面意思理解,这个方法就是执行 Click 的方法, 他将 mListenerInfo 对象传给了一个静态的 ListenerInfo 对象, li,后边的故事大家都知道了

li.mOnClickListener.onClick(this);

这个方法执行了点击事件,并调用了我们实现的 onClick(View view) 方法

让我们来梳理一遍流程,我们在 Activity 或者 Fragment 中调用

View.setOnClickListener 方法,传入一个 OnCLickListener 对象,实现了 onCLick(View view)方法,然后在 View 中的某个地方,我们实现的 onCLick(View view)被调用,实现了回调,这就是回调的流程

异步

回调有什么用呢,就是异步,想象一下,系统一直在监听着屏幕的点击事件,在我们触摸到屏幕的时候进行响应,这是一个线程操作,因为如果这个放在主线程,那在事件被响应之前,我们的线程都是阻塞的,因为屏幕的资源被占用了,无法进行其他操作,而在子线程中,系统监听着屏幕的活动,然后在我们触摸时,调用 performClick()方法实现了点击,并且调用了 onClick(View view)方法实现了点击事件的回调,我们就可以恰恰刚好在点击时间触发的时候,进行我们想要的操作,也就是我们实现的 on CLick(View view)方法

半伪代码实现一个回调给你看

A.class

//先定义一个接口
public interface Listener {
    //回调方法
    void 回调方法();
}

//申明一个接口
private Listener mLinstener;

//一个 set 接口的方法
public void setListener(Listener listener) {
    //把传入的 listener 赋值给 mLinstener
    mLinstener = listener
}

... 
//在某个地方,进行某个操作的时候

private void 某个操作() {
    //回调方法执行
    mLinstener.回调方法();
}

另一个类 B.class

private A a = new A();

a.setListener(new Linstener() {
    public void 回调方法() {
        //我要在 A 中某个操作()执行的时候要搞的事情
        搞事情阿搞事情();
    }
});

然后在某个操作()调用的时候,我们的回调方法()也就被调用开始搞事情了

你如果看不懂的话,自己写一遍,这就是 Android 中回调的一般写法,你可以在各种自定义 View 中用来了,用着用着就理解了

为啥要绕那一圈

那一圈保证了 View 中只有一个 mOnClickListener 对象,保证了我们一次只执行一次 onClick() 方法

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

通过点击事件监听 setOnClickListener 彻底理解回调-Android 的相关文章

  • 汇编中的寻址方式

    存储器 存储器 xff08 Memory xff09 是现代信息技术中用于保存信息的记忆设备 其概念很广 xff0c 有很多层次 xff0c 在数字系统中 xff0c 只要能保存二进制数据的都可以是存储器 xff1b 在集成电路中 xff0
  • x86的32位汇编快速入门

    本文描述基本的32位X86汇编语言的一个子集 xff0c 其中涉及汇编语言的最核心部分 xff0c 包括寄存器结构 xff0c 数据表示 xff0c 基本的操作指令 xff08 包括数据传送指令 逻辑计算指令 算数运算指令 xff09 xf
  • 80x86汇编指令详解

    80x86指令系统 xff0c 指令按功能可分为以下七个部分 1 数据传送指令 2 算术运算指令 3 逻辑运算指令 4 串操作指令 5 控制转移指令 6 处理器控制指令 7 保护方式指令 3 3 1数据传送指令 数据传送指令包括 xff1a
  • 汇编中addr和offset

    汇编中addr和offset的异同点 xff01 xff01 xff01 一 相同点 1 addr 和 offset 操作符都是获得操作数的偏移地址 xff1b 2 addr 和 offset 的处理都是先检查处理的是全局还是局部变量 xf
  • 汇编语言,数据段中,标号前面加不加offset有什么区别?

    data segment string db 40h dup 0 string 1 db 39 Input characters 39 0dh 39 39 data ends 那么 xff0c mov bx string 和 mov bx
  • SD-WAN如何推动企业上云

    企业上云是指企业将其业务系统 应用程序 数据等资源迁移到云端 xff08 云服务提供商的数据中心 xff09 xff0c 以实现更高效 灵活 安全 可靠的信息化运营和服务 麦肯锡公司发布了名为 企业上云2 0 xff1a 加速数字化转型 x
  • nvm常用命令,配置镜像

    1 配置镜像 阿里云 nvm npm mirror https npmmirror com mirrors npm nvm node mirror https npmmirror com mirrors node 腾讯云 nvm npm m
  • 汇编语言中MOV和OFFSET指令的两个问题?

    xff08 1 xff09 往段寄存器送段地址时要写成 MOV AX DATA MOV DS AX 不能写成 MOV DS DATA 这是为什么 xff1f 有人说由于段寄存只能进行16位的读写 xff0c 因此需要用ax来倒一下 xff0
  • C++动态链接库的制作

    输入函式 declspec dllimport 与输出函式 declspec dllexport 有什么区别呢 xff1f 我知道他们不同 xff0c 但差别在哪呢 我用的全是 declspec dllexport xff0c declsp
  • C#调用C++函数来与串口通信

    前些日子帮朋友写个小软件 xff0c 要求用C 来实现主程序 xff0c 主要的功能是与一些通信设备打交道 xff0c 当然就是通过串口了 xff0c 以十进制发送和读取串口 的数据 xff0c 考虑到C 调用API并没有C 43 43 来
  • c++获取系统错误提示!getLastError()

    void showError LPVOID lpMsgBuf FormatMessage FORMAT MESSAGE ALLOCATE BUFFER FORMAT MESSAGE FROM SYSTEM FORMAT MESSAGE IG
  • c++串口操作

    0 前言 做串口方面的程序 xff0c 使用CreateFile打开串口通信端口 在对串口操作之前 xff0c 需要首先打开串口 使用C 43 43 进行串口编程 xff0c 如果采用VS开发 xff0c 则可以直接借助于串口通信控件来操作
  • vs环境下C++dll生成和使用(基础篇)

    动态库和静态库 xff1a 动态库 xff1a 全名动态链接库 xff0c 用于将你的函数封装 xff0c 让别人只能调用 xff0c 不能看你的实现代码 由引入库和dll组成 xff1a 引入库包含导出的函数和变量名 xff0c dll包
  • #ifndef.#define, #endif 的用法

    文件中的 ifndef define endif 很关键 xff0c 是为了避免多重包含 xff0c 比如如果两个C文件同时包含同一头文件 xff0c 那么就会出现问题 xff0c 所以使用这种方法可以有效避免这种情况 一般用法 xff1a
  • 关于.NET编译的目标平台(AnyCPU,x86,x64)

    在VisualStudio中项目平台属性包含x86 x64 AnyCPU三个选项 xff0c 之前的项目中并没有特别去关注这一点 xff0c 最近的项目中涉及到了在不同平台运行的问题 xff0c 所以专门了解并整理了这方面的知识 x86 x
  • IP地址不是唯一的吗?为什么路由器的IP地址都是这样的呢?

    路由器同时连接外部网络和内部网络 xff0c 外部网络的IP是都不一样 xff0c 但内部网络的IP xff0c 就没有问题 xff0c 但内部网络所有设备的IP也不能相同 域名是对你的网站存放主机的ip的解析 xff0c 可以理解为你的别
  • HTTP简介

    HTTP协议是Hyper Text Transfer Protocol xff08 超文本传输协议 xff09 的缩写 是用于从万维网 xff08 WWW World Wide Web xff09 服务器传输超文本到本地浏览器的传送协议 H
  • OMF和flash_recovery_area的关系!

    在OMF出现之前 9i R1 oracle db的文件 xff0c 主要指dbf xff0c redo xff0c ctl是通过os管理的 xff0c 为了简化对数据库文件的管理 xff0c oracle引入了OMF 通过omf创建的dbf
  • asp.net发布网站的详细步骤

    1 用VS2013打开解决方案 2 选中解决方案 xff0c 点击鼠标 右键 gt 从弹出对话框中 xff0c 选择 清理解决方案 3 待第2步 清理解决方案 结束后 xff0c 选中 解决方案 gt 点击鼠标 右键 gt 在弹出对话框中
  • 域名和IP地址是一回事吗?建网站要买域名还要买IP地址吗?

    去年我在网通花了150元买了一个域名 xff0c 他们在登记单上的域名费一栏填了150元 xff0c 登记单上还有IP使用费一栏他们什么也没有填写 xff0c 如果域名与IP地址是同一个项目 xff0c 是一回事 xff0c 他们不是在重复

随机推荐