内部类(转载http://www.cnblogs.com/devinzhang/archive/2012/02/09/2344059.html)

Java内部类总结

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 

b

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中最复杂深奥的概念之一,而且内部类在访问控制,修饰符,继承,实现,抽象,序列化等等很多方面都是一个很让人迷惑的问题,在实际中,这些问题也许永远没机会没时间搞清,但是一般说来,懂得以上的内部类的知识就足够用了。

--------------------------------------------------------------------

PS: 欢迎关注公众号"Devin说",会不定期更新Java相关技术知识。

--------------------------------------------------------------------

请参见:http://lavasoft.blog.51cto.com/62575/179484/

时间: 2024-11-04 03:53:19

内部类(转载http://www.cnblogs.com/devinzhang/archive/2012/02/09/2344059.html)的相关文章

深入浅出之正则表达式(二)转载http://www.cnblogs.com/dragon/archive/2006/05/09/394923.html

深入浅出之正则表达式(二) 前言:       本文是前一片文章<深入浅出之正则表达式(一)>的续篇,在本文中讲述了正则表达式中的组与向后引用,先前向后查看,条件测试,单词边界,选择符等表达式及例子,并分析了正则引擎在执行匹配时的内部机理.        本文是Jan Goyvaerts为RegexBuddy写的教程的译文,版权归原作者所有,欢迎转载.但是为了尊重原作者和译者的劳动,请注明出处!谢谢!   9.      单词边界 元字符<<\b>>也是一种对位置进行匹

c# 和 java &lt;转载&gt;http://www.cnblogs.com/zhucai/archive/2011/02/16/csharp-compare-java.html

从语法层面来讲,c# 和 java 是非常相似的. <转载> 这篇文章对C#与Java做一个语言级的对比,方便C#转Java或Java转C#的人有个大致了解.这里大致用C#3.0与Java6.0做比较. 写完后得知维基百科里有更加全面得多的比较: http://en.wikipedia.org/wiki/Comparison_of_C_Sharp_and_Java .NET(C#)   Java   基本类型   基本类型 C#中有无符号数,Java没有. C#中有值类型,且可自己定义值类型的

z-index总结【转载http://www.cnblogs.com/mind/archive/2012/04/01/2198995.html】

元素位置重叠的背景常识 (x)html文档中的元素默认处于普通流(normal flow)中,也就是说其顺序由元素在文档中的先后位置决定,此时一般不会产生重叠(但指定负边距可能产生重叠).当我们用css为某个元素指定float浮动或者position定位后,元素的定位将会依情况发生如下改变: 1. 指定float值left/right 行内元素也会隐形变成块元素,元素会脱离文档的普通流,向左或右浮动,直到其外边缘碰到包含框或另一个浮动框. 2. 指定position值relative 可以相对于

C#:this索引器《转载http://www.cnblogs.com/ArmyShen/archive/2012/08/27/2659405.html》

索引器允许类或者结构的实例按照与数组相同的方式进行索引取值,索引器与属性类似,不同的是索引器的访问是带参的. 索引器和数组比较: (1)索引器的索引值(Index)类型不受限制 (2)索引器允许重载 (3)索引器不是一个变量 索引器和属性的不同点 (1)属性以名称来标识,索引器以函数形式标识 (2)索引器可以被重载,属性不可以 (3)索引器不能声明为static,属性可以 一个简单的索引器例子 using System; using System.Collections; public clas

Android实现推送方式解决方案【转载地址:http://www.cnblogs.com/hanyonglu/archive/2012/03/04/2378971.html】

本文介绍在Android中实现推送方式的基础知识及相关解决方案.推送功能在手机开发中应用的场景是越来起来了,不说别的,就我们手机上的新闻客户端就时不j时的推送过来新的消息,很方便的阅读最新的新闻信息.这种推送功能是好的一面,但是也会经常看到很多推送过来的垃圾信息,这就让我们感到厌烦了,关于这个我们就不能多说什么了,毕竟很多商家要做广告.本文就是来探讨下Android中实现推送功能的一些解决方案,也希望能够起到抛砖引玉的作用.^_^ 1.推送方式基础知识:  在移动互联网时代以前的手机,如果有事情

Android实现下拉导航选择菜单效果【转载地址:http://www.cnblogs.com/hanyonglu/archive/2012/07/31/2617488.html】

本文介绍在Android中如何实现下拉导航选择菜单效果.   关于下拉导航选择菜单效果在新闻客户端中用的比较多,当然也可以用在其他的项目中,这样可以很方便的选择更多的菜单.我们可以让我们的应用顶部有左右滑动或进行切换的导航菜单,也可以为了增强用户体验在应用中添加这样的下拉导航选择菜单效果. 关于它的实现原理,其实也是挺简单的,就是使用PopupWindow来进行展现,在显示时控制其高度并配置以相应的动画效果.在PopupWindow中我使用GridView来控制里面的菜单项,每个菜单项对应相应的

C#加密算法汇总(转载)http://www.cnblogs.com/zengxiangzhan/archive/2010/01/30/1659687.html

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 方法一:     //须添加对System.Web的引用     using System.Web.Security;          ...          /// <summary>     /// SHA1加密字符串     /// </summary>     /// <param name="source"

Android学习CursorWrapper与Decorator模式 (转至http://www.cnblogs.com/bastard/archive/2012/05/31/2527944.html)

一 Decorator模式 意图: 动态的给一个对象添加一些额外的职责.就增加功能来说,Decorator模式相比生成子类更为灵活. 动态的给一个对象,而不是对整个类添加额外职责,说明此模式将采用的结构是组合而不是继承: 要给一个对象添加职责,通常可以使用其类的派生类对象替换当前对象,但这显得不够灵活, 继承结构属于静态形式,系统往往并不知道有这样的类存在: 而我们需要的在某些时刻让对象具有一些额外的职责,额外的职责很可能是来自我们自己的扩展或者某些时刻的特定需求等. 于是通过一种方式对现有的对

Asp.Net页面生命周期【转载,地址:http://www.cnblogs.com/xhwy/archive/2012/05/20/2510178.html】

一.什么是Asp.Net页面生命周期 当我们在浏览器地址栏中输入网址,回车查看页面时,这时会向服务器端(IIS)发送一个request请求,服务器就会判断发送过来的请求页面,  完全识别 HTTP 页面处理程序类后,ASP.NET 运行时将调用处理程序的 ProcessRequest 方法来处理请求,来创建页面对象.通常情况下,无需更改此方法的实现,因为它是由 Page 类提供的.接下来被创建页面对象的ProcessRequest方法使页面经历了各个阶段:初始化.加载视图状态信息和回发数据.加载