Java内部类之间的闭包和回调详解

前言

闭包(closure)是一个可调用的对象,它记录了一些信息,这些信息来自于创建它的作用域。通过这个定义,可以看出内部类是面向对象的闭包,因为它不仅包含外围类对象(创建内部类的作用域)的信息,还自动拥有一个指向此外围类对象的引用,在此作用城内,内部类有权操作所有的成员,包括private成员。

Java最引人争议的问题之一就是,人们认为Java应该包含某种类似指针的机制,以允许回调(callback)。通过回调,对象能够携带一些信息,这些信息允许它在稍后的某个时刻调用初始的对象。如果回调是通过指针实现的,那么就只能寄希望于程序员不会误用该指针。

一、成员内部类

可以把一个内部类看做一个成员。成员内部类可以无条件访问外部类的所有成员属性和成员方法。

?


1

2

3

4

5

6

7

8

9

10

11

12

13

class OutterClass {//外部类

 private int in = 0;

 static int inn=4;

 public OutterClass(int in) {

   this.in = in;

 }

 class InnerClass {   //内部类

   public void output() {

     System.out.println(in);

     System.out.println(inn);

   }

 }

}

当成员内部类拥有和外部类同名的成员变量或者方法时,默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:

?


1

OutterClass(外部类).this.成员

外部类访问内部类,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问。

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

class OutterClass {

 private int in = 0;

 static int inn=4;

 public OutterClass(int in) {

   InnerClass inner=new InnerClass();

   this.in=inner.innerNum;

 }

 class InnerClass {  //内部类

   public int innerNum=1;

   public void output() {

     System.out.println(in);

     System.out.println(inn);

     int a=OutterClass.this.inn;

   }

 }

}

成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。创建成员内部类对象的一般方式如下:

?


1

2

3

4

5

6

public class classa {

 public static void main(){

   OutterClass oc=new OutterClass(3);

   OutterClass.InnerClass in=oc.new InnerClass();

 }

}

二、局部内部类

局部内部类就像是方法里面的一个局部变量一样,是不能有publicprotectedprivate以及static修饰符的。

?


1

2

3

4

5

6

7

class OutterClass {

 public OutterClass(int in) {

   class InnerClass {  //局部内部类

      int innerNum=1;

   }

 }

}

三、嵌套内部类

嵌套内部类,就是修饰为static的内部类。声明为static的内部类,不需要内部类对象和外部类对象之间的联系,就是说我们可以直接引用outer.inner,即不需要创建外部类,也不需要创建内部类。

?


1

2

3

4

5

6

7

8

9

10

11

12

class OutterClass {

 public OutterClass(int in) {   

 }

 static class InnerClass {  //局部内部类

   int innerNum=1;

 }

}

public class classa {

 public static void main(){

   OutterClass.InnerClass in=new OutterClass.InnerClass();

 }

}

四、匿名内部类

匿名内部类是我们使用最多的,因为我们并不想给它赋予名字,于是就有了匿名。匿名内部类需要提前定义的。

?


1

2

3

4

5

btnSan.setOnClickListener(newOnClickListener() {

  @Override

  publicvoidonClick(View v) {

  }

});

五、闭包和回调

闭包(Closure)是一种能被调用的对象,它保存了创建它的作用域的信息。JAVA并不能显式地支持闭包,但是在JAVA中,闭包可以通过“接口+内部类”来实现。

例如:一个接口程序员和一个基类作家都有一个相同的方法work,相同的方法名,但是其含义完全不同,这时候就需要闭包。

?


1

2

3

4

5

6

class Writer {//作家基类

 void work(){};

}

interface programmer{//程序员接口

 void work();

}

闭包实现代码如下:

?


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

public class WriterProgrammer extends Writer {

 @Override

 public void work(){

   //写作

 }

 public void code(){

   //写代码

 }

 class ProgrammerInner implements programmer{

   @Override

   public void work(){

     code();

   }

 }

}

在子类中定义了遵循程序员接口规则的内部类,然后使用内部类实现程序员的work()方法回调code()方法,在子类中直接实现父类作家的work()方法。

六、内部类的作用

内部类可以很好的实现隐藏。

一般的非内部类,是不允许有 private protected权限的,但内部类可以

内部类拥有外围类的所有元素的访问权限

可是实现多重继承

可以避免修改接口而实现同一个类中两种同名方法的调用。

七、总结

以上就是这篇文章的全部内容,希望对大家学习或使用Java能有一定的帮助,如果有疑问大家可以留言交流。

时间: 2024-08-04 23:42:39

Java内部类之间的闭包和回调详解的相关文章

