内部类

内部类(inner class)是定义在另一个类中的类

内部类的好处:

|--1.隐藏机制:内部类封装性更好,隐藏在一个类之中,同一包中的其他类也不能访问

|--2.内部类可以访问外围类的私有数据

成员内部类

member inner class

可以访问外部类的所有成员变量和方法(无论静态、非静态)

自身没有静态成员

|--外部类里面创建成员内部类的实例:this.new Innerclass();

|--在外部类之外创建内部类的实例:new Outerclass().new Innerclass();

|--在内部类里访问外部类的成员:Outerclass.this.member

class Hero {
void callHorse() {
// 自己使用自己的成员内部类
System.out.print("吕布");
this.new Horse().show();
}
// 成员内部类
class Horse {
public void show() {
System.out.println("---赤兔---");
}
}
}
public class InnerClassMember {
public static void main(String[] args) {
Hero h = new Hero();
h.callHorse();
// 创建一个类的内部类
Hero.Horse horse = new Hero().new Horse();
horse.show();
}
}
吕布---赤兔---
---赤兔---

静态内部类

static inner class (nested class)

使用内部类只是为了把一个类隐藏在另外一个类的内部,并不需要内部类引用外围对象,可以将内部类声明为static。

生成静态内部类对象的方式为:

OuterClass.InnerClass inner = new OuterClass.InnerClass();// 里面的类不用new

|--成员内部类没有静态成员;

|--静态内部类只能访问外围类的静态成员

局部内部类

local inner class

在方法中定义、使用

不能被public, protected, private和static修饰(像局部变量一样)

只能访问方法中定义的final类型的局部变量(初始化之后便不能更改)

import java.util.*;
// 以集合排序需要实现Comparator接口为例
public class TestInnerClassLocal {
public static void main(String[] args) {
class InnerComparator implements Comparator<Integer> {
@Override
public int compare(Integer o1, Integer o2) {
return o2 - o1;
}
}
// 需要先定义局部内部类,再使用
InnerComparator com = new InnerComparator();
List<Integer> lst = new ArrayList<>();
Collections.addAll(lst, 2, 3, 5, 1, 4);
Collections.sort(lst, com);
System.out.println(lst);
}
}
[5, 4, 3, 2, 1]

匿名内部类

anonymous inner class

没有名字的局部内部类。

用于只创建这个类的一个对象的场合(不必命名),通常是作为方法的参数

隐式地继承了一个父类或者实现了一个接口

限制:由于构造方法的名字与类名相同,而匿名类没有类名,所以,匿名类不能有构造方法。

综合代码:

