博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java内部类总结
阅读量:6468 次
发布时间:2019-06-23

本文共 8787 字,大约阅读时间需要 29 分钟。

 
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语言本身的先天不足吧,作为语言来说,这个特性太变态了点,难道就没别的法了?
 
以上的总结完全是建立在实践基础上的,所列举的例子也许偏颇,不能全面反映问题的本质,希望有兴趣的博友多多发表自己的看法与观点。
 
本文转自 leizhimin 51CTO博客,原文链接:http://blog.51cto.com/lavasoft/179484,如需转载请自行联系原作者
你可能感兴趣的文章
spring boot configuration annotation processor not found in classpath问题解决
查看>>
由中序遍历和后序遍历求前序遍历
查看>>
我学习参考的网址
查看>>
[Processing]点到线段的最小距离
查看>>
考研随笔2
查看>>
GitHub使用教程、注册与安装
查看>>
<<The C Programming Language>>讀書筆記
查看>>
git代码冲突
查看>>
解析查询 queryString 请求参数的函数
查看>>
学生选课系统数据存文件
查看>>
git bash 风格调整
查看>>
linux 脚本map,Linux Shell Map的用法详解
查看>>
linux操作系统加固软件,系统安全:教你Linux操作系统的安全加固
查看>>
linux中yum源安装dhcp,24.Linux系统下动态网络源部署方法(dhcpd)
查看>>
C#技术------垃圾回收机制(GC)
查看>>
漫谈并发编程(三):共享受限资源
查看>>
【转】github如何删除一个仓库
查看>>
状态模式
查看>>
HDOJ-1010 Tempter of the Bone
查看>>
JavaNIO基础02-缓存区基础
查看>>