JavaScript闭包和回调详解

一.闭包 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 闭包有三个特性: 1.函数嵌套函数; 2.函数内部可以引用外部的参数和变量; 3.参数和变量不会被垃圾回收机制回收. 闭包是指有权访问另一个函数作用域中的变量的函数,创建闭包的最常见的方式就是在一个函数内创建另一个函数,通过另一个函数访问这个函数的局部变量.使用闭包有一个优点,也是它的缺点,就是可以把局部变量驻留在内存中,可以避免使用全局变量.全局变量在每个模块都可调用,这势必将是

解析activity之间数据传递方法的详解

转自:http://www.jb51.net/article/37227.htm 本篇文章是对activity之间数据传递的方法进行了详细的分析介绍,需要的朋友参考下 1  基于消息的通信机制 Intent--------boudle,extra用这种简单的形式,一般而言传递一些简单的类型是比较容易的,如int.string等详细介绍下Intent机制Intent包含两部分:1 目的[action]-------要去到哪里去2 内容[category.data]----------路上带些什么,

Java多线程编程中Future模式的详解<转>

Java多线程编程中,常用的多线程设计模式包括:Future模式.Master-Worker模式.Guarded Suspeionsion模式.不变模式和生产者-消费者模式等.这篇文章主要讲述Future模式,关于其他多线程设计模式的地址如下:关于其他多线程设计模式的地址如下:关于Master-Worker模式的详解: Java多线程编程中Master-Worker模式的详解关于Guarded Suspeionsion模式的详解: Java多线程编程中Guarded Suspeionsion模式

Java 多线程(六) synchronized关键字详解

Java 多线程(六) synchronized关键字详解 多线程的同步机制对资源进行加锁,使得在同一个时间,只有一个线程可以进行操作,同步用以解决多个线程同时访问时可能出现的问题. 同步机制可以使用synchronized关键字实现. 当synchronized关键字修饰一个方法的时候,该方法叫做同步方法. 当synchronized方法执行完或发生异常时,会自动释放锁. 下面通过一个例子来对synchronized关键字的用法进行解析. 1.是否使用synchronized关键字的不同 例子

Java知多少(54)断言详解

断言的概念 断言用于证明和测试程序的假设,比如“这里的值大于 5”.断言可以在运行时从代码中完全删除,所以对代码的运行速度没有影响. 断言的使用 断言有两种方法: 一种是 assert<<布尔表达式>> : 另一种是 assert<<布尔表达式>> :<<细节描述>>. 如果布尔表达式的值为false , 将抛出AssertionError 异常: 细节描述是AssertionError异常的描述文本使用 javac –source

Java的String和StringBuffer和StringBuilder详解

Java的String和StringBuffer和StringBuilder详解 作者:chszs,转载需注明.博客主页:http://blog.csdn.net/chszs 前言 最近发现团队成员在Java代码方面的质量不够高,准备写一些基础的文章,供大家参考. 一.定义 String是不可变字符序列. StringBuffer是可变的字符序列. StringBuilder也是可变的字符序列. 1.StringBuffer和StringBuilder的唯一区别 StringBuffer对象是线

Java知多少(41)泛型详解

我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值. 假如我们现在要定义一个类来表示坐标,要求坐标的数据类型可以是整数.小数和字符串,例如: 1 x = 10.y = 10 2 x = 12.88.y = 129.65 3 x = "东京180度".y = "北纬210度" 针对不同的数据类型,除了借助方法重载,还可以借助自动装箱和向上转型.我们知道,基本数据类型可以自动装箱,被转换成对应的包装类:Object 是所有

java定时任务类Timer和TimerTask用法详解

原文:java定时任务类Timer和TimerTask用法详解 代码下载地址:http://www.zuidaima.com/share/1550463277550592.htm package com.zuidaima.util; import java.io.IOException; import java.util.Timer; import test.MyTask; /* * 本类给出了使用Timer和TimerTaske的主要方法,其中包括定制任务,添加任务 * 退出任务,退出定时器.

Java中堆内存和栈内存详解2

Java中堆内存和栈内存详解 Java把内存分成两种,一种叫做栈内存,一种叫做堆内存 在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配.当在一段代码块中定义一个变量时,java就在栈中为这个变量分配内存空间,当超过变量的作用域后,java会自动释放掉为该变量分配的内存空间,该内存空间可以立刻被另作他用. 堆内存用于存放由new创建的对象和数组.在堆中分配的内存,由java虚拟机自动垃圾回收器来管理.在堆中产生了一个数组或者对象后,还可以在栈中定义一个特殊的变量,这个变量的