嵌套类

嵌套类有两种类别:static and non-static,分别对应为静态嵌套类和内部类。

 class OuterClass {
...
static class StaticNestedClass {
...
}
class InnerClass {
...
}
}

其中静态嵌套类只能访问外部类的静态成员,内部类可以访问外部类的任意成员;它们可以被声明为privatepublicprotected, 或 package private。

  • 静态嵌套类实例化方式为: OuterClass.StaticNestedClass nestedObject = new OuterClass.StaticNestedClass();
  • 内部类实例化方式:OuterClass.InnerClass innerObject = outerObject.new InnerClass(); 即通过外部类实例才能访问内部类。

有两个比较特殊的内部类,分别为局部内部类和匿名类。

局部内部类

  • 局部内部类(Local CLasses)可声明在类中任意块(block)中,如方法、for或if块中
  • 局部内部类可以访问外部类的成员,若局部内部类声明在静态块中,则可访问外部类的静态成员;若声明在非静态块中,则可访问外部类所有成员;
  • 局部内部类可以访问所在块的局部变量,但该局部变量必须声明为final;在JDK8中进行了改进,局部变量可以声明为final或effectively final;
  • 其他特性类似于普通内部类

其中effectively final与final局部变量的区别在于,前者可以不显式声明变量为final,只要在整个过程中,该变量不会被修改(编译器默认该情况为final)。具体为什么局部内部类为什么必须引用final变量,可参考

java为什么匿名内部类的参数引用时final? 。大致意思是局部内部类引用局部变量,其实是进行的值引用(或者说是值拷贝)。可以认为避免外部代码块在内部类运行结束前结束,导致局部变量回收而出错。

匿名类

匿名类与局部内部类相似,只是没有命名,并且同时进行声明和实例化。如下:

 HelloWorld frenchGreeting = new HelloWorld() {
String name = "tout le monde";
public void greet() {
greetSomeone("tout le monde");
}
public void greetSomeone(String someone) {
name = someone;
System.out.println("Salut " + name);
}
};

匿名内部类适用于只用一次的情况。其他的特性与局部内部类相同。

Lambda表达式

 在使用匿名内部类的时候,无需提供类名。对于只有一个方法的接口,使用Lambda显然比匿名类的实现简单明了。如下所示,定义一个LambdaTest接口,该接口只包含一个opt方法:
 interface LambdaTest {
int opt(int a , int b);
} LambdaTest sumTest = (a,b) -> a+b;

第5行即为Lambda表达式声明,其中(a,b)为方法的参数,a+b为方法体,->表示将参数传递给方法体。

  • Lambda表达式的方法体中,可以是一个表达式,也可以是代码块。若为表达式,Java运行期会计算表达式,并返回结果;若为代码块,可以添加return语句,将结果返回。
  • Lambda表达式其实是一个方法的声明,可以认为Lambda表达式是匿名方法
  • Lambda表达式与局部内部类和匿名类相似,可以访问外部类和外部代码块的变量;但与后两者不同,其不存在变量覆盖的问题,可以认为没有引入新的代码块,其与外部代码块中的局部变量同级
  • 由于第三条,所以在表达式的参数中,不能声明与同级作用域相同的变量名,否则会出现重复定义的异常。
  • Lambda表达式是匿名内部类实现形式的一种,其访问的外部变量必须是final或effectively final。

举例如下:

 public class Lambda {

     private int var = 100;
private String x = "hello"; interface Cal{
int op(int a, int b);
} interface Print{
void print(String msg);
} public int operator(int a, int b, Cal cal) {
return cal.op(a, b);
} public void operator1(String msg, Print print) {
print.print(msg);
} public void operator2(String x) { // x = ""; Print print = (msg) -> {
System.out.println("Lambda访问外部变量:");
System.out.println(x);
System.out.println(msg);
System.out.println(Lambda.this.x);
}; print.print(x);
} public static void main(String[] args) {
Cal add = (a,b) -> {return a+b;};
Cal mul = (a,b) -> a*b; Lambda lambda = new Lambda();
System.out.println("2+3="+lambda.operator(2, 3, add));
System.out.println("2*3="+lambda.operator(2, 3, mul)); lambda.var = 200;
Print print = (msg) -> {
System.out.println(msg);
System.out.println(lambda.var);
};
lambda.operator1("Hello World", print); lambda.operator2("Hello Lambda");
} }

