匿名内部类精讲
Java中内部类详解—匿名内部类
Java中内部类详解 —匿名内部类
什么是内部类?
将一个类A定义在另一个类B里面,里面的那个类A就称为内部类,B则称为外部类。
成员内部类
定义在类中方法外的类。
定义格式:
class 外部类 { class 内部类{ }
}
在描述事物时,若一个事物内部还包含其他事物,就可以使用内部类这种结构。比如,汽车类Car 中包含发动机类Engine ,这时,Engine就可 以使用内部类来描述,定义在成员位置。
}
创建匿名内部类,并调用:
public class InnerDemo { public static void main(String[] args) { /* 1.等号右边:是匿名内部类,定义并创建该接口的子类对象 2.等号左边:是多态赋值,接口类型引用指向子类对象 */ FlyAble f = new FlyAble(){ public void fly() { System.out.println("我飞了~~~"); } };
}
以上两步,也可以简化为一步,代码如下:
public class InnerDemo3 { public static void main(String[] args) { /* 创建匿名内部类,直接传递给showFly(FlyAble f) */ showFly( new FlyAble(){ public void fly() { System.out.println("我飞了~~~"); } }); }
}
定义测试类:
public class InnerDemo { public static void main(String[] args) { // 创建外部类对象 Person p = new Person(); // 创建内部类对象 Heart heart = p.new Heart();
java中的匿名内部类总结
java中的匿名内部类总结java中的匿名内部类总结匿名内部类也就是没有名字的内部类正因为没有名字,所以匿名内部类只能使⽤⼀次,它通常⽤来简化代码编写但使⽤匿名内部类还有个前提条件:必须继承⼀个⽗类或实现⼀个接⼝实例1:不使⽤匿名内部类来实现抽象⽅法abstract class Person {public abstract void eat();}class Child extends Person {public void eat() {System.out.println("eat something");}}public class Demo {public static void main(String[] args) {Person p = new Child();p.eat();}}运⾏结果:eat something可以看到,我们⽤Child继承了Person类,然后实现了Child的⼀个实例,将其向上转型为Person类的引⽤但是,如果此处的Child类只使⽤⼀次,那么将其编写为独⽴的⼀个类岂不是很⿇烦?这个时候就引⼊了匿名内部类实例2:匿名内部类的基本实现public abstract void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something可以看到,我们直接将抽象类Person中的⽅法在⼤括号中实现了这样便可以省略⼀个类的书写并且,匿名内部类还能⽤于接⼝上实例3:在接⼝上使⽤匿名内部类interface Person {public void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something由上⾯的例⼦可以看出,只要⼀个类是抽象的或是⼀个接⼝,那么其⼦类中的⽅法都可以使⽤匿名内部类来实现最常⽤的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或是继承Runnable接⼝实例4:Thread类的匿名内部类实现public class Demo {public static void main(String[] args) {Thread t = new Thread() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};t.start();}}运⾏结果:1 2 3 4 5实例5:Runnable接⼝的匿名内部类实现public class Demo {public static void main(String[] args) {Runnable r = new Runnable() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};Thread t = new Thread(r);t.start();}}运⾏结果:1 2 3 4 5如有疑问请留⾔或者到本站社区交流讨论,感谢阅读,希望能帮助到⼤家,谢谢⼤家对本站的⽀持!。
chapter03-内部类匿名内部类匿名对象
2
类里面的两个重要方法使用说明
1、equals()方法:用于测试某个对象是否同另一个对象相等。它在 Object类中的实现是判断两个对象是否指向同一块内存区域。这 中测试用处不大,因为即使内容相同的对象,内存区域也是不同 的。如果想测试对象是否相等,就需要覆盖此方法,进行更有意 义的比较。
2 、 toString():返回该对象的字符串表示。Object类中的toString() 方法会打印出类名和对象的内存位置。几乎每个类都会覆盖该方 法,以便打印对该对象当前状态的表示。大多数(非全部) toString()方法都遵循如下格式:类名[字段名=值,字段名=值 ...],当然,子类应该定义自己的toString()方法。
7
内部类的几种具体应用
❖1、静态内部类 ❖2、局部内部类 ❖3、匿名内部类
8
匿名对象的使用
❖匿名对象通常使用在以下两种情况: ❖(1)如果对一个对象只需要进行一次方法
调用,那么就可以使用匿名对象。
❖(2)将匿名对象作为实参传递给一个方法 调用。
9
10
6
内部类的例子-使用内部类
如示例代码所示,对于给定的一个外部类实例outerClass,可以直接创建其内部 类实例,语法形式为: OuterClass.InnerClass innerClass = outerClass.new InnerClass(); 成员类不能与外部类重名 。 不能在成员类中定义static字段、方法和类。 成员类不能是接口(interface)。
java匿名内部类底层原理
java匿名内部类底层原理
Java匿名内部类是Java编程语言中的一个特性,允许你在一个方法内部定义一个没有名字的类。
这种类被称为匿名内部类。
匿名内部类的底层原理主要涉及到Java的字节码和内存模型。
下面是一些关键的概念和步骤:
1. 字节码:Java源代码在编译时会被转换成字节码。
字节码是一种平台无关的中间代码,可以在任何安装了Java虚拟机(JVM)的设备上运行。
2. 内存模型:当你在Java中创建对象时,JVM会在堆内存中为这个对象分配空间。
每个对象都包含类信息(即类的元数据)、实例变量和一些用于跟踪和管理的额外信息。
3. 匿名内部类的生成:当你在代码中定义一个匿名内部类时,JVM会做以下几件事情:
字节码生成:编译器会将匿名内部类的定义转换成字节码。
由于这个类没有名字,编译器会为它生成一个独特的类名,通常是基于它所在的方法和代码的位置。
堆内存分配:JVM会在堆内存中为这个匿名内部类对象分配空间。
这个对象会包含这个匿名类的所有实例变量和任何实现的方法。
实例化:当你在代码中创建这个匿名内部类的实例时,JVM会在堆内存中为这个新对象分配空间,并调用其构造函数来初始化这个对象。
4. 垃圾收集:当一个对象不再被引用时,JVM的垃圾收集器会回收这个对象占用的内存。
对于匿名内部类,如果它是局部变量并且它的范围已经结束(例如,方法已经返回),那么这个匿名类的对象和它占用的内存可能会被回收。
需要注意的是,由于JVM的内存管理策略和垃圾收集机制,实际的行为可能会因JVM的实现和运行时的环境而有所不同。
匿名内部类——精选推荐
匿名内部类内部类是指在⼀个外部类的内部再定义⼀个类。
匿名内部类也就是没有名字的内部类。
正因为没有名字,所以匿名内部类只能使⽤⼀次,它通常⽤来简化代码编写。
使⽤匿名内部类的前提条件是必须继承⼀个⽗类或实现⼀个接⼝。
继承抽象类的匿名内部类1abstract class Person {2public abstract void eat();3 }4public class Demo {5public static void main(String[] args) {6 Person p = new Person() {7public void eat() {8 System.out.println("eat something");9 }10 };11 p.eat();12 }13 }------------------------------------------------------------------------------------------------------------------------------------------------------------实现接⼝的匿名内部类1interface Person {2public void eat();3 }4public class Demo {5public static void main(String[] args) {6 Person p = new Person() {7public void eat() {8 System.out.println("eat something");9 }10 };11 p.eat();12 }13 }匿名内部类不能有构造⽅法;匿名内部类不能定义任何静态成员、⽅法和类;匿名内部类不能是public、protected、private、static;只能创建匿名内部类的⼀个实例;⼀个匿名内部类⼀定是在new的后⾯,⽤其隐含实现⼀个接⼝或实现⼀个类;因匿名内部类为局部内部类,所以局部内部类的所有限制都对其⽣效;在匿名类中⽤this时,这个this指的是匿名类本⾝。
内部类之匿名内部类
内部类之匿名内部类•内部类标识符:•每个类会产生一个.class文件,文件名即为类名。
同样,内部类也会产生一个.class文件,但是它的名称却不是内部类的类名,而是有着严格限制:外围类的名字,加上$,再加上内部类名字•匿名内部类•也就是没有名字的内部类,其名称由Java编译器给出,一般是形如:外部类名称$ 匿名类顺序,没有名称也就是其他地方就不能引用,不能实例化,只用一次,当然也就不能有构造器。
它通常用来简化代码编写•匿名内部的创建格式:new 父类构造器(参数列表)|实现接口(){//匿名内部类的类体部分}•匿名类分类•匿名类由于是一个new的结果,所以其实可以赋值给一个父类对象。
因此可以分为两种匿名类,成员匿名类和局部匿名类(作为函数参数)。
•使用匿名内部类前提条件•必须继承一个父类或实现一个接口当然也仅能只继承一个父类或者实现接口。
同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的引用。
当然这个引用是隐式•匿名内部类使用的注意事项•1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
同时要实现父类或接口中所有抽象方法,可以改写父类中的方法,添加自定义方法。
•2、匿名内部类因为没有类名,可知匿名内部类中是不能定义构造函数的。
•3、匿名内部类中不能存在任何的静态成员变量和静态方法。
•4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内类生效。
•5、因为在创建匿名内部类的时候,会立即创建它的实例,匿名内部类不能是抽象的,它必须要实现继承的类或者实现接口的类的所有抽象方法。
•6、匿名内部类和外部类有同名变量方法)时,默认访问的是匿名内部类的变量(方法),要访问外部类的变量(方法)则需要加上外部类的类名。
匿名内部类访问外部类成员变量或成员方法必须用static修饰•匿名内部类的使用package com.vg.demo04;//不使用匿名内部类来实现抽象方法abstract class Person{public abstract void eat();}class Child extends Person{public void eat() {System.out.println("eat something");}}public class TestDemoniming {public static void main(String[] args) {Person p = new Child();p.eat();}}package com.vg.demo04;//匿名内部类的基本实现abstract class Person{public abstract void eat();}public class TestDemoniming {public static void main(String[] args) { Person p = new Person() {public void eat() {System.out.println("eat something"); }};p.eat();}}package com.vg.demo04;//在接口上使用匿名内部类interface Person{public abstract void eat();}public class TestDemoniming {public static void main(String[] args) { Person p = new Person() {public void eat() {System.out.println("eat something"); }};p.eat();}}package com.vg.demo04;//Thread类的匿名内部类实现public class TestDemoniming {public static void main(String[] args) { Thread t = new Thread() {public void run () {for(int i = 1;i<=5;i ) {System.out.println(i " ");}}};t.start();}}package com.vg.demo04;//Runnable接口的匿名内部类实现public class TestDemoniming {public static void main(String[] args) { Runnable r = new Runnable() { public void run () {for(int i = 1;i<=5;i ) {System.out.println(i " ");}}};Thread t = new Thread(r);t.start();}}来源:。
Java:匿名类,匿名内部类
Java:匿名类,匿名内部类内部类匿名类⾸发⽇期:2018-03-25内部类:在⼀个类中定义另⼀个类,这样定义的类称为内部类。
【包含内部类的类可以称为内部类的外部类】如果想要通过⼀个类来使⽤另⼀个类,可以定义为内部类。
【⽐如苹果⼿机类,苹果⼿机类中的黄⾦版的是特别定制的】内部类的外部类的成员变量在内部类中仍然有效,内部类中的⽅法也可以调⽤外部类中的⽅法。
【不论是静态还是⾮静态的,内部类都可以直接调⽤外部类中的属性,】内部类的类体中不可以声明类变量和类⽅法。
内部类可以由外部类使⽤外部类中在函数中创建内部类对象来使⽤,,如果内部类的权限是⾮私有,⾮静态的,就可以在外部其他程序中被访问到,就可以通过创建外部类对象完成,;如果内部类是静态的,⾮私有的,静态成员可以直接类名调⽤,⾮静态成员通过创建外部类对象使⽤。
class Outer{int a=5;static int b=6;void show() {System.out.println("hello world");}class Inner{void use() {System.out.println(a);//5System.out.println(b);//6show();//hello world}}void create() {new Inner().use();}}public class Demo {public static void main(String[] args) {new Outer().create();Outer.Inner oi=new Outer().new Inner();e();}}补充:内部类的字节码⽂件会不⼀样。
会变成外部类名$内部类名将内部类定义在了局部位置上。
可以访问外部类的所有成员局部内部类不能访问所在局部的局部变量(本来他们⽣命周期不同,本来内部类对象的空间没有消失,对象⽣命长)若需访问,加final修饰变量即可,加final变成常量。
匿名内部类定义方法
匿名内部类定义方法嘿,朋友们!今天咱来聊聊匿名内部类定义方法这档子事儿。
啥是匿名内部类呀?简单来说,就像是一个隐藏在代码世界里的小神秘。
它没有名字,但却有着独特的作用。
想象一下啊,你在写代码的时候,突然需要一个类来做一件特定的小事儿,就这一下子的事儿,专门给它起个名字好像有点太麻烦了。
这时候,匿名内部类就闪亮登场啦!比如说,你有个接口,里面有个方法需要实现。
通常呢,你得专门写个类去实现它,然后再创建这个类的对象来用。
但有了匿名内部类,你可以直接在使用的地方就把这个类定义了,并且同时创建对象,多方便快捷呀!就好像你饿了,不用大老远跑去餐馆,而是直接在眼前就变出了美食。
它的定义方法其实也不难理解。
你就在需要的地方,直接用 new 关键字,后面跟上要实现的接口或者要继承的父类,然后在大括号里面写实现的方法或者重写的方法。
嘿,就这么简单粗暴!咱举个例子呗,比如说有个动物接口,里面有个叫叫的方法。
那你就可以在使用的地方这样写:new 动物接口 {public void 叫() {System.out.println("汪汪汪");}}你看,这就瞬间有了一个实现了动物接口,并且会“汪汪汪”叫的类啦!神奇不神奇?它的好处可不少呢!不仅让代码看起来更简洁,而且还能让你的代码更贴近具体的业务需求。
不用到处去找那个专门的实现类,就在当下,一下子就搞定了。
而且哦,匿名内部类还能访问外部类的变量呢,就像一个贴心的小助手,知道你需要什么,随时给你提供帮助。
当然啦,用匿名内部类也得注意一些事儿。
比如说,它可不能定义构造方法哦,毕竟它都没名字,怎么定义构造方法呢?还有啊,它的作用范围也比较有限,可别指望它能做太多太复杂的事儿。
总之呢,匿名内部类就像是代码世界里的一把小巧而锋利的剑,用好了能让你的代码如虎添翼,不好好掌握可就浪费啦!大家可得好好琢磨琢磨,把它用得恰到好处呀!怎么样,是不是对匿名内部类定义方法有了更深的了解呢?赶紧去试试吧!。
java中的匿名内部类总结
java中的匿名内部类总结匿名内部类也就是没有名字的内部类正因为没有名字,所以匿名内部类只能使⽤⼀次,它通常⽤来简化代码编写但使⽤匿名内部类还有个前提条件:必须继承⼀个⽗类或实现⼀个接⼝实例1:不使⽤匿名内部类来实现抽象⽅法abstract class Person {public abstract void eat();}class Child extends Person {public void eat() {System.out.println("eat something");}}public class Demo {public static void main(String[] args) {Person p = new Child();p.eat();}}运⾏结果:eat something可以看到,我们⽤Child继承了Person类,然后实现了Child的⼀个实例,将其向上转型为Person类的引⽤但是,如果此处的Child类只使⽤⼀次,那么将其编写为独⽴的⼀个类岂不是很⿇烦?这个时候就引⼊了匿名内部类实例2:匿名内部类的基本实现abstract class Person {public abstract void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something可以看到,我们直接将抽象类Person中的⽅法在⼤括号中实现了这样便可以省略⼀个类的书写并且,匿名内部类还能⽤于接⼝上实例3:在接⼝上使⽤匿名内部类interface Person {public void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something由上⾯的例⼦可以看出,只要⼀个类是抽象的或是⼀个接⼝,那么其⼦类中的⽅法都可以使⽤匿名内部类来实现最常⽤的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或是继承Runnable接⼝实例4:Thread类的匿名内部类实现public class Demo {public static void main(String[] args) {Thread t = new Thread() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};t.start();}}运⾏结果:1 2 3 4 5实例5:Runnable接⼝的匿名内部类实现public class Demo {public static void main(String[] args) {Runnable r = new Runnable() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};Thread t = new Thread(r);t.start();}}运⾏结果:1 2 3 4 5。
详解Java匿名内部类
详解Java匿名内部类匿名内部类:先举个例⼦吧,给⼤家看⼀下什么是匿名内部类,Endeavor刚刚接触的时候,觉得哇哦,好奇怪的样⼦,这也太别扭了吧,不知道⼤家是什么感觉。
为了进⾏对⽐,先举⼀个正常的类⽅法调⽤的例⼦(⼤家应该都看的懂吧):输出结果为:接下来便开始说正题吧,匿名内部类,通过名字,想必⼤家就知道什么是匿名内部类了吧,1、定义:就是没有名字的内部类(内部类之前介绍过了哦)。
2、使⽤内部类有什么好处呢,⼀句话就概括了:简化书写,⾄于是怎么简化的,哪⾥简化了等下再说。
3、先说⼀下什么时候使⽤匿名内部类,即使⽤前提和条件:必须存在继承和实现关系的时候才可以使⽤,其实这也很好理解,⾸先,匿名内部类没有名字,那该如何描述以及new个对象呢?对,没错,要通过继承它的⽗类或者实现⼀个接⼝来达成这⼀⽬的。
下⾯举个例⼦:输出结果:看到这⼏⾏代码,应该就不⽤我来解释为什么说匿名内部类可以简化书写了吧。
如何调⽤匿名内部类中的⽅法这个应该也是⼤家好奇的吧,毕竟匿名内部类没有类名该如何调⽤⽅法就成为了⼀个问题。
1、匿名内部类中只有⼀个⽅法的情况2、匿名内部类有⼀个⽅法调⽤起来很容易,那如果有多个⽅法呢,⼤家不会想这样吧?输出结果试⼀下:那么该如何调⽤匿名内部类中的多⽅法呢?Endeavor有两种⽅法,如果⼤家还有更好的⽅法多多赐教哦!1、利⽤多态的思想(匿名内部类的前提是存在继承或实现关系的)输出结果:2、改变⽅法类型,返回this输出结果:不知道看完之后⼤家有没有什么疑问,如果想调⽤匿名内部类⾃⼰特有的⽅法的时候呢?该如何调⽤呢?输出结果:显然是不⾏的,所以Endeavor觉得内名内部类可以有⾃⼰特有的⽅法,但是前提条件是这个匿名内部类只有这⼀个⽅法。
如果有多个⽅法的时候,他只能继承⽗类的⽅法以及重写这个⽅法或实现接⼝,绝不可能在在多个⽅法的情况下,调⽤⾃⾝特有的⽅法,但是这个特有的⽅法可以存在,但⽆法调⽤,不知道⼤家认同与否。
Java内部类之匿名内部类(使用抽象类,附详细代码示例)
Java内部类之匿名内部类(使⽤抽象类,附详细代码⽰例)此篇和上⼀篇的“使⽤接⼝”⽅式相互印证。
/*** 匿名内部类的概念和特性:* 1、匿名内部类只能也必须(有且仅有)继承⼀个⽗类或者实现⼀个接⼝,也可以重写⽗类普通⽅法;* 2、匿名内部类不能是抽象类,因为定义匿名内部类时⽴即⽣成对象,⽽抽象类不允许⽣成对象;* 3、匿名内部类没有类型,所以不能显式定义构造器,有⼀个隐式的⽆参构造器;* 4、匿名内部类可以定义初始化块以初始化对象,以弥补没有构造器的缺陷;* 5、最常⽤的匿名内部类的形式是:创建某个接⼝/抽象类的对象;* 6、定义之后必须马上使⽤,没办法通过引⽤进⾏使⽤;* 7、在匿名内部类和局部内部类中被访问到的局部变量,Java8之后默认为有final修饰,不可以重新赋值;*/定义⼀个抽象类:关于校园活动,其中有⼀个⽅法有⽅法体,可以被⼦类继承和重写。
1public abstract class CampusActivity {2public abstract String getProject(); // 获取活动项⽬3public abstract int getStuNumber(); // 获取活动⼈数4public String printInfo() {5return "正在进⾏班级活动";6 }7 }定义⼀个类,包含⼀个⽅法,需要有⼀个抽象类型的参数:1public class AnonymousClass { //定义⼀个类2public void test(CampusActivity ca) { //test⽅法需要⼀个抽象类类型作为参数3 System.out.println("正在进⾏的活动是:" + ca.getProject());4 System.out.println("参加活动的⼈数是:" + ca.getStuNumber());5 System.out.println("现在活动的状态是:" + ca.printInfo());6 }7 }编写⼀个测试类:实现了定义匿名内部类:1public class Test {23public static void main(String[] args) {4 AnonymousClass ac = new AnonymousClass();5 ac.test(new CampusActivity() { //使⽤抽象类来定义匿名内部类,实现抽象⽅法67 @Override8public int getStuNumber() {9return 100;10 }1112 @Override13public String getProject() {14return "拔河⽐赛";15 }1617 @Override18public String printInfo() { //这⾥重写了抽象类中的普通⽅法19return "活动马上要结束了";20 }21 });22 }23 }运⾏结果:欢迎各路⼤神多多评论,指教!。
匿名内部类——精选推荐
匿名内部类匿名内部类也就是没有名字的内部类使⽤匿名内部类有个前提条件:必须继承⼀个⽗类或实现⼀个接⼝。
例1:不使⽤匿名内部类来实现抽象⽅法可以看到,我们⽤Child继承了Person类,然后实现了Child的⼀个实例,将其向上转型为Person类的引⽤。
例2:匿名内部类的基本实现例3:在接⼝上使⽤匿名内部类匿名内部类最常⽤的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或者实现Runnable接⼝。
例4:Thread类的匿名内部类实现例5:Runnable接⼝的匿名内部类实现在使⽤匿名内部类时,要记住以下⼏个原则: · 匿名内部类不能有构造⽅法。
(因为匿名内部类是没有类名的) · 匿名内部类不能定义任何静态成员、⽅法和类。
· 匿名内部类不能是抽象类。
(因为在创建匿名内部类的时候,会⽴即创建匿名内部类的对象。
从⽽也可知匿名内部类必须实现它的抽象⽗类或者接⼝⾥包含的所有抽象⽅法) · 匿名内部类不能是public,protected,private,static。
· 只能创建匿名内部类的⼀个实例。
· ⼀个匿名内部类⼀定是在new的后⾯,⽤其隐含实现⼀个接⼝或实现⼀个类。
· 因匿名内部类为局部内部类,所以局部内部类的所有限制都对其⽣效。
· 内部类只能访问外部类的静态变量或静态⽅法。
匿名类和内部类中的中的this :有时候,我们会⽤到⼀些内部类和匿名类。
当在匿名类中⽤this时,这个this则指的是匿名类或内部类本⾝。
这时如果我们要使⽤外部类的⽅法和变量的话,则应该加上外部类的类名。
关于内部类的作⽤:⼀个内部类对象可以访问创建它的外部类对象的内容,甚⾄包括私有变量!Java编译器在创建内部类对象时,隐式的把其外部类对象的引⽤也传了进去并⼀直保存着。
这样就使得内部类对象始终可以访问其外部类对象,同时这也是为什么在外部类作⽤范围之外向要创建内部类对象必须先创建其外部类对象的原因。
Java培训系列课程-匿名内部类(精品培训)
public void print();
Байду номын сангаас
}
public class TestDemo {
public static void main(String args[]) {
fun(new Message() {
// 直接实例化接口对象
public void print() { // 匿名内部类中覆写print()方法
匿名内部类
• 内部类指的是在一个类的内部定义了另外的类 结构,利用内部类可以方便的实现私有属性的 互相访问,但是内部类是需要明确的使用class 进行定义的。而匿名内部类的是没有名字的内 部类,但是其必须在抽象类或接口基础上才可 以定义。
定义匿名内部类
interface Message {
// 定义接口
System.out.println("Hello World !");
}
});
// 传递匿名内部类实例化
}
public static void fun(Message msg) {
// 接收接口对象
msg.print();
}
}
程序执行结果:
Hello World !
2019年x月
工程师培训资料
• 标题:Java培训系列课程-匿名内部类(精品培训) • 培训人:xx
本章学习目标
• 掌握继承性的主要作用、实现、使用限制; • 掌握方法覆写的操作; • 掌握final关键字的使用; • 掌握对象多态性的概念以及对象转型的操作; • 掌握抽象类和接口的定义、使用、常见设计模式; • 掌握Object类的主要特点及实际应用; • 掌握匿名内部类的使用; • 掌握基本数据类型包装类的使用。
匿名内部类的使用
匿名内部类的使⽤⽬标1,匿名内部类的作⽤。
2,匿名内部类的定义格式。
具体内容匿名内部类的定义:没有名字的内部类。
表⾯上看起来那是它的名字,实际上不是它的名字。
使⽤匿名内部类的前提:必须继承⽗类或实现⼀个接⼝匿名内部类的分类:1.继承式的匿名内部类2.接⼝式的匿名内部类3.参数式的匿名内部类特点:匿名内部类⼀定是跟在new的后⾯,⽤其隐含实现⼀个接⼝或⼀个类,没有类名,根据多态,我们使⽤其⽗类名。
因为匿名内部类属于局部类,所以局部类的所有限制对其⽣效。
匿名内部类是唯⼀⼀种⽆构造⽅法的类。
匿名内部类在编译时,系统⾃动起名Out$1.class。
如果⼀个对象编译时的类型是接⼝,那么其运⾏的类型为实现这个接⼝的类。
注意点:1.不能有构造⽅法2.不能定义任何静态成员、⽅法或类3.不能是public,protected,private,static。
4.只能创建匿名内部类的⼀个实例。
匿名内部类的优点:可以使命名变得简洁使代码更加紧凑,简洁,封装性⽐内部类更优⼀个类⽤于继承其他类或是实现接⼝,⽆需增加其他的⽅法,只是对继承⽅法实现覆盖。
内部类:在⼀个类的内部还有另⼀个类。
那么匿名类,如果⼀个类在整个操作中,只使⽤⼀次的话,就可以将其定义成匿名内部类。
匿名内部类是在接⼝及抽象类的基础上发展起来的。
interface A{public void printInfo() ; //}class B implements A{ // 实现接⼝public void printInfo(){System.out.println("Hello World") ;}};class X {public void fun1(){this.fun2(new B()) ;}public void fun2(A a){a.printInfo() ;}};public class NoInnerClassDemo01{public static void main(String args[]){new X().fun1() ; // 实例化X类的对象,并调⽤fun1()⽅法}};如果此时,B类只使⽤⼀次的话,那么还有必要为其单独定义⼀次吗?此时可以使⽤匿名内部类解决问题。
彻底搞懂java匿名内部类final
彻底搞懂java匿名内部类final1. 其实是局部匿名内部类,跟内部类是有本质区别2. 对于局部变量是final,对全局变量不是的3. 为什么对局部变量是final呢?a.从最初的设计讲:匿名内部类传进来就是那个局部变量的引用,而普通调用,认为传进来的是对象,而不是引用;b.常规说法:当方法运行完后,局部变量被销毁,如果局部匿名内部类还可以访问到局部变量,超出了作用域。
说法没错,但必须要正确理解:a1.首先销毁的并不是方法内产生的对象,而是对象的局部变量引用;看下面两个图即可知;a2.超出作用域:因为传进来的是局部变量的引用,所以不应该允许引用都被销毁了,还可以做更改;所以这是为什么说“超出作用域”,这也是为什么需要定义final的原因;c.底层原理还是从最初的设计来的,匿名内部类的简化操作,他的存在的最大作用就是执行一段代码,所以它不需要构造方法,所以外面的对象应该以怎样的方式传进来呢?那就没有办法传进来了,但我又想用外面的东西怎么办?那就直接用,直接用的效果就是,看起来是直接用的外面的引用。
但事实上,底层是将外面的对象拷贝了一份到匿名内部类里,帮我们做了构造,这么做的原因就是在于:就是为了避免局部方法执行完销毁了引用和对象的原因,生命周期差异的原因,所以才做了一个拷贝动作,而对于全局变量就不用这样做了,因为全局变量的生命周期足够长。
而就是这个拷贝,就会让外部的修改和内部对象的修改不同步了,但这种明显是不对的,因为开发者认为他们是同一个东西,所以既然这样,干脆直接禁止修改好了,加final。
4. 个人观点:个人认为这是java设计的一个不好的地方。
好的方式:就直接让开发者认为不论是普通的函数调用,还是匿名内部类的调用,都是传进来的对象即可,而不是引用。
为了简单,这里直接用了内部类也是为了简化,用了static全局变量。
匿名内部类的原理
匿名内部类的原理
匿名内部类是Java编程语言中的一个重要概念,其设计理念在于简化代码和提高可读性。
匿名内部类的核心原理主要体现在以下几个方面。
首先,匿名内部类是一种没有类名的内部类。
通常我们在定义一个类时,需要给出类的名字,但是在某些情况下,我们只需要创建一个类的实例,而不需要再次使用这个类,这时候就可以使用匿名内部类。
由于没有类名,匿名内部类只能在创建的时候使用一次,不能在其他地方重复使用。
其次,匿名内部类必须继承一个父类或者实现一个接口。
这是因为Java是面向对象的语言,所有的类都是对象,而对象必须属于某个类。
由于匿名内部类没有类名,所以它不能直接被实例化,必须通过继承父类或者实现接口的方式来获取实例。
在使用匿名内部类时,我们需要在创建实例的同时定义类的内容。
这通常是通过在new 关键字后面跟上父类构造器或者接口名,然后在括号内编写类的内容来实现的。
这种方式可以让我们在不需要重复使用类的情况下,快速创建出需要的对象。
总的来说,匿名内部类的原理在于通过省略类名,实现在创建实例的同时定义类,从而达到简化代码和提高可读性的目的。
它是Java编程语言中的一种重要特性,对于提高编程效率和代码质量有着重要的作用。
匿名内部类
• }
• 先举一个简单的例子,如果你想实现一个接口, 先举一个简单的例子,如果你想实现一个接口, 但是这个接口中的一个方法和你构想的这个类中 的一个方法的名称,参数相同,你应该怎么办? 的一个方法的名称,参数相同,你应该怎么办? 这时候,你可以建一个内部类实现这个接口。 这时候,你可以建一个内部类实现这个接口。由 于内部类对外部类的所有内容都是可访问的, 于内部类对外部类的所有内容都是可访问的,所 以这样做可以完成所有你直接实现这个接口的功 能。 • 不过你可能要质疑,更改一下方法的不就行了吗? 不过你可能要质疑,更改一下方法的不就行了吗? • 的确,以此作为设计内部类的理由,实在没有说 的确,以此作为设计内部类的理由, 服力。 服力。 • 真正的原因是这样的,java中的内部类和接口加 真正的原因是这样的, 中的内部类和接口加 在一起,可以的解决常被C++程序员抱怨 程序员抱怨java中 在一起,可以的解决常被 程序员抱怨 中 没有多继承。实际上, 存在的一个问题 没有多继承。实际上,C++的多 的多 继承设计起来很复杂, 继承设计起来很复杂,而java通过内部类加上接 通过内部类加上接 可以很好的实现多继承的效果。 口,可以很好的实现多继承的效果。
特点
• 一个匿名内部类一定是在new的后面,用其隐含实现一个 接口或实现一个类,没有类名,根据多态,我们使用其父 类名。因他是局部内部类,那么局部内部类的所有限制都 对其生效。匿名内部类是唯一一种无构造方法类。大部分 匿名内部类是用于接口回调用的。匿名内部类在编译的时 候由系统自动起名Out$1.class。如果一个对象编译时的 类型是接口,那么其运行的类型为实现这个接口的类。因 匿名内部类无构造方法,所以其使用范围非常的有限。当 需要多个对象时使用局部内部类,因此局部内部类的应用 相对比较多。匿名内部类中不能定义构造方法。如果一个 对象编译时的类型是接口,那么其运行的类型为实现这个 接口的类。
java中的匿名内部类总结
java中的匿名内部类总结匿名内部类也就是没有名字的内部类正因为没有名字,所以匿名内部类只能使⽤⼀次,它通常⽤来简化代码编写但使⽤匿名内部类还有个前提条件:必须继承⼀个⽗类或实现⼀个接⼝实例1:不使⽤匿名内部类来实现抽象⽅法abstract class Person {public abstract void eat();}class Child extends Person {public void eat() {System.out.println("eat something");}}public class Demo {public static void main(String[] args) {Person p = new Child();p.eat();}}运⾏结果:eat something可以看到,我们⽤Child继承了Person类,然后实现了Child的⼀个实例,将其向上转型为Person类的引⽤但是,如果此处的Child类只使⽤⼀次,那么将其编写为独⽴的⼀个类岂不是很⿇烦?这个时候就引⼊了匿名内部类实例2:匿名内部类的基本实现abstract class Person {public abstract void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something可以看到,我们直接将抽象类Person中的⽅法在⼤括号中实现了这样便可以省略⼀个类的书写并且,匿名内部类还能⽤于接⼝上实例3:在接⼝上使⽤匿名内部类interface Person {public void eat();}public class Demo {public static void main(String[] args) {Person p = new Person() {public void eat() {System.out.println("eat something");}};p.eat();}}运⾏结果:eat something由上⾯的例⼦可以看出,只要⼀个类是抽象的或是⼀个接⼝,那么其⼦类中的⽅法都可以使⽤匿名内部类来实现最常⽤的情况就是在多线程的实现上,因为要实现多线程必须继承Thread类或是继承Runnable接⼝实例4:Thread类的匿名内部类实现public class Demo {public static void main(String[] args) {Thread t = new Thread() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};t.start();}}运⾏结果:1 2 3 4 5实例5:Runnable接⼝的匿名内部类实现public class Demo {public static void main(String[] args) {Runnable r = new Runnable() {public void run() {for (int i = 1; i <= 5; i++) {System.out.print(i + " ");}}};Thread t = new Thread(r);t.start();}}运⾏结果:1 2 3 4 5。
匿名内部类 Anoymous Inner Class, 可以继承抽象类,实现接口
匿名内部类Anoymous Inner Class, 可以继承抽象类,实现接口匿名内部类 Anoymous Inner Class, 可以继承抽象类,实现接口匿名类本身就是通过继承类或者接口来实现的。
但是不能再显式的extends 或者implements了。
举个例子:Java代码1.JFrame frame = new JFrame();我们为frame添加窗口事件可以采用如下两种方式:Java代码1. frame.addWindowListener(new WindowListener() {2.public void windowOpened(WindowEvent e) {3.throw new UnsupportedOperationException("Not supported yet.");}4.public void windowClosing(WindowEvent e) {5.throw new UnsupportedOperationException("Not supported yet.");}6. public void windowClosed(WindowEvent e) {7.throw new UnsupportedOperationException("Not supported yet.");}8.public void windowIconified(WindowEvent e) {9.throw new UnsupportedOperationException("Not supported yet.");}10.public void windowDeiconified(WindowEvent e) {11.throw new UnsupportedOperationException("Not supported yet.");}12.public void windowActivated(WindowEvent e) {13.throw new UnsupportedOperationException("Not supported yet.");}14. public void windowDeactivated(WindowEvent e) {15.throw new UnsupportedOperationException("Not supported yet.");} });这个匿名类就是实现了接口WindowListener或者:Java代码1.frame.addWindowListener(new WindowAdapter() {});上面这个匿名类就是继承了WindowAdapter这个抽象类。
匿名内部类主要原则
匿名内部类主要原则
匿名内部类的主要原则包括:
1. 匿名内部类不能有构造方法。
因为匿名内部类没有类名,所以不能通过构造方法进行实例化。
2. 匿名内部类不能定义任何静态成员、方法和类。
因为匿名内部类没有类名,所以无法定义静态成员。
3. 一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类。
4. 因匿名内部类为局部内部类,所以局部内部类的所有限制都对其生效。
例如,匿名内部类不能是public、protected、private,也不能是static。
5. 内部类只能访问外部类的静态变量或静态方法。
这是因为内部类在编译后会被生成独立的类文件,为了防止对外部类的修改,所以限制了其访问权限。
6. 匿名内部类不能是abstract类,因为无法创建其实例。
7. 匿名内部类不能是public、protected、private,因为它们的作用范围受到了限制。
8. 匿名内部类不能被直接引用,因此无法创建其实例。
9. 匿名内部类不能包含静态初始化块和实例初始化块。
10. 匿名内部类的实例只能创建一次,因为它是单例的。
遵循这些原则可以确保代码的正确性和可读性,同时也符合Java语言的设计理念和规范。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
1.new父类构造器(参数列表)|实现接口()
2.{
3.//匿名内部类的类体部分
4.}
从上面定义可以看出,匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,或实现一个接口。
关于匿名内部类还有如下两条规则:
1)匿名内部类不能是抽象类,因为系统在创建匿名内部类的时候,会立即创建内部类的对象。
因此不允许将匿名内部类
定义成抽象类。
2)匿名内部类不等定义构造器,因为匿名内部类没有类名,所以无法定义构造器,但匿名内部类可以定义实例初始化块,
通过实例初始化块来完成构造器需要完成的事情。
最常用的创建匿名内部类的方式是需要创建某个接口类型的对象,如下程序所示:[java]view plaincopy
1.interface Product{
2.public double getPrice();
3.public String getName();
4.}
5.public class TestAnonymous{
6.public void test(Product p){
7. System.out.println("购买了一个"+p.getName()+",花掉了
"+p.getPrice());
8. }
9.public static void main(String[]args){
10. TestAnonymous ta = new TestAnonymous();
11. ta.test(new Product(){
12.public double getPrice(){
13.return567;
14. }
15.public String getName(){
16.return"AGP显卡";
17. }
18. });
19. }
20.}
上面程序中的TestAnonymous类定义了一个test方法,该方法需要一个Product对象作为参数,但Product只是一个接口,
无法直接创建对象,因此此处考虑创建一个Product接口实现类的对象传入该方法---如果这个Product接口实现类需要重复
使用,则应该经该实现类定义一个独立类;如果这个Product接口实现类只需一次使用,则可采用上面程序中的方式,定义
一个匿名内部类。
正如上面程序中看到,定义匿名类不需要class关键字,而是在定义匿名内部类时直接生成该匿名内部类的对象。
上面
粗体字代码部分就是匿名类的类体部分。
由于匿名内部类不能是抽象类,所以匿名内部类必须实现它的抽象父类或者接口里包含的所有抽象方法。
对于上面创建Product实现类对象的代码,可以拆分成如下代码:
[java]view plaincopy
1.class AnonymousProduct implements Product{
2.public double getPrice(){
3.return567;
4. }
5.public String getName(){
6.return"AGP显卡";
7. }
8. }
9. ta.test(new AnonymousProduct());
当通过实现接口来创建匿名内部类时,匿名内部类也不能显示创建构造器,因此匿名内部类只有一个隐式的无参数构造
器,故new接口名后的括号里不能传入参数值。
但如果通过继承父类来创建匿名内部类是,匿名内部类将拥有和父类相似的构造器,此处的相似指的是拥有相同的形参
列表。
[c-sharp]view plaincopy
1.abstract class Device{
2.private String name;
3.public Device(){
4. }
5.public Device(String name){
= name;
7. }
8.public abstract double getPrice();
9.//此处省略了name属性的setter和getter方法
10. }
11.public class AnonymousInner{
12.public void test(Device d){
13. System.out.println("购买了一个"+d.getName()+",花掉了"+d.getPrice());
14. }
15.public static void main(String[] args){
16. AnonymousInner ai = new AnonymousInner();
17.//调用有参数的构造器创建Device匿名实现类的对象
18. ai.test(new Device("电子示波器"){
19.public double getPrice(){
20.return 67;
21. }
22. });
23.//调用无参数的构造器创建Device匿名实现类的对象
24. Device d = new Device(){
25.//初始化块
27. System.out.println("匿名内部类的初始化块...");
28. }
29.//实现抽象方法
30.public double getPrice(){
31.return 56;
32. }
33.public Sting getName(){
34.return"键盘";
35. }
36. };
37. ai.test(d);
38. }
39. }
上面程序创建了一个抽象父类Device,这个抽象父类里包含两个构造器:一个无参数的,一个有参数的。
当创建以Device
为父类的匿名内部类时,即可以传入参数(如上面程序中第一段粗体字部分),也可以不传入参数(如上面程序中第二段粗体
字部分)。
当创建匿名内部类时,必须实现接口或抽象父类里的所有抽象方法。
如果有需要,也可以重写父类中的普通方法,如上面
程序的第二段粗体字代码部分,匿名内部类重写了抽象父类Device类的getName方法,其中getName方法并不是抽象方法。
如果匿名内部类需要访问外部类的局部变量,则必须使用final修饰符来修饰外部类的局部变量,
否则系统将报错。
[java]view plaincopy
1.interface A{
2.void test();
3. }
4.public class TestA{
5.public static void main(Strign[] args){
6.int age = 0;
7. A a = new A(){
8.public void test(){
9.//下面语句将提示错误:匿名内部类内访问局部变量必须使用final修饰
10. System.out.println(age);
12. };
13. }
14. }
上面程序中粗体子代码是匿名内部类访问了外部类的局部变量,由于age变量没有使用final修饰符修饰,所以粗体字代码将
引起编译异常。