`

java内部类

    博客分类:
  • java
阅读更多
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
a
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中最复杂深奥的概念之一,而且内部类在访问控制,修饰符,继承,实现,抽象,序列化等等很多方面都是一个很让人迷惑的问题,在实际中,这些问题也许永远没机会没时间搞清,但是一般说来,懂得以上的内部类的知识就足够用了。
 
内部类的设计也许是弥补Java语言本身的先天不足吧,作为语言来说,这个特性太变态了点,难道就没别的法了?
 
以上的总结完全是建立在实践基础上的,所列举的例子也许偏颇,不能全面反映问题的本质,希望有兴趣的博友多多发表自己的看法与观点。
分享到:
评论

相关推荐

    java内部类详解

    Java 内部类详解 Java 内部类是一种高级特性,允许在一个类的定义内部创建另一个类。这种设计模式提供了更灵活的代码组织方式,同时还可以实现特定的封装和访问控制。内部类主要分为四种类型:静态内部类、成员内部...

    Java内部类总结

    ### Java内部类总结 在Java编程语言中,内部类是一个重要的概念,它允许开发者在一个类的内部定义另一个类。这种特性极大地增强了代码的封装性和复用性,同时也为解决特定问题提供了灵活的方法。本文将围绕Java内部...

    java内部类的讲解

    ### Java内部类详解 #### 一、内部类的分类与概念 Java的内部类机制是其强大特性之一,它允许类作为另一个类的成员存在,从而增强了代码的封装性和复用性。根据定义和作用域的不同,Java内部类主要分为四类: 1. ...

    Java 内部类

    Java 内部类是Java语言特性中的一个重要组成部分,它允许我们在一个类的内部定义另一个类。内部类可以访问外部类的所有成员,包括私有成员,这使得内部类在实现复杂逻辑和封装上具有很大的灵活性。下面我们将详细...

    浅谈Java内部类的四个应用场景

    ### Java内部类的应用场景 #### 场景一:当某个类除了它的外部类,不再被其他的类使用时 内部类的使用场景之一是当某个类仅需被其外部类使用时。这种情况下,将此类定义为内部类可以提高代码的封装性和可维护性。 ...

    java内部类使用例子

    Java内部类是Java语言提供的一种独特特性,它允许我们在一个类的定义内部定义另一个类。这种内部类可以是成员内部类、局部内部类、匿名内部类或静态内部类,每种都有其特定的用途和使用场景。在这个"java内部类使用...

    12.java内部类.zip

    12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类.zip12.java内部类...

    java 内部类应用

    Java内部类是Java语言的一个特色特性,它允许我们在一个类的内部定义另一个类。这种设计模式在处理一些特定情况时非常有用,例如实现匿名回调、封装特定逻辑或创建与外部类有紧密关系的类。本篇文章将深入探讨Java...

    JAVA 内部类 PPT

    Java内部类是Java语言的一个独特特性,它允许我们在一个类的内部定义另一个类。这种设计提供了更高级别的封装和组织代码的方式。以下是关于内部类及其相关知识点的详细说明: 1. **内部类基本语法** - **实例内部...

    JAVA内部类总结

    ### JAVA内部类总结 在Java编程语言中,内部类(Inner Classes)是一种非常重要的特性,它允许我们在一个类的内部定义另一个类。这种结构不仅能够提高代码的组织性,还能帮助我们更好地处理类与类之间的关系。根据...

    java 内部类使用(内部匿名类)

    Java内部类是Java语言提供的一种独特机制,它允许在一个类的内部定义另一个类。这种设计模式使得代码结构更紧凑,可以更好地封装和隐藏实现细节,同时也增强了代码的复用性。内部类分为几种类型,包括成员内部类、...

    java内部类总结(含概念和实例)

    Java 内部类总结 Java 内部类是指在一个外部类的内部再定义一个类。内部类作为外部类的一个成员,并且依附于外部类而存在的。内部类可为静态,可用 protected 和 private 修饰(而外部类只能使用 public 和缺省的包...

    java 内部类 局部内部类 匿名类 实例代码

    Java内部类是Java语言特性中一个独特而强大的部分,它允许在一个类的内部定义另一个类。内部类可以访问外部类的所有成员,包括私有成员,这使得内部类在实现某些特定设计模式时非常有用。本篇文章将深入探讨Java中的...

    Java语法总结 - 内部类

    Java内部类详解 Java内部类是Java语言中的一种特殊类别,它是指定义在另外一个类内部的类。内部类可以访问外部类的所有成员变量和方法,包括私有的变量和方法。内部类可以分为四种:成员内部类、静态嵌套类、方法...

    Java内部类的四个应用场景

    谈Java内部类的四个应用场景

    java内部类使用总结

    Java内部类是Java语言中一个独特且强大的特性,它允许我们在一个类的内部定义另一个类。内部类提供了增强封装的能力,使得内部类可以被隐藏在外部类中,仅对外部类可见,从而增加了代码的隐私性和安全性。同时,内部...

    Java内部类的作用

    ### Java内部类的作用 #### 一、定义及概念 内部类,顾名思义,指的是一个定义在另一个类内部的类。这样的结构让内部类能够访问外部类的成员(包括私有成员),并且允许内部类拥有不同的访问级别,如`private`、`...

    Java内部类_动力节点Java学院整理

    Java内部类是Java语言的一个重要特性,它允许我们在一个类的内部定义另一个类。这种设计模式使得代码结构更加紧凑,同时也提供了更高级的封装和抽象。内部类主要有两种类型:成员内部类和局部内部类。 1. 成员内部...

Global site tag (gtag.js) - Google Analytics