一、命名内部类——用于临时创建实例或者类型,且该类型或实例需要控制是否允许使用该类的人调用
1-1:命名局部内部类
public class OutClass{
public int f()
{
private class InnerClass
{
int i=10;
}
return new InnerClass().i;
}
public static void main(String[] args)
{
OutClass o=new OutClass();
System.out.println(o.f());
}
}
public class OutClass{
public class InnerClass{
int i=10;
}
public int f()
{
return new InnerClass().i;
}
public static void main(String[] args)
{
OutClass o=new OutClass();
System.out.println(o.f());
}
}
此例将简单内部类提升至成员等级,在此等级中,我们可以自由设定该内部类是允许OutClass使用者实例化InnerClass实例,如果为public 类型,则允许实例化,如果为private 类型,则不允许实例化(除开OutClass的成员方法将InnerClass类型的实例作为返回值返回)
1-3:命名内部类的实例化方法
public class OutClass{
public class InnerClass{
public InnerClass()
{
System.out.println("InnerClass construction start");
}
}
//public static void test()
//{
// InnerClass ic=new InnerClass();//错误的实例化内部类
//}
//public static InnerClass ic=new InnerClass();//错误的实例化内部类
public InnerClass ic=new InnerClass();
public InnerClass test()
{
return new InnerClass();
}
public static void main (String[] args)
{
OutClass oc=new OutClass();
OutClass.InnerClass ic=oc.new InnerClass();
}
}
注意:此处的输出结果为:
InnerClass construction start
InnerClass construction start
原因是外部类实例化时,需要产生一个内部类InnerClass成员,因此会调用内部类的构造函数,所以会有第一条
InnerClass construction start
而当在主方法中实例化内部类时,也需要调用内部类的构造函数,因此产生了第二条
InnerClass construction start
二、匿名内部类——用于临时创建实例,且该实例不希望类的使用者关注
匿名内部类的存在前提有两种:1、已知该类型的父类 或者 2、已知该类型的接口标准
通常对于一个临时的实例,我们并不希望他被使用者关注,比如一个人类Person,我们在处理一些他的特例的时候——比如BlindMan类型的人,而这种类型并不常见,所以我们可以不需要实体创建BlindMan这个类型,而直接使用它的具体实例:
1-1:已知父类的匿名内部类
public class Test
{
public void see()
{
Person p=new Person()
{
name="I";
void r(){
System.out.println(this.name+" can see nothing.");
}
};
p.r();
}
}
注意:由于此处的匿名类是创建p指针的具体膜板,必须在{}外添加";"号,表示是创建Person 实例的语句结束。
1-2:已知接口标准的内部类
interface IFly{
void fly();
}
class Plan{
public void run()
{
IFly f=new IFly()//使用匿名类创建飞机实例
{
public void fly()
{
System.out.println("Plan is flying");
}
};
f.fly();
}
}