热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android的回调事件详解

这篇文章主要介绍了Android的回调事件的相关资料,相当的详细,有需要的小伙伴可以参考下

看见网上一些回调的解释都很复杂的,特别基于Android的自定义回调,感觉一头雾水,于是乎,我也写了这篇基于我对回调的解释。

先来看一个简单的例子:
有两个类 ClassA ,和 ClassB, ClassA调用ClassB里面的方法,

public class ClassB {

  public void method_from_classB(){
    
    for(int i=0;i<10;i++)
      System.out.print("..."+i);
  }
}
public class ClassA {

   
  
  public static void main(String args[]){
   
    ClassB classB = new ClassB();
    
    classB.method_from_classB();
  }
}

输出:

...0...1...2...3...4...5...6...7...8...9

卧槽,哪个傻逼写的博文,侮辱我的智商不是吗,嘻嘻,是为了做比较,接下来看看利用回调, ClassA 是怎么调用 ClassB中的 方法的,注意是回调:

让ClassB 实现 ClassA定义的接口

public class ClassB implements ClassA.ClassAInterface{

  public ClassB(){
    
    new ClassA().RegisterInterface(this);
    System.out.println("...ClassB..."+this);
  }

  @Override
  public void method_from_interface() {

    for(int i=0;i<10;i++)
      System.out.print("..."+i);
  }
  
  /*  public void method_from_classB(){
  
  for(int i=0;i<10;i++)
    System.out.print("..."+i);
  }*/
}

ClassA里面定义接口和抽象方法:

public class ClassA {

  public static ClassAInterface classAInterface;
  
  public interface ClassAInterface{
    
    public void method_from_interface();
  }
  public void RegisterInterface(ClassAInterface a_interface){

    this.classAInterface = a_interface;
    System.out.println("...a_interface..."+a_interface);
  }
 
  public static void main(String args[]){
   
    ClassB classB = new ClassB();// 标记@1,最后面做解释
    //classB.method_from_classB();
    System.out.println("...classAInterface..."+classAInterface);
     if(classAInterface != null){
        classAInterface.method_from_interface();
      }
  }
}

输出:

...0...1...2...3...4...5...6...7...8...9

整理下,也就是 我在ClassA里面定义了一个接口(interface),接口里面又定义了一个方法,但没有方法体,也就不做任何事情。

当 ClassA 执行到 mian() 函数时,就会调用接口的方法,但前面讲了,接口的方法没有实现具体的事情,它就会找到 ClassB 里面对应的 方法,来实现具体的事情。

呦呦呦,ClassA 的接口的方法是怎么找到 ClassB 的方法,难道会上天???

也就是下面分析这句代码是怎么上天的:

// 利用接口的回调实现 ClassB中 的方法的 具体事情

 classAInterface.method_from_interface();

我在上面的代码中用 System.out.println 打印出了日志做分析:

第一个(ClassA中的方法):

  public void RegisterInterface(ClassAInterface a_interface){

    this.classAInterface = a_interface;
    System.out.println("...a_interface..."+a_interface);
  }

输出:

...a_interface...ClassB@3ddb8962

第二个:

public ClassB(){
    
    new ClassA().RegisterInterface(this);
    System.out.println("...ClassB..."+this);
  }

输出:

...ClassB...ClassB@3ddb8962

第三个:

System.out.println("...classAInterface..."+classAInterface);
     if(classAInterface != null){
        classAInterface.method_from_interface();
      }

输出:

...classAInterface...ClassB@3ddb8962

看到这里是不是恍然大悟呢 ,输出都是 “ ClassB@3ddb8962 ” 也就是ClassB 对象的引用!!!

啊!接口只不过是将 ClassB 对象的引用 传到 ClassA中而已,那这句会上天的语句是不是很好解释了呢。

       classAInterface.method_from_interface();

相当于  ClassB@3ddb8962.method_from_interface();

这是不是跟最上面到的代码:

    ClassB classB = new ClassB();  
    classB.method_from_classB();

一样呢,这也是为什么我最开始要举这个例子的原因!!!

相信看到这里应该理解了接口的回调是怎么回事了吧。
但有一点又糊涂了,为什么 要接口回调这么麻烦的,最上面的在ClassA里面执行:

    ClassB classB = new ClassB();  
    classB.method_from_classB();

