Java内部类总结

2023-11-15

Java内部类其实在J2EE编程中使用较少,不过在窗口应用编程中特别常见,主要用来事件的处理。其实,做非GUI编程,内部类完全可以不用。
 
内部类的声明、访问控制等于外部类有所不同,要灵活使用内部类来编写程序,还是有相当难度的,Java发明了这种难懂的玩意儿,在其他语言中是没有的,但是在Java中,内部类也相当的重要,尤其做GUI开发时候,事件的响应处理全靠内部类了。
 
内部类所做的功能使用外部类也同样可以实现,只是有时候内部类做的更巧妙些。
 
内部类按照其所在位置不同,可分为以下几种:
1、(普通的)内部类(最常见的内部类,内部类的定义与类成员平级,)
2、方法内部类
3、匿名类
4、静态内部类
5、接口内部类
 
一、内部类声明与访问
 
1、内部类直接在类的内部进行声明。可以声明为private、protected、public或者默认访问权限,这个访问权限约定和外部类完全一样。
 
2、内部类自动拥有对其外围类所有成员(方法、属性)的访问权。如果内部类和外部类成员的名字完全相同,在内部类方法中要访问外部类成员,则需要使用下面的方式来访问: 外部类名.this.外部成员名,例如Outer.this.i++;  (看例子)
 
3、必须使用外部类对象来创建内部类对象,而不是直接去new一个。
格式为: 外部对象名.new 内部类构造方法
 
比如要创建一个内部类iner对象,需要这么做:
        Outer outer = new Outer();
        Outer.Inner iner = outer.new Inner();
 
/** 
* 内部类创建与初始化 

* @author leizhimin 2009-7-17 13:51:52 
*/
 
public  class Outer { 
         private  int i = 10; 
         private  int y = 8; 

        Outer() { 
                System.out.println( "调用Outer构造方法:outer"); 
        } 

         public  void sayMsg() { 
                System.out.println( "Outer class!"); 
        } 

         class Inner { 
                 int i = 1000; 

                Inner() { 
                        System.out.println( "调用Inner构造方法:inner"); 
                } 

                 void innerMsg() { 
                        System.out.println( ">>>>>Inner class!"); 
                        sayMsg(); 
                         //访问内部类自己的成员i,也可以写成 this.i++ 
                         this.i++; 
                         //访问外部类的成员 i和y 
                        Outer. this.i++; 
                        y--; 
                } 

                 int getI() { 
                         return i; 
                } 
        } 

         public  void test() { 
                Inner in =  new Inner(); 
                in.innerMsg(); 
        } 

         public  int getI() { 
                 return i; 
        } 

         public  void setI( int i) { 
                 this.i = i; 
        } 


class Test1 { 
         public  static  void main(String[] args) { 
                Outer outer =  new Outer(); 
                outer.test(); 
                System.out.println(outer.getI()); 
                System.out.println( "-------1--------"); 

                Outer.Inner iner = outer. new Inner(); 
                iner.innerMsg(); 
                System.out.println(iner.getI()); 
                System.out.println( "-------2--------"); 

                System.out.println(outer.getI()); 
        } 
}
 
运行结果:
调用Outer构造方法:outer 
调用Inner构造方法:inner 
>>>>>Inner  class
Outer  class
11 
-------1-------- 
调用Inner构造方法:inner 
>>>>>Inner  class
Outer  class
1001 
-------2-------- 
12 

Process finished with exit code 0
 
二、内部类与接口
 
1、内部类可以实现接口。
2、内部类之间相互可见,但并非内部类之间方法都可见。
 
public  interface Foo{ 
          void say(); 
}
 
public  interface Bar { 
         void readme(); 
}
 
/** 
* 内部类实现接口 

* @author leizhimin 2009-7-17 14:57:50 
*/
 
public  class Test2 { 
         public  static  void main(String[] args) { 
                Outer outer =  new Outer(); 
                Foo f = outer.genFoo(); 
                Bar b = outer.genBar(); 
                f.say(); 
                b.readme(); 
        } 


class Outer { 
         private  class FooImpl  implements Foo { 
                 public  void say() { 
                        System.out.println( "say foo!"); 
                } 
        } 

