Java List和ArrayList的区别

 List是一个接口,而ArrayList是List接口的一个实现类。 

ArrayList类继承并实现了List接口。 

因此,List接口不能被构造,也就是我们说的不能创建实例对象,但是我们可以像下面那样为List接口创建一个指向自己的对象引用,而ArrayList实现类的实例对象就在这充当了这个指向List接口的对象引用。 

个人见解:

要是你已经学过了OOP,上面的部分是不难理解的,这是面向对象重要的知识点,面向对象最重要的就是多态,
我们都知道接口和抽象不能被实例化,但是它们可以创建一个指向自己的对象引用,它们的实现类或子类就在充当这
样的角色,我想这就是面向对象编程中多态的优势。前些日子在学习UML建模语言和Java设计模式的时候,深深地的
体会到了面向对象编程的好处,Java集合框架中用到也不足为奇,Java本身就是面向对象的编程语言。

上面的理解可能有点难度,但是我们找一个具体的实例,就会理解起来比较容易。我们定义一个动物的抽象类
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("狗有看门的独特本领!");  
    }  
}

 List list;//正确,list = null; 

List list = new List();//是错误的用法

List list = new ArrayList(); 这句创建了一个ArrayList实现类的对象后把它上溯到了List接口。此时它就是一个List对象了,它有些ArrayList类具有的,但是List接口没有的属性和方法,它就不能再用了。

而ArrayList list=newArrayList(); 创建对象则保留了ArrayList所有的属性和方法。 

个人见解

我们继续上面的例子,如果我们创建的是抽象类的对象引用,那么这个对象只能调用自己的非抽象方法,
下面的是shout()方法,不能调用继承它的子类的独有的方法,在下面的就是guard()方法不能被a1调用,继续测试代码:

public class Test{  
    public static void main(String[] args){  
        Animal a1 = new Dog();  
        a1.shout();//编译通过  
        //a1.guard();//编译出错  
    }  
}  
  
abstract class Animal{  
    //动物名字  
    String name;  
      
    //动物叫声  
    public void shout(){  
        System.out.println("叫声...");  
    }  
}  
  
class Dog extends Animal{  
    //狗类独有的方法  
    public void guard(){  
        System.out.println("狗有看门的独特本领!");  
    }  
}

如果我们采用Dog d1 = new Dog();那么d1可以调用抽象类和子类的所有属性和方法,这里不再测试。

另外一个例子:

List接口中并没有trimToSize()方法,但这个方法在它的实现类ArrayList中有。

public class Demo{  
    public static void main(String[] args){  
        List list = new ArrayList();   
                ArrayList arrayList = new ArrayList();  
                list.trimToSize();//错误,没有该方法。  
                arrayList.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() 呢? 

1,问题就在于List接口有多个实现类,现在你用的是ArrayList,也许哪一天你需要换成其它的实现类,如LinkedList或者Vector等等,这时你只要改变这一行就行了: List list = new LinkedList(); 其它使用了list地方的代码根本不需要改动。 假设你开始用ArrayList alist = new ArrayList(), 这下你有的改了,特别是如果你使用了ArrayList实现类特有的方法和属性。

2,List接口里面的方法以及足够平时开发使用,没必要再使用ArrayList() 里面的特殊方法。

3,如果特殊业务需要使用ArrayList()里面的特殊方法,就可以用 ArrayList alist = new ArrayList() 。

 

另外的例子就是,在类的方法中,如下声明:

private void doMyAction(List
   // TODO
}

这样这个方法能处理所有实现了List接口的类,一定程度上实现了泛型函数与多肽。