不是照样可以 ClassA 调用 ClassB 里面的 方法。。。。但要是ClassA 要调用ClassC,ClassD ...,里面的方法呢,是不是还要改变ClassA里面的代码,实例化ClassC,ClassD ... 的对象,显然是不好的,要是使用接口那就不用改变ClassA 里面的代码了,任何类只要实现ClassA 里面的接口就可以.

解释一下 标记@1 :

上面那段话好像跟 标记@1 违背了,在 ClassA 里面确实也需要实例化 ClassB对象。

因为要 【利用】 初始化的时候执行构造方法里面的代码:

public ClassB(){
    // 相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了
    new ClassA().RegisterInterface(this);
  }

将this 传递给 ClassA ,作用也就是 上面利用 日志分析的作用。

但再 Android 开发中救你不必这样了,
可以在 Activity 的初始化时执行:

@Override
  protected void onCreate(Bundle savedInstanceState) {

     // 相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了
    new ClassA().RegisterInterface(this);
}

在Android 开发中 ClassA 里面的 mian() 函数可以用事件来代替,触发:
如:

button.setOnClickListener(new OnClickListener() {
      
      @Override
      public void onClick(View v) {
        // TODO 自动生成的方法存根
        if(classAInterface != null){
        classAInterface.method_from_interface();
      }
    });


推荐阅读
  • 本文深入探讨了 Java 编程语言的基础,特别是其跨平台特性和 JVM 的工作原理。通过介绍 Java 的发展历史和生态系统,帮助初学者理解如何编写并运行第一个 Java 程序。 ... [详细]
  • MATLAB实现n条线段交点计算
    本文介绍了一种通过逐对比较线段来求解交点的简单算法。此外,还提到了一种基于排序的方法,但该方法较为复杂,尚未完全理解。文中详细描述了如何根据线段端点求交点,并判断交点是否在线段上。 ... [详细]
  • 本文深入探讨了 com.example.android.sunshine.data.TestUtilities 中 validateThenCloseCursor() 方法的使用方法及其代码示例,旨在帮助开发者更好地理解和应用该方法。 ... [详细]
  • 当在 Android 应用中使用 NDK 时,可能会遇到 java.lang.UnsatisfiedLinkError: Native method not found 的错误。本文将详细探讨该错误的原因及解决方案。 ... [详细]
  • 本文介绍如何使用Python进行文本处理,包括分词和生成词云图。通过整合多个文本文件、去除停用词并生成词云图,展示文本数据的可视化分析方法。 ... [详细]
  • 本文介绍如何在Linux服务器之间使用SCP命令进行文件传输。SCP(Secure Copy Protocol)是一种基于SSH的安全文件传输协议,支持从远程机器复制文件到本地服务器或反之。示例包括从192.168.45.147复制tomcat目录到本地/home路径。 ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • 本文详细介绍了如何在CentOS 7操作系统上安装和配置Grafana,包括必要的依赖项安装、插件管理以及服务启动等步骤。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本文详细介绍了Git分布式版本控制系统中远程仓库的概念和操作方法。通过具体案例,帮助读者更好地理解和掌握如何高效管理代码库。 ... [详细]
  • 使用GDI的一些AIP函数我们可以轻易的绘制出简 ... [详细]
  • 本题探讨如何通过最大流算法解决农场排水系统的设计问题。题目要求计算从水源点到汇合点的最大水流速率,使用经典的EK(Edmonds-Karp)和Dinic算法进行求解。 ... [详细]
  • 本文介绍了如何通过配置 Android Studio 和 Gradle 来显著提高构建性能,涵盖内存分配优化、并行构建和性能分析等实用技巧。 ... [详细]
  • 本文介绍如何通过SSH协议使用Xshell远程连接到Ubuntu系统。为了实现这一目标,需要确保Ubuntu系统已安装并配置好SSH服务器,并保证网络连通性。 ... [详细]
  • 落樱3D v0.5是一款在Android平台上发布的3D美少女格斗游戏,本次更新带来了多项新功能和优化。 ... [详细]
author-avatar
奋怒的小超_656
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有