         private  class BarImpl  implements Bar { 
                 public  void readme() { 
                        System.out.println( "say bar!"); 
                } 
        } 

         public Foo genFoo() { 
                 return  new FooImpl(); 
        } 

         public Bar genBar() { 
                 return  new BarImpl(); 
        } 
}
 
输入结果:
say foo! 
say bar! 

Process finished with exit code 0
 
三、访问权限
 
外部类分两种:
一种嵌入了内部类声明代码外部类,称为直接外部类。
另一种是与内部类没有任何关系的外部类,称为外部类。
 
在同一个直接外部类中,内部类之间所有的方法都是相互可见的,包含在直接外部类的main()中可见。

在外部类中,要看到一个类的内部类成员,则至少要求这个内部类的class和成员权限大于或等于protected。
 
/** 
* 内部类实现接口 

* @author leizhimin 2009-7-17 14:57:50 
*/
 
public  class Test2 { 
         public  static  void main(String[] args) { 
                Outer o =  new Outer(); 
                Outer.Bar b = o.genBar(); 
                b.readme(); 
        } 


class Outer { 

         protected  class Foo { 
                 protected  void say() { 
                        System.out.println( "say foo!"); 
                } 

                 private  void test() { 
                        System.out.println( "----test------"); 
                } 
        } 

         protected  class Bar { 
                 protected  void readme() { 
                        System.out.println( "say bar!"); 
                         new Foo().test(); 
                } 
        } 

         public Foo genFoo() { 
                 return  new Foo(); 
        } 

         public Bar genBar() { 
                 return  new Bar(); 
        } 
}
 
四、方法内部类
 
方法内部类只在该方法内部可见,方法内部类可以定义在方法中的任何位置。
/** 
* 内部类实现接口 

* @author leizhimin 2009-7-17 14:57:50 
*/
 
public  class Test2 { 
         public  static  void main(String[] args) { 
                Outer outer =  new Outer(); 
                Foo f = outer.genFoo(); 
                Bar b = outer.genBar(); 
                f.say(); 
                b.readme(); 
        } 


class Outer { 
         public Foo genFoo() { 
                 //方法内的内部类 
                 class FooImpl  implements Foo { 
                         public  void say() { 
                                System.out.println( "say foo!"); 
                        } 
                } 
                 return  new FooImpl(); 
        } 

         public Bar genBar() { 
                Bar b =  null
                 if ( true) { 
                         //任意位置的内部类 
                         class BarImpl  implements Bar { 
                                 public  void readme() { 
                                        System.out.println( "say bar!"); 
                                } 
                        } 
                        b =  new BarImpl(); 
                } 
                 return b; 
        } 
}
 
运行结果:
say foo! 
say bar! 

Process finished with exit code 0
 
五、匿名类
 
匿名类不给出类名,直接定义一个类,通常这个类实现了某种接口或者抽象。匿名类的访问权限更没有讨论价值了,看个例子就行了。
 
在一些多线程程序中比较常见,有点变态,呵呵。
/** 
* 匿名类. 

* @author leizhimin 2009-7-17 15:56:17 
*/
 
public  class Test3 { 
         public Foo f =  new Foo() { 
                 public  void say() { 
                        System.out.println( "O(∩_∩)O哈哈~!"); 
                } 
        }; 

         public Foo test() { 
                 return  new Foo() { 
                         public  void say() { 
                                System.out.println( "say foo!"); 
                        } 
                }; 
        } 

         public  static  void main(String[] args) { 
                Test3 t =  new Test3(); 
                t.f.say(); 
                t.test().say(); 
        } 


interface Foo { 
         void say(); 
}
 
运行结果:
O(∩_∩)O哈哈~! 
say foo! 

Process finished with exit code 0
 
/** 
* 普通类的匿名初始化 

* @author leizhimin 2009-7-17 16:13:31 
*/
 
public  class Fk { 
         private String x; 

         public Fk(String x) { 
                 this.x = x; 
        } 

        @Override 
         public String toString() { 
                 return  "Fk{" + 
                                 "x='" + x + '\'' + 
                                '}'; 
        } 


class Test4 { 
         public Fk hehe() { 
                 //把后面的一对大括号去掉呢,呵呵 
                 return  new Fk( "fk") { 
                }; 
        } 

