我们定义一个动物的抽象类Animal,再定义一个继承自Animal基类的Dog类,看下面的代码就会理解抽象类和接口不能被实例化:
public class Test{
public static void main(String[] args){
Animal a1 = new Animal();//编译出错
Animal a2 = new Dog(); //多态
}
}
abstract class Animal{
//动物名字
String name;
//动物叫声
public void shout(){
System.out.println("叫声...");
}
}
class Dog extends Animal{
//狗类独有的方法
public void guard(){
System.out.println("狗有看门的独特本领!");
}
如果我们创建的是抽象类的对象引用,那么这个对象只能调用自己的非抽象方法。 Animal a2 = new Dog();就是利用了多态,而List list = new ArrayList();与这相似。
List list = new ArrayList();这句创建了一个ArrayList实现类的对象后把它上溯到了List接口。此时它就是一个List对
象了,它有些ArrayList类具有的,但是List接口没有的属性和方法,它就不能再用了。 而ArrayList list=newArrayList();
创建一对象则保留了ArrayList的所有属性和方法。
再举一例子
import java.util.*;
public class Demo{
public static void main(String[] args){
List list = new ArrayList();
ArrayList arrayList = new ArrayList();
list.trimToSize();//错误,没有该方法。
arrayList.trimToSize();//ArrayList里有该方法。
}
可以看到List接口中并没有trimToSize()方法,但这个方法在它的实现类ArrayList中有。
如果是下面这个样子的:
List a=new ArrayList();
则a拥有List的所有属性和方法,不会拥有其实现类ArrayList的独有的属性和方法。
如果List与ArrayList中有相同的属性(如int i),有相同的方法(如void f()),
则a.i是调用了List中的i
a.f()是调用了ArrayList中的f();
那么问题的关键:
为什么要用 List list = new ArrayList() ,而不用 ArrayList alist = new ArrayList()呢?
问题就在于List接口有多个实现类,现在你用的是ArrayList,也许哪一天你需要换成其它的实现类,如
LinkedList或者Vector等等,这时你只要改变这一行就行了: List list = new LinkedList(); 其它使用了list地方的代码根
本不需要改动。
假设你开始用ArrayList alist = new ArrayList(), 这下你有的改了,特别是如果你使用了ArrayList实现类特有的方法
和属性。
(如有错误,欢迎指正)