加载中…
个人资料
一叶知秋
一叶知秋
  • 博客等级:
  • 博客积分:0
  • 博客访问:424,048
  • 关注人气:83
  • 获赠金笔:0支
  • 赠出金笔:0支
  • 荣誉徽章:
相关博文
推荐博文
谁看过这篇博文
加载中…
正文 字体大小:

Java内部类总结

(2012-08-21 17:07:02)
标签:

java

内部类

it

分类: java技术

Java内部类总结

 

Java内部类其实在J2EE编程中使用较少,不过在窗口应用编程中特别常见,主要用来事件的处理。其实,做非GUI编程,内部类完全可以不用。

 

内部类的声明、访问控制等于外部类有所不同,要灵活使用内部类来编写程序,还是有相当难度的,Java发明了这种难懂的玩意儿,在其他语言中是没有的,但是在Java中,内部类也相当的重要,尤其GUI开发时候,事件的响应处理全靠内部类了。

 

内部类所做的功能使用外部类也同样可以实现,只是有时候内部类做的更巧妙些。

 

内部类按照其所在位置不同,可分为以下几种:

1、(普通的)内部类(最常见的内部类,内部类的定义与类成员平级,)
2
、方法内部类
3
、匿名类
4
、静态内部类
5
、接口内部类

 

一、内部类声明与访问

 

1、内部类直接在类的内部进行声明。可以声明为privateprotectedpublic或者默认访问权限,这个访问权限约定和外部类完全一样。

 

2、内部类自动拥有对其外围类所有成员(方法、属性)的访问权。如果内部类和外部类成员的名字完全相同,在内部类方法中要访问外部类成员,则需要使用下面的方式来访问外部类名.this.外部成员名,例如Outer.this.i++;  (看例子)

 

3必须使用外部类对象来创建内部类对象,而不是直接去new一个。

格式为:外部对象名.new 内部类构造方法

 

比如要创建一个内部类iner对象,需要这么做:

        Outer outer = new Outer();
        Outer.Inner iner = outer.new Inner();

 


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!");
        }
       
//创建内部类Inner
        
class Inner {
                
int i = 1000;

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

                
void innerMsg() {
                        System.out.println(
">>>>>Inner class!");

                 //访问外部类的方法
                        sayMsg();
                        
//访问内部类自己的成员i,也可以写成 this.i++
                        
this.i++;
                        
//访问外部类的成员 iy
                        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();
}

 


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

 


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();
        }
}

 

四、方法内部类

 

方法内部类只在该方法内部可见,方法内部类可以定义在方法中的任何位置。


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

 

五、匿名类

 

匿名类不给出类名,直接定义一个类,通常这个类实现了某种接口或者抽象。匿名类的访问权限更没有讨论价值了,看个例子就行了。

 

在一些多线程程序中比较常见,有点变态,呵呵。


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

 


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 外部类名.内部类构造方法

来创建,给个例子:


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>

 

下面我给个例子,


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();
        }
}

 

八、内部的类的嵌套

 

所谓内部类嵌套,就是内部类里面再定义内部类。其实这种用法还真没见过,试试写个简单例子看看吧:

 


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

 

八、内部类的继承

 

内部类的继承,可以继承内部类,也可以继承外部类。


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

 

0

阅读 评论 收藏 转载 喜欢 打印举报/Report
  • 评论加载中,请稍候...
发评论

    发评论

    以上网友发言只代表其个人观点,不代表新浪网的观点或立场。

      

    新浪BLOG意见反馈留言板 电话:4000520066 提示音后按1键(按当地市话标准计费) 欢迎批评指正

    新浪简介 | About Sina | 广告服务 | 联系我们 | 招聘信息 | 网站律师 | SINA English | 会员注册 | 产品答疑

    新浪公司 版权所有