         public  static  void main(String[] args) { 
                Test4 t =  new Test4(); 
                Fk f = t.hehe(); 
                System.out.println(f); 
        } 
}
 
运行结果:
Fk{x='fk'} 

Process finished with exit code 0
 
还有一个不得不提的经典实例,来自thining in java,有改动:
interface Service { 
     void method1(); 
     void method2(); 


interface ServiceFactory { 
    Service getService(); 


class Implementation1  implements Service { 
     private Implementation1() {} 
     public  void method1() {System.out.println( "Implementation1 method1");} 
     public  void method2() {System.out.println( "Implementation1 method2");} 
     public  static ServiceFactory factory =  new ServiceFactory() { 
             public Service getService() { 
                 return  new Implementation1(); 
            } 
        }; 


class Implementation2  implements Service { 
     private Implementation2() {} 
     public  void method1() {System.out.println( "Implementation2 method1");} 
     public  void method2() {System.out.println( "Implementation2 method2");} 
     public  static ServiceFactory factory =  new ServiceFactory() { 
             public Service getService() { 
                 return  new Implementation2(); 
            } 
        }; 


public  class Factories { 
     public  static  void serviceConsumer(ServiceFactory fact) { 
        Service s = fact.getService(); 
        s.method1(); 
        s.method2(); 
    } 
     public  static  void main(String[] args) { 
        serviceConsumer(Implementation1.factory); 
        serviceConsumer(Implementation2.factory); 
    } 
}
 
这个应用给了我们很多思考,我就不说了,不同人看了会有不同的感受。
 
内部类的巧妙使用会让你的代码很牛,如果要形容下,那就是:没看懂的时候感觉神出鬼没,看懂后感觉鬼斧神工。不过这些代码多了,别人想看懂都难,想看懂你思路就难上加难了。呵呵!
 
六、静态内部类
 
静态内部类是static class型的内部类,这种内部类特点是:它不能访问外部类的非静态成员。要创建静态内部类对象时候,也不需要外部类对象了,直接可以:
new 外部类名.内部类构造方法
来创建,给个例子:
/** 
* 静态内部类 

* @author leizhimin 2009-7-17 16:53:05 
*/
 
public  class Outer { 
         public  static  int i =500; 
         protected  static  class Inner { 
                 int i =100; 
                String name; 

                Inner(String name) { 
                         this.name = name; 
                } 

                 void sayHello() { 
                        System.out.println( "Hello " + name); 
                        Outer.i++; 
                } 
        } 

         public Inner genInner(String name) { 
                 return  new Inner(name); 
        } 


class Test { 
         public  static  void main(String[] args) { 
                Outer.Inner in1 =  new Outer.Inner( "1111"); 
                in1.sayHello(); 
                System.out.println(Outer.i); 

                Outer.Inner in2 =  new Outer().genInner( "2222"); 
                in2.sayHello(); 
                System.out.println(Outer.i); 
        } 
}
 
运行结果:
Hello 1111 
501 
Hello 2222 
502 

Process finished with exit code 0
 
七、接口内部类
 
接口内部类自动都是public static的,相当于为接口定义了一种变量类型,这在java的设计中就有使用,比如在HashMap中,就有:
static class Entry<K,V> implements Map.Entry<K,V>
 
下面我给个例子,
/** 
* 接口内部类 

* @author leizhimin 2009-7-17 17:20:28 
*/
 
public  interface AInterface { 
         void readme(); 

         class Inner1  implements AInterface { 
                 public  void readme() { 
                        System.out.println( "我是一个接口内部类"); 
                } 
        } 


class Main { 
         public  static  void main(String[] args) { 
                AInterface.Inner1 in1 =  new AInterface.Inner1(); 
                in1.readme(); 
        } 
}
 
八、内部的类的嵌套
 
所谓内部类嵌套,就是内部类里面再定义内部类。其实这种用法还真没见过,试试写个简单例子看看吧:
 
/** 
* 嵌套内部类 

* @author leizhimin 2009-7-17 17:33:48 
*/
 
public  class Outer { 
         private  void f0() { 
                System.out.println( "f0"); 
        } 

         class A { 
                 private  void a() { 
                        f0(); 
                        System.out.println( "a"); 
                } 