import java.util.Date;
class OutClass {
private static String field_Static = "外围类:静态变量";
private String field_Common = "外围类:普通变量";
private static void methodStatic() {
System.out.println("外围类:静态方法");
}
private void methodCommon() {
System.out.println("外围类:普通方法");
}
// 成员内部类
class InnerClassMember {
// Err:
// private static String inner_fieldStatic = "内部类:静态变量";
private String inner_fieldCommon = "|---内部类:普通变量";
// Err:
// private static void innerMethodStatic() {}
private void innerMethodCommon() {
System.out.println("|---内部类:普通方法");
}
public void methodInner() {
// 变量·外
System.out.println(OutClass.this.field_Common);
System.out.println(field_Static);
// 变量·内
System.out.println(inner_fieldCommon);
// 方法·外
methodStatic();
OutClass.this.methodCommon();
// 方法·内
innerMethodCommon();
}
}
// 静态内部类
static class InnerClassStatic {
private static String inner_fieldStatic = "|---静态内部类:静态变量";
private String inner_fieldCommon = "|---静态内部类:普通变量";
private static void innerMethodStatic() {
System.out.println("|---静态内部类:静态方法");
}
private void innerMethodCommon() {
System.out.println("|---静态内部类:普通方法");
}
public void methodInner() {
// 变量·外
System.out.println(OutClass.field_Static);
// 变量·内
System.out.println(inner_fieldCommon);
System.out.println(inner_fieldStatic);
// 方法·外
OutClass.methodStatic();
// 方法·内
innerMethodStatic();
innerMethodCommon();
}
}
// 局部內部類
public void method4InnerClassLocal() {
String localField = "局部变量";
final String localFieldFinal = "局部变量(final)";
// ↑Java 8中不用加final,但局部内部类中不能修改其值
// 局部内部类
class InnerClassLocal {
public void localClassM() {
// Err: System.out.println(localField);
System.out.println(localFieldFinal);
System.out.println(field_Common);
System.out.println(field_Static);
methodCommon();
methodStatic();
}
}
InnerClassLocal _local_in_c = new InnerClassLocal();
_local_in_c.localClassM();
}
// --------------------
public void testAnonymousClass() {
// 匿名内部类
PrintDate(new Date() {
public long getTime() {
System.out.println(field_Common);
System.out.println(field_Static);
methodCommon();
methodStatic();
return 10086;
}
});
}
private void PrintDate(Date dt) {
System.out.println(dt.getTime());
}
}
public class InnerClassTest1 {
public static void main(String[] args) {
System.out.println("===↓↓↓成员内部类===================");
OutClass.InnerClassMember ic = new OutClass().new InnerClassMember();
ic.methodInner();
System.out.println("===↓↓↓静态内部类===================");
OutClass.InnerClassStatic ics = new OutClass.InnerClassStatic();
ics.methodInner();
System.out.println("===↓↓↓局部内部类===================");
OutClass oc = new OutClass();
oc.method4InnerClassLocal();
System.out.println("===↓↓↓匿名内部类===================");
oc.testAnonymousClass();
}
}
===↓↓↓成员内部类===================
外围类:普通变量
外围类:静态变量
|---内部类:普通变量
外围类:静态方法
外围类:普通方法
|---内部类:普通方法
===↓↓↓静态内部类===================
外围类:静态变量
|---静态内部类:普通变量
|---静态内部类:静态变量
外围类:静态方法
|---静态内部类:静态方法
|---静态内部类:普通方法
===↓↓↓局部内部类===================
局部变量(final)
外围类:普通变量
外围类:静态变量
外围类:普通方法
外围类:静态方法
===↓↓↓匿名内部类===================
外围类:普通变量
外围类:静态变量
外围类:普通方法
外围类:静态方法
10086

对比和总结

面试题

面试题:

1、静态内部类和成员内部类有什么不同?(美团)

2、Anonymous Inner Class (匿名内部类)是否可以继承其它类,是否可以implements(实现)interface(接口)?

3、补全代码:

interface Inter {
public void show();
}
class Outer {
public static Inter method() {// 返回值是接口的方法
// 补全代码↓↓↓
return
// 补全代码↑↑↑
}
}
public class OuterDemo {
public static void main(String[] args) {
Outer.method().show();
}
}

