Java内部类

将一个类的定义放在另一个类的定义的内部,这就是内部类。

1. 普通内部类

public class Example {
    int index = 0;
    public void increaseIndex() {
        index++;
    }
    public class InnerClass {
        private int getIndex() {
            increaseIndex();
            return index;
        }
    }
    
    public static void main(String[] args) {
        Example example = new Example();
        InnerClass innerClass = example.new InnerClass();
        innerClass.getIndex();
    }
}

先来看这个例子,innerClass类的定义在example类里面,内部类可以访问外部类的属性和方法,比如代码中内部类的getIndex方法中可以直接调用外部类的increaseIndex方法和index属性(第8和第9行)。为什么能这样呢?当某个外部类的对象创建了一个内部类对象时,此内部类对象必定会秘密地捕获一个指向那个外部类对象的引用。

但如果想创建内部类的实例,必须先创建外部类,再通过外部类实例的.new方法创建内部类。

2. 局部内部类和匿名内部类

如果在一个代码块或者一个函数中创建内部类,就是局部内部类。

public class Example {
    interface Destination {
        String location();
    }

    public Destination makeInner() {
        class MDestination implements Destination {
            private String dest;

            MDestination(String str) {
                this.dest = str;
            }

            @Override public String location() {
                return dest;
            }
        }
        return new MDestination("1");
    }
}

内部类MDestination的定义在方法makeInner()的内部,那么这个类就只能在方法内被初始化,在方法外的任何地方都不能使用。这类似于方法的局部变量。

还有另一种更简单的写法,匿名内部类:

public class Example {
    interface Destination {
        String location();
    }

    public Destination makeInner() {
        return new Destination() {
            @Override public String location() {
                return "1";
            }
        };
    }
}

makeInner方法里直接返回了new Detination(){},返回的类没有定义名称,所以称作匿名内部类。匿名内部类没有显式的名称,其实编译器在编译的时候给了一个默认的名称。

方法的内部类,一般情况下我们都不需要内部类的名称,但有些时候还是需要的:

1. 需要一个已命名的构造器,或者需要重载构造器。(匿名内部类没有名称,所以没有构造器)

2. 需要不止一个该内部类的对象。

3. 静态内部类(嵌套类)

如果不需要内部类对象与其外部类进行联系,可以将内部类声明为static。这通常称为嵌套类。

public class Example {
    interface Destination {
        String location();
    }

    public int index = 0;

    public int getIndex() {
        return index;
    }

    public static class MDestination implements Destination {
        public String dest;

        public MDestination(String str) {
            this.dest = str;
        }

        @Override public String location() {
            //getIndex();   error:can not access
            //index++;    error:can not access
            return dest;
        }
    }

    public static void main(String[] args) {
        MDestination destination = new MDestination("peking");
    }

}

与普通内部类不同的是,静态内部类不再能访问外部类的普通方法和普通变量(第20和第21行)。但如果外部类包含静态方法或者是静态变量,静态内部类中依然是可以访问的。

如果需要创建静态内部类的对象,不再需要先创建外部类的对象。

4. 为什么需要内部类?

外部类可以继承一个子类外加N个接口,为什么还需要内部类呢?因为Java里是单继承体系,内部类是为了解决多重继承的问题,有了内部类,就可以创建N个内部类分别实现不同的子类,去做不同的事情。而且Java语言的内部类和接口的实现方式,要比c++的直接多重继承要好。

public static abstract class BaseClass1 {
}

public abstract class BaseClass2 {
}

public class aaa extends BaseClass1 {
    public class bbb extends BaseClass2 {
    }
}

aaa类中继承自BaseClass1,需要做一些事情,但其中有些事情需要BaseClass2中的方法来做。那么我们就可以创建一个内部类bbb继承自BaseClass2,在内部类中做这部分事情,做好后再把结果回调给外部类。

总结:

内部类的实现看起来很诡异,一开始的感觉很奇怪。但作用却是很大的,像类的多重继承只能通过内部类来解决。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java测试题2》<br><br>一、 选择<br>1.欲构造ArrayList类的一个实例,此类继承了List接口,下列哪个方法是正确的 ? B<br>A ArrayList myList=new Object();<br>B List myList=new ArrayList();<br>C ArrayList myList=new List();<br>D List myList=new List();<br>2.paint()方法使用哪种类型的参数? A<br>A Graphics<br>B Graphics2D<br>C String<br>D Color<br>3.指出正确的表达式 D<br>A byte=128;<br>B Boolean=null;<br>C long l=0xfffL;<br>D double=0.9239d;<br>4.指出下列程序运行的结果 D<br>public class Example{<br>String str=new String("good");<br>char[]ch={'a','b','c'};<br>  public static void main(String args[]){<br>    Example ex=new Example();<br>    ex.change(ex.str,ex.ch);<br>    System.out.print(ex.str+" and ");<br>    Sytem.out.print(ex.ch);<br>  }<br>  public void change(String str,char ch[]){<br>    str="test ok";<br>    ch[0]='g';<br>  }<br>}<br>A good and abc<br>B good and gbc<br>C test ok and abc<br>D test ok and gbc<br>5.运行下列程序, 会产生什么结果 B<br>public class X extends Thread implements Runable{<br> public void run(){<br>  System.out.println("this is run()");<br> }<br> public static void main(String args[])<br> {<br>  Thread t=new Thread(new X());<br>  t.start();<br> }<br>}<br>A 第一行会产生编译错误B 第六行会产生编译错误C 第六行会产生运行错误D 程序会运行和启动6.要从文件" file.dat"文件中读出第10个字节到变量C中,下列哪个方法适合? A<br>A FileInputStream in=new FileInputStream("file.dat"); in.skip(9); int c=in.read();B FileInputStream in=new FileInputStream("file.dat"); in.skip(10); int c=in.read();C FileInputStream in=new FileInputStream("file.dat"); int c=in.read();D RandomAccessFile in=new RandomAccessFile("file.dat"); in.skip(9); int c=in.readByte();7.容器被重新设置大小后,哪种布局管理器的容器中的组件大小不随容器大小的变化而改变? B<br>A CardLayoutB FlowLayout<br>C BorderLayout<br>D GridLayout<br>8.给出下面代码: C<br>public class Person{<br>  static int arr[] = new int[10];<br><br>  public static void main(String a[])<br>  {<br>   System.out.println(arr[1]);<br>  }<br>}<br>那个语句是正确的?<br>A 编译时将产生错误;<br>B 编译时正确,运行时将产生错误;<br>C 输出零;<br>D 输出空。<br>9.哪个关键字可以对对象加互斥锁? B<br>A transient<br>B synchronized<br>C serialize<br>D static<br>10.下列哪些语句关于内存回收的说明是正确的? B<br>A 程序员必须创建一个线程来释放内存;<br>B 内存回收程序负责释放无用内存<br>C 内存回收程序允许程序员直接释放内存<br>D 内存回收程序可以在指定的时间释放内存对象<br>11.下列代码哪几行会出错: c<br>1) public void modify() {<br>2) int I, j, k;<br>3) I = 100;<br>4) while ( I > 0 ) {<br>5) j = I * 2;<br>6) System.out.println (" The value of j is " + j );<br>7) k = k + 1;<br>8) I--;<br>9) }<br>10) }<br>A line 4<br>B line 6<br>C line 7<br>D line 8<br>二、多项选择<br>1.执行下列代码后,哪个结论是正确的 String[] s=new String[10]; BD<br>A s[10] 为 "";<br>B s[9] 为 null;<br>C s[0] 为 未定义<br>D s.length 为10<br>2.下面的表达式哪个是正确的?AE<br>A String s="你好";int i=3; s+=i;<br>B String s="你好";int i=3; if(i==s){ s+=i};<br>C String s="你好";int i=3; s=i+s;<br>D String s="你好";int i=3; s=i+;<br>E. String s=null; int i=(s!=null)&&(s.length>0)?s.length():0;<br>3.选出合理的标识符AC<br>A _sys1_lll<br>B 2mail<br>C $change<br>D class<br>4.哪个布局管理器使用的是组件的最佳尺寸( preferred size) AE<br>A FlowLayout<br>B BorderLayout<br>C GridLayout<br>D CardLayout<br>E.GridBagLayout<br>5.下列哪个方法可用于创建一个可运行的类?AE<br>A public class X implements Runable{ public void run(){ ......} }<br>B public class X implements Thread{ public void run(){ ......} }<br>C public class X implements Thread{ public int run(){ ......} }<br>D public class X implements Runable{ protected void run(){ ......} }<br>E.public class X implements Thread{ public void run(){ ......} }<br>6.下面哪个方法可以在任何时候被任何线程调用?DEF<br>A notify()<br>B wait()<br>C notifyAll()<br>D sleep()<br>E.yield()<br>F.synchronized(this)<br>7.构造BufferedInputStream的合适参数是哪个? AC<br>A BufferedInputStream<br>B BufferedOutputStream<br>C FileInputStream<br>D FileOuterStream<br>E. File<br>8.下列说法正确的是 BC<br>A java.lang.Clonable是类<br>B java.lang.Runnable是接口<br>C Double对象java.lang包中<br>D Double a=1.0是正确的java语句<br>9.指出正确的表达式AB<br><br>B Double a=new Double(1.0);<br>C byte a = 340;<br>D Byte a = 120;<br>10.定义一个类名为"MyClass.java"的类,并且该类可被一个工程中的所有类访问,那么该类的正确声明应为:CD<br>A double a=1.0;<br>A private class MyClass extends Object<br>B class MyClass extends Object<br>C public class MyClass<br>D public class MyClass extends Object<br>11.指出下列哪个方法与方法public void add(int a){}为合理的重载方法。 CD<br><br><br>A public int add(int a)<br>B public void add(long a)<br>C public void add(int a,int b)<br>D public void add(float a)<br>12.如果下列的方法能够正常运行,在控制台上将显示什么? ACD<br><br>public void example(){<br>  try{<br>    unsafe();<br>    System.out.println("Test1");<br>    }<br>    catch(SafeException e)<br>    {System.out.println("Test 2");}<br>  finally{System.out.println("Test 3");}<br>  System.out.println("Test 4");<br>}<br>A Test 1<br>B Test 2<br>C Test 3<br>D Test 4<br>13.下列哪些情况可以终止当前线程的运行? ABD<br>A 抛出一个例外时。<br>B 当该线程调用sleep()方法时。<br>C 当创建一个新线程时。<br>D 当一个优先级高的线程进入就绪状态时。<br>三、 填空题<br>1.执行下列代码后的结果是什么? int x,a=2,b=3,c=4; x=++a+b+++c++;<br>2. 包包含了Collection的接口和类的API<br>.main方法的声明格式包括<br>4.下列程序中构造了一个SET并且调用其方法add(),输出结果是<br>public class A{<br>public int hashCode(){return 1;}<br>public Boolean equals(Object b){return true}<br>public static void main(String args[]){ Set set=new HashSet();<br>set.add(new A());<br>set.add(new A());<br>set.add(new A());<br>System.out.println(set.size());<br>}<br>}<br>5.下列程序的运行结果是<br>class A{<br>class Dog{<br>  private String name;<br>  private int age;<br>  public int step;<br>  Dog(String s,int a){<br>   name=s;<br>   age=a;<br>   step=0;<br>   }<br>  public void run(Dog fast){<br>  fast.step++;<br>  }<br>}<br><br>  public static void main (String args[]){<br>  A a=new A();<br>  Dog d=a.new Dog("Tom",3);<br>  d.step=25;<br>  d.run(d);<br>  System.out.println(d.step);<br>  }<br>}<br><br>答案::填空第1题<br>x=10,a=3,b=4,c=5<br>填空第2题<br>java.util<br>填空第3题<br>(public )(static )(void)(main)(String args[])<br>填空第4题<br>1<br>填空第5题<br>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值