                 class B { 
                         protected  void b() { 
                                a(); 
                                System.out.println( "b"); 
                        } 
                } 
        } 

class Test{ 
         public  static  void main(String[] args) { 
                Outer o =  new Outer(); 
                Outer.A    a =     o. new A(); 
                Outer.A.B b = a. new B(); 
                b.b(); 
        } 
}
 
运行结果:
f0 



Process finished with exit code 0
 
八、内部类的继承
 
内部类的继承,可以继承内部类,也可以继承外部类。
/** 
* 内部类的继承,可以继承内部类,也可以继承外部类 

* @author leizhimin 2009-7-22 13:50:01 
*/
 
public  class Outer { 
         class Inner { 
                 void doSomething() { 
                        System.out.println( "Inner doing ..."); 
                } 
        } 

         class Inner2  extends Inner { 
                 void doSomething() { 
                        System.out.println( "Inner2 doing ..."); 
                } 

                 void readme() { 
                        System.out.println( "HeHe!"); 
                } 
        } 


class Test { 
         public  static  void main(String[] args) { 
                Outer outer =  new Outer(); 
                Outer.Inner in = outer. new Inner(); 
                Outer.Inner2 in2 = outer. new Inner2(); 
                in.doSomething(); 
                in2.doSomething(); 
                in2.readme(); 
        } 
}
 
运行结果:
Inner doing ... 
Inner2 doing ... 
HeHe! 

Process finished with exit code 0
 
 
总结:
 
内部类是Java中最复杂深奥的概念之一,而且内部类在访问控制,修饰符,继承,实现,抽象,序列化等等很多方面都是一个很让人迷惑的问题,在实际中,这些问题也许永远没机会没时间搞清,但是一般说来,懂得以上的内部类的知识就足够用了。
 
内部类的设计也许是弥补Java语言本身的先天不足吧,作为语言来说,这个特性太变态了点,难道就没别的法了?
 
以上的总结完全是建立在实践基础上的,所列举的例子也许偏颇,不能全面反映问题的本质,希望有兴趣的博友多多发表自己的看法与观点。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Java内部类总结 的相关文章

随机推荐

  • vscode搭建linux内核开发环境

    vscode在linux下搭建内核驱动开发环境 一 前言 Souce insight是一个阅读 开发linux内核驱动模块的好工具 但是Source insight是收费的软件 而且没有原生linux版本 要是想在纯linux环境下进行li
  • Unity3D RPG实现 3 —— 对话、任务系统

    目录 成果展示 对话系统 对话的存储数据结构 对话的UI面板设置 创建对话 任务的 NPC 实现对话控制器显示主对话窗口的内容 创建对话的选项内容 任务系统 创建任务 UI 面板 任务的存储数据结构 任务管理器与接受任务 任务控制相关脚本
  • Linux内核网络:实现与理论--介绍

    这本书主要涉及了Linux内核网络协议栈的实现和它背后的理论 你会在后续章节发现更深层次和更细节地针对网络子系统的分析和它的结构 我不会讨论和网络没有直接关系的话题内容 比如你在读内核里网络代码的时候会遇到锁 同步 SMP 原子操作等等 关
  • 2023年03月 C/C++(二级)真题解析#中国电子学会#全国青少年软件编程等级考试

    C C 编程 1 8级 全部真题 点这里 第1题 数字字符求和 请编写一个程序实现以下功能 从一个字符串中 提取出所有的数字字符即0 9 并作为数求和 时间限制 1000 内存限制 65536 输入 一行字符串 长度不超过100 字符串中不
  • facebook网络架构学习总结(F4)

    1 简介 1 1 大规模快速演进 Facebook 的生产网络本身就是一个大型分布式系统 针对不同任务划分成不同层次并采 用不同的技术 a large distributed system with specialized tiers an
  • mysql中如何设置时区_如何设置MySQL的时区?

    我认为这可能是有用的 有三个位置可以在MySQL中设置时区 在 mysqld 部分中的 my cnf 文件中default time zone 00 00 global Timezone变量 若要查看它们设置为什么值 请执行以下操作 SEL
  • 如何为Kafka集群选择合适的Topic/Partitions数量