运行结果:

 2+3=5
2*3=6
Hello World
200
Lambda访问外部变量:
Hello Lambda
Hello Lambda
hello

其中operator2方法可以验证后三条,如果将24行的注释取消,28行就会报“local variables referenced from a lambda expression must be final or effectively final”的异常。

目标类型(Target Type)

目标类型为外部类方法期望调用的类型,如上例中operator期望调用的目标方法为Cal。Java会根据Lambda表达式所处的语境和上下文信息判断目标类型,并实现调用。

举例如下:
 public class TargetType {

     interface Cal{
String op();
} interface Cal1{
int op1();
} interface Cal2{
void op1();
} public static String invoke(Cal cal) {
return cal.op();
} public static void invoke(Cal1 cal1) {
cal1.op1();
} public static void invoke(Cal2 cal2) {
cal2.op1();
} public static void main(String[] args) {
invoke(() -> "done");
invoke(() -> 100);
invoke(() -> {return;});
}
}

声明三个接口(Cal Cal1 Cal2),具有相同名称的方法,但他们的返回值不同。另声明了3个invoke方法,分别接收3个类,即期望的目标类型不同。然后进行测试:

main方法中的三个语句都通过编译,并且eclipse提示28行调用目标类型为Cal的invoke,29行调用目标类型为Cal1的invoke,30行调用目标类型为Cal2的invoke,目标类型如下图所示:

(1)如果再添加一句如:invoke(() -> 100.0);  则编译器会报错,Type mismatch: cannot convert from double to String;

(2)如果将Cal接口方法的返回值改为int,则除了28行报错,29行也报错:The method invoke(TargetType.Cal) is ambiguous for the type TargetType,即编译器无法确定调用哪个目标类型。

 官网文档中举的例子为Runnable和Callable,原理一样,如下:
 public interface Runnable {
void run();
} public interface Callable<V> {
V call();
}

方法声明:

 void invoke(Runnable r) {
r.run();
} <T> T invoke(Callable<T> c) {
return c.call();
}
 根据上下文确定目标类型,由于有返回值,所以会调用参数为Callable的invoke方法:

 String s = invoke(() -> "done");

总结:

  • 静态嵌套类与内部类区别
  • 两类特殊的内部类,局部内部类和匿名内部类;
  • 匿名内部类的特殊实现:Lambda表达式,可认为匿名方法的实现;
  • Lambda表达式会根据上下文环境确定目标类型

参考:

 