Java基础教程——内部类的更多相关文章

  1. Java基础教程:内部类

    Java基础教程:内部类 内部类 内部类,是指在一个类的内部定义的类.就像下面这样: public class EnclosingClass {   . . .   public class Nest ...

  2. Java基础教程(12)--深入理解类

    一.方法的返回值   当我们在程序中调用方法时,虚拟机将会跳转到对应的方法中去执行.当以下几种情况发生时,虚拟机将会回到调用方法的语句并继续向下执行: 执行完方法中所有的语句: 遇到return语句: ...

  3. Java基础教程(18)--继承

    一.继承的概念   继承是面向对象中一个非常重要的概念,使用继承可以从逻辑和层次上更好地组织代码,大大提高代码的复用性.在Java中,继承可以使得子类具有父类的属性和方法或者重新定义.追加属性和方法. ...

  4. Java基础教程:注解

    Java基础教程:注解 本篇文章参考的相关资料链接: 维基百科:https://zh.wikipedia.org/wiki/Java%E6%B3%A8%E8%A7%A3 注解基础与高级应用:http: ...

  5. Java基础教程:网络编程

    Java基础教程:网络编程 基础 Socket与ServerSocket Socket又称"套接字",网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个s ...

  6. Java基础教程(5)--变量

    一.变量 1.变量的定义   正如上一篇教程<Java基础教程(4)--面向对象概念>中介绍的那样,对象将它的状态存在域中.但是你可能仍然有一些疑问,例如:命名一个域的规则和惯例是什么?除 ...

  7. Java基础教程:Lambda表达式

    Java基础教程:Lambda表达式 本文部分内容引用自OneAPM:http://blog.oneapm.com/apm-tech/226.html 引入Lambda Java 是一流的面向对象语言 ...

  8. Java基础教程:泛型基础

    Java基础教程:泛型基础 引入泛型 传统编写的限制: 在Java中一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义类型.如果要编写可以应用于多种类型的代码,这种刻板的限制就会束缚 ...

  9. Java基础教程:多线程基础(1)——基础操作

    Java:多线程基础(1) 实现多线程的两种方式 1.继承Thread类 public class myThread extends Thread { /** * 继承Thread类,重写RUN方法. ...

随机推荐

  1. 项目实战:流水线图像显示控件(列刷新、1ms一次、缩放、拽拖、拽拖预览、性能优化、支持OpenGL GPU加速)

      需求   流水线图像扫描采集控件(带模拟数据测试)性能需求  1.需至少满足可1ms接收一次列数据,而不丢包(接收后可不必立马显示)  2.图片刷新率可达30HZ:限制需求  1.图片高度最小只能 ...

  2. 排名靠前的几个JS框架发展趋势和前景

    转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者.原文出处:https://blog.bitsrc.io/top-5-javascript-frameworks ...

  3. 【总结】git

    一.基本介绍 1.历史 git版本控制工具,Linus开发linux时最开始用的是bitkeeper,后来该公司取消了linux社区的免费使用权力,linus用一周的时间开发出第一版git 2.git ...

  4. Visual Studio 2017 创建Winfrom工程

    1.打开Visual Studio 2017,出现界面点击-创建新项目 2.选择-Window桌面,选择windows 窗体应用(.NET Framework) 3.完成窗体程序创建,可在左边工具栏里 ...

  5. 关于windows下activeMQ的安装

    1.下载地址http://activemq.apache.org/activemq-5154-release.html 2.修改登录账号和密码,在配置文件jetty-realm.properties中 ...

  6. Raft算法原理剖析

    一.复制状态机(replicated state machine) Raft协议可以使得一个集群的服务器组成复制状态机,在详细了解Raft算法之前,我们先来了解一下什么是复制状态机.一个分布式的复制状 ...

  7. 仅用六种字符来完成Hello World,你能做到吗?

    Hello World 对于每一个开发者来说都不陌生,因为在我们学习任何一个语言或框架的时候,都会有一个Hello World的案例来帮助我们快速入门. 如果我们使用JavaScript来输出Hell ...

  8. monkey及其的日志管理和分析

    1.   monkey 1.1.  介绍 通过monkey程序模拟用户触摸屏幕,滑动Trackball.按键等操作来对设备上的程序进行压力测试,检测程序多久的时间会发生异常,检查和评估被测程序的稳定性 ...

  9. Java进阶专题(十六) 数据结构与算法的应用(上)

    前言 ​ 学习算法,我们不需要死记硬背那些冗长复杂的背景知识.底层原理.指令语法--需要做的是领悟算法思想.理解算法对内存空间和性能的影响,以及开动脑筋去寻求解决问题的最佳方案.相比编程领域的其他技术 ...

  10. 2020最常见的200+Java面试题汇总(含答案解析)

    前言 2020年快要结束了,很多朋友问题,有没有整理今年的一些面试题,最近抽时间整理了一份Java面试题.或许这份面试题还不足以囊括所有 Java 问题,但有了它,我相信足以应对目前市面上绝大部分的 ...