    这是许多kafka使用者经常会问到的一个问题 本文的目的是介绍与本问题相关的一些重要决策因素 并提供一些简单的计算公式 越多的分区可以提供更高的吞吐量 首先我们需要明白以下事实 在kafka中 单个patition是kafka并行操作的最小
  • 2023华为OD机试真题【最左侧冗余覆盖子串/滑动窗口】

    题目描述 给定两个字符串 s1 和 s2 和正整数k 其中 s1 长度为 n1 s2 长度为 n2 在s2中选一个子串 满足 1 该子串长度为n1 k 2 该子串中包含s1中全部字母 3 该子串每个字母出现次数不小于s1中对应的字母 我们称
  • Element-UI 前端UI 组件库

    目录 Element UI 前端UI 组件库 配置Element UI 组件库 通过 UI 界面方式实现 配置插件 遇到的问题 Element UI 前端UI 组件库 个人博客地址 Element UI 前端UI 组件库 配置Element
  • Java代码连接数据库

    一 JDBC 全称 Java database connectivity java连接数据库 使用 Java 代码去连接数据库 然后对数据库中的数据进行增删改查 加载驱动 首先导入mysql的jar包 1 在 src 文件夹下面新建一个 l
  • 最长上升子序列(C语言 动态规划)

    描述 一个数的序列bi 当b1 lt b2 lt lt bS的时候 我们称这个序列是上升的 对于给定的一个序列 a1 a2 aN 我们可以得到一些上升的子序列 ai1 ai2 aiK 这里1 i1 lt i2 lt lt iK N 比如 对
  • linux设备利用率,linux – 对dm设备100%利用率的影响

    我们这里有一台RHEL 5 6服务器 有4条到单个LUN的活动路径 我们怀疑它无法将管道中的足够的IO塞进到另一端的XIV mpath0 XXXXXXXXXXXXXXX dm 9 IBM 2810XIV size 1 6T features
  • WARN Received a PartitionLeaderEpoch assignment for an epoch < latestEpoch

    Topic 处于 under replicated 状态 server log 充满 2020 11 30 19 00 00 006 WARN Received a PartitionLeaderEpoch assignment for a
  • MySQL---创建表和连表、分组、排序等操作

    首先这里有两张表做测试用 代码均在命令提示符终端完成 创建部门表 create table department table department id int auto increment primary key department v
  • TCP三次握手如果使用二次握手代替则会出现的问题

    简述 第一次握手 发送SYN报文 传达信息 你好 我想建立连接 第二次握手 回传SYN ACK报文 传达信息 好的 可以建立链接 第三次握手 回传ACK报文 传到信息 好的 我知道了 那我们连接 然后就建立连接了 在发送报文之前各方都要确认
  • Java基础——Math类、Random类、System类

    目录 1 Math类 2 Random类 3 System类 1 Math类 Math 类包含用于执行基本数学运算的方法 如初等指数 对数 平方根和三角函数 成员变量 public static final double E 自然底数 pu
  • java设计模式——代理模式(Proxy Pattern)

    概述 在某些情况下 一个客户不想或者不能直接引用一个对 象 此时可以通过一个称之为 代理 的第三者来实现 间接引用 代理对象可以在客户端和目标对象之间起到 中介的作用 并且可以通过代理对象去掉客户不能看到 的内容和服务或者添加客户需要的额外
  • 在vscode中创建vue自定义模板

    首先找到用户代码片段的位置 https img blog csdnimg cn 20201105094443610 png pic center 在输入框中输入vue 然后选择vue json或vue 打开之后是这样的 这些注释就是介绍怎么
  • SQL拼接存在的误区

    这两天修复一个SQL拼接的问题 给自己挖了个坑 对于指定的字符串作为查询条件时 当在SQL语句中使用单引号包裹查询条件时 需要注意以下几点 1 SQL注入攻击 如果接受外部输入的值直接拼接到SQL语句中 在查询条件中包含特殊字符如单引号或双
  • Java内部类总结

    Java内部类其实在J2EE编程中使用较少 不过在窗口应用编程中特别常见 主要用来事件的处理 其实 做非GUI编程 内部类完全可以不用 内部类的声明 访问控制等于外部类有所不同 要灵活使用内部类来编写程序 还是有相当难度的 Java发明了这