当前位置: 首页 > news >正文

Java内部类

在一个类的内部再定义一个类,所以称为内部类

内部类主要有几种:成员内部类局部内部类静态内部类匿名内部类

成员内部类

一个类中直接定义的内部类,可以理解为类的普通成员。

而作为类的成员,成员内部类可以无限制地访问外部类的所有成员(属性 + 方法)。


但有一点特别的是,成员内部类的实例不能单独存在,必须依附于一个外部类的实例

即要实例化一个成员内部类,必须首先创建一个外部类的实例,然后调用外部类实例的 new 来创建内部类实例 outer.new Inner()

这其实也很好理解,因为类的成员就是要依附于具体的实例的

// inner class
public class Main {public static void main(String[] args) {Outer outer = new Outer("Nested"); // 实例化一个OuterOuter.Inner inner = outer.new Inner(); // 实例化一个Innerinner.hello();}
}class Outer {private String name;Outer(String name) {this.name = name;}class Inner {void hello() {System.out.println("Hello, " + Outer.this.name);}}
}

静态内部类(嵌套类)

与成员内部类类似、也是定义在类中,只是多了 static 修饰。

因此与成员内部类所不同的是,静态内部类不再依附于外部类的实例,可以独立地实例化,是一个完全独立的类。

但是,它只可以访问外部类 静态的属性和方法。

// Static Nested Class
public class Main {public static void main(String[] args) {Outer.StaticNested sn = new Outer.StaticNested();sn.hello();}
}class Outer {private static String NAME = "OUTER";private String name;Outer(String name) {this.name = name;}static class StaticNested {void hello() {System.out.println("Hello, " + Outer.NAME);}}
}

局部内部类

在方法中定义的内部类称为局部内部类

与局部变量类似,局部内部类不能有访问说明符,因为它不是外围类的一部分。

但是它可以访问当前代码块内的常量,和外部类所有的成员


局部内部类的生命周期只限于定义该内部类的方法,只能在此方法内实例化,不可以在此方法外对其实例化。

public class Outer {private static int number = 100;private int j = 20;private String name = "Java";//定义外部类方法public void outer_funOne(int k){final int number = 100;int j = 50;//方法内部的类(局部内部类)class Demo{public Demo(int k){demo_funOne(k);}int number = 300; //可以定义与外部类同名的变量// static int j = 10;  //不可以定义静态变量//内部类的方法public void demo_funOne(int k){System.out.println("内部类方法:demo_funOne");//访问外部类的变量,如果没有与内部类同名的变量,则可直接用变量名System.out.println(name);//访问外部类与内部类同名的变量System.out.println(Outer.this.number);System.out.println("内部类方法传入的参数是:"+k);}}new Demo(k);}public static void main(String[] args) {//访问内部类必须要先有外部类对象Outer out = new Outer();out.outer_funOne(11);}}

匿名内部类

匿名内部类 是定义在方法中的,其语法为 new 某个接口或者类、并在后面加上类的定义

其实是 首先定义了一个匿名类、该类实现某个接口或继承某个类,并在定义后马上实例化了这个匿名类


与局部内部类一样,可以访问当前代码块内的常量,和外部类所有的成员

class Outer {private String name;Outer(String name) {this.name = name;}void asyncHello() {Runnable r = new Runnable() {@Overridepublic void run() {System.out.println("Hello, " + Outer.this.name);}};new Thread(r).start();}
}

日常开发用的最多的场景就是,用匿名内部类来实现函数式接口,并且通常是使用Lambda 表达式来简化代码。

如以上例子的匿名内部类就是实现了函数式接口 Runnable。

函数式接口:只有一个抽象方法的接口。

匿名类也可以继承自普通类

import java.util.HashMap;public class Main {public static void main(String[] args) {HashMap<String, String> map1 = new HashMap<>();HashMap<String, String> map2 = new HashMap<>() {};HashMap<String, String> map3 = new HashMap<>() {{put("A", "1");put("B", "2");}};System.out.println(map3.get("A"));}
}

map1 是一个普通的 HashMap 实例。

map2 是一个匿名类实例,只是该匿名类继承自 HashMap,并且类定义为空。

map3 也是一个继承自 HashMap 的匿名类实例,并且添加了构造代码块来初始化数据。

构造代码块:构造函数之前执行,并且每次实例化都会执行。


http://www.mrgr.cn/news/94920.html

相关文章:

  • Business processes A bridge to SAP and a guide to SAP TS410 certification
  • 贪心算法作业参考:P1106,P4995,P5019
  • 基于大模型的喉癌全程预测与治疗方案优化研究报告
  • AcWing 3533:查找第K小数 ← sort+unique
  • Docker和containerd之概览(Overview of Docker and Containerd)
  • 蓝桥杯备赛(基础语法4)
  • ngx_http_core_srv_conf_t
  • JUC大揭秘:从ConcurrentHashMap到线程池,玩转Java并发编程!
  • Java高级编程深度解析:JVM底层原理、设计模式与Java 8+新特性实战
  • 剑指 Offer II 109. 开密码锁
  • Windows 图形显示驱动开发-WDDM 3.0功能- 硬件翻转队列(三)
  • 基于PyQt5与Open3D的轻量化BIM工具开发指南(上)‌
  • ★ Linux ★ 进程(上)
  • 从C语言开始的C++编程生活(1)
  • 深入剖析React中setState的执行机制与实现原理
  • DeepSeek辅助学术写作中期能力及提示词分享
  • keepalived的工作原理和脑裂
  • 微服务面试题:服务网关和链路追踪
  • PHP、Java、Go、Python、Node.js、Ruby 写的接口,服务器承载量对比
  • STC89C52单片机学习——第25节: [11-1]蜂鸣器