Java学习笔记-嵌套类的更多相关文章

  1. Java学习笔记——File类之文件管理和读写操作、下载图片

    Java学习笔记——File类之文件管理和读写操作.下载图片 File类的总结: 1.文件和文件夹的创建 2.文件的读取 3.文件的写入 4.文件的复制(字符流.字节流.处理流) 5.以图片地址下载图 ...

  2. Java学习笔记之---类和对象

    Java学习笔记之---类和对象 (一)类 类是一个模板,它描述一类对象的行为和状态  例如:动物类是一个类,动物们都有属性:颜色,动物们都有行为:吃饭 public class Dog { Stri ...

  3. Java7编程 高级进阶学习笔记--嵌套类

    定义: 在一个类中定义的类叫做嵌套类. 作用: 1.允许对相关类进行逻辑分组 2.增强了代码的封装性 3.使代码具有更强的可读性和维护性 使用方式: package com.cmz.baseTest; ...

  4. Java学习笔记-File类的基本方法

    要渐渐养成写博客的习惯-----> 前段时间看Mars的java中的I/O流没怎么懂,发现I/O流好难啊.今天重新看一遍其他教学,还有书籍,做些笔记,记录下每天的学习生活. File类的一些方法 ...

  5. Java学习笔记 04 类和对象

    一.类和对象的概念 类 >>具有相同属性和行为的一类实体 对象 >>实物存在的实体.通常会将对象划分为两个部分,即静态部分和动态部分.静态部分指的是不能动的部分,被称为属性,任 ...

  6. Java学习笔记——SequenceInputStream类合并文件的综合举例分析

    SequenceInputStream 介绍 SequenceInputStream 类表示其他输入流的逻辑串联,即文件的合并. 它从输入流的有序集合开始,并从第一个输入流开始读取,直到到达文件末尾, ...

  7. 0018 Java学习笔记-面向对象-类的基本要素

    类与对象 大街上一个个的人,就是一个个对象 类是对一群对象的抽象,比如人都有性别.年龄.姓名,都会吃饭.睡觉等.姓名性别可以抽象为变量,吃饭睡觉可以抽象为方法,像下面一样定义个类来形容人 public ...

  8. Java学习笔记7---父类构造方法有无参数对子类的影响

    子类不继承父类的构造方法,但父类的构造方法对子类构造方法的创建有影响.具体来说就是: ①.当父类没有无参构造方法时,子类也不能有无参构造方法:且必须在子类构造方法中显式以super(参数)的形式调用父 ...

  9. Java学习笔记之——类与对象

    1.参数的传递方式 1)值传递 2)引用传递 2.类和对象: (1)类的定义: public class 类名{ 类型 属性1: 类型 属性2: ……… public 返回值类型 方法名1(形参){ ...

随机推荐

  1. Code Kata:螺旋矩阵 javascript实现

    1 2 3 4  5 16 17 18 19 6 15 24 25 20 7 14 23 22 21 8 13 12 11 10 9  如图所示,就是一个5*5的螺旋矩阵 我的思路如下: 第一步:拆分 ...

  2. constructor.prototype

    一个很好玩的小问题考大家对js的理解function alert (){}; ________________ // 填空 alert(1); 使1弹出  http://perfectionkills ...

  3. 修改 docker image 安装目录 (解决加载大image时报错:"no space left on device")

    修改 docker image 安装目录 (解决加载大image时报错:"no space left on device" ) 基于Ubuntu16.04 docker版本: 17 ...

  4. android动画介绍之 自己定义Animation动画实现qq抖一抖效果

    昨天我们介绍了Animation的基本使用方法.小伙伴们了解的怎么样了?假设还没有了解过Animation的小伙伴能够看看这篇博客 android动画介绍--Animation 实现loading动画 ...

  5. Android PopupWindows

    今天了解到PopupWindows这个布局,PopupWindow这个类用来实现一个弹出框,能够使用随意布局的View作为其内容,这个弹出框是悬浮在当前activity之上的. 以下是一个实例 xml ...

  6. C语言之——文件操作模式

    转载请注明出处:http://blog.csdn.net/l1028386804/article/details/47450667 "rt" 仅仅读打开一个文本文件.仅仅同意读数据 ...

  7. 注册Azure AD 2.0 应用程序

    作者:陈希章 发表于 2017年3月22日 上一篇 介绍了Microsoft Graph应用程序的一些概念,以及目前还比较普遍的Azure AD 1.0应用程序的注册方式.但正如我多次提到的那样,虽然 ...

  8. ASP.NET Core 使用 Hangfire 定时任务

    定时任务组件,除了 Hangfire 外,还有一个 Quarz.NET,不过 Hangfire .NET Core 支持的会更好些. ASP.NET Core 使用 Hangfire 很简单,首先,N ...

  9. TableML-GUI篇(Excel编译/解析工具)

    项目情况 本文接上篇TableML Excel编译/解析工具,本文主要介绍GUI工具的使用,及配置项,如果你想了解此工具更加详细的说明,请阅读上篇文章. 项目地址:https://github.com ...

  10. 自学Zabbix3.10.1.2-事件通知Notifications upon events-媒介类型SMS

    自学Zabbix3.10.1.2-事件通知Notifications upon events-媒介类型SMS 服务器安装串口GSM短信猫之后,zabbix可以使用它来发送短信通知给管理员,如下注意事项 ...