类型安全的List
Java的一个巨大优势是其类型安全性。在Java中,除了基本类型以外的所有事物都是一个对象,而且这个对象或显示,或隐式地继承自Object类,这给Java带来了巨大的类型安全性。任何方法的参数都必须具有某一特定类型,调用该方法时,给定的实参的类型要么是该类型本身,要么可以转换(映射)为该类型,否则就会产生编译时错误,这避免了很多类型不匹配带来运行时错误的情况发生。然而,Java的类型安全性一直以来有一个严重的漏洞,这就是集合中的元素的数据类型。不信的话,看看下面的例子:
package com.jiang.tiger.chap1;
import java.util.ArrayList;
import java.util.List;
public class ListTester {
public static void safelessList() {
List list = new ArrayList();
int number = 1010 ;
list.add( " good " );
list.add( " ok " );
list.add( " nice " );
list.add(number);
list.add( " wrong " );
int length = list.size();
String value = null ;
for ( int i = 0 ; i < length; i ++ ) {
value = (String) list.get(i);
System.out.println(value);
}
}
/** */ /**
* @param args
*/
public static void main(String[] args) {
// TODO 自动生成方法存根
safelessList();
}
}
在上例中,本来我是想得到一个字符串列表的,然而一不小心我加入了一个int类型的数据,很可惜,由于Java 5.0以前的集合无法保证类型安全性,因此上面的代码编译正确,然而运行时就会出错:
good
ok
nice
Exception in thread " main " java.lang.ClassCastException: java.lang.Integer
at com.jiang.tiger.chap1.ListTester.safelessList(ListTester.java: 18 )
at com.jiang.tiger.chap1.ListTester.main(ListTester.java: 28 )
上面的代码还有一个恼人的问题:每次我从List中获得数据时,虽然我知道获得的肯定是字符串,然而我每次仍然得进行一次映射将取得的数据转换为字符串。没办法,谁让List不是类型安全的呢,它压根就不知道我往里面放置的是String,它唯一知道的就是它里面放置了很多Object。
在Java 5.0中,这一漏洞已经被完全堵上了,因为Java 5.0引入了泛型这一令人激动万分的新特性。在Java 5.0中,我们可以指定List只能接受特定类型的数据,而不是接受所有的Object。我们修改上面的例子,用List<String>替换掉List,编译时就会出错,我们就知道,我们往List中插入了不合适的数据,我们就有可能在产品交付之前发现问题,而且恼人的映射也不需要了。在Java 5.0中,采用的是泛型技术来实现这些的。所谓泛型,如果初次结束,可以简单地理解为一个占位符,放在那儿告诉别人,这个位置是有人(类型)的,你可以安排一个人(类型)过来。至于这个人(类型)是男是女(具体的类型),是高是矮无所谓,完全由你决定。那么这个人什么时候会来呢?答案就是你使用占位符的时候,他就会来,而且是由你指定的。
可见,泛型不仅带来了类型安全性,还给我们编程带来了灵活性。唯一比较麻烦的是往List中插入基本类型数据,由于为List指定类型只能是引用类型,因此无法给List指定基本类型。如果要用List存放基本类型数据,必须指定该类型对应的引用类型,例如下例:
package com.jiang.tiger.chap1;
import java.util.ArrayList;
import java.util.List;
public class ListTester {
public static void safefulList() {
List < String > list = new ArrayList < String > ();
int number = 1010 ;
list.add( " good " );
list.add( " ok " );
list.add( " nice " );
// 下面注释的代码编译会出错
// list.add(number);
list.add( " wrong " );
int length = list.size();
String value = null ;
for ( int i = 0 ; i < length; i ++ ) {
value = list.get(i);
System.out.println(value);
}
}
public static void intList() {
// 给List指定类型只能是引用类型,而不能是基本类型
// List<int> list = new ArrayList<int>();
// 如果要往List中插入基本类型,可以给List指定其对应的引用类型
List < Integer > list = new ArrayList < Integer > ();
list.add( 12 );
list.add( 23 );
list.add( 100 );
int value = 0 ;
int length = list.size();
for ( int i = 0 ; i < length; i ++ ) {
value = list.get(i);
System.out.println(value);
}
}
/** */ /**
* @param args
*/
public static void main(String[] args) {
// TODO 自动生成方法存根
safefulList();
intList();
}
}
上例的运行结果我就不给出了,太简单了,估计你还没看完程序就已经知道结果了。需要说明的是,当采用引用类型制定类型,而插入的是基本类型数据时,会带来额外的开销,这就是装箱和解箱的开销(boxing and unboxing)。泛型是一门艺术,需慢慢琢磨方能体会它的美。
分享到:
相关推荐
1. 为什么要使用泛型? 在 Java 中,如果我们想要将一个字符串类型的值放入到集合中,这个时候,这个值放到集合之后,失去本身的类型,只能是 object 类型。如果想要对这个值进行类型转换,很容易出现类型转换错误...
1. **为什么要使用泛型?** 在没有泛型的情况下,集合只能存储Object类型,当我们从集合中取出元素时,需要进行类型转换,这可能导致`ClassCastException`。泛型的引入解决了这个问题,它允许我们在创建集合时指定...
##### 2.2 为什么使用泛型? 1. **类型安全**:泛型可以在编译阶段就检测到类型错误,避免了运行时的`ClassCastException`异常。 2. **代码复用**:通过参数化类型,可以编写更加通用的代码,减少代码重复。 3. **...
首先,我们需要理解什么是泛型。泛型允许我们在定义类、接口、委托和方法时,不指定具体的类型参数,而是使用一个占位符,如`T`、`K`等。这样,当我们实例化这些泛型类型时,可以传入任何类型的参数,从而实现泛型...
通过代码进行理解java的泛型,以及为什么要有泛型
Java编程语言是面向对象的、高性能的编程语言,被广泛应用于企业级应用开发。...在实际开发中,这两个概念经常结合使用,比如在处理文件列表时,我们可以使用泛型集合配合File类来高效、安全地管理文件操作。
例如,我们可以创建一个名为`GenericContainer<T>`的泛型类,其中`T`就是类型参数。这个类可以存储任何类型的对象,但具体类型在实例化时确定。这样做的好处是,我们无需为每种可能的数据类型创建单独的类,从而提高...
例如,可以使用`where T : class`约束确保`T`必须为引用类型,或者`where T : struct`约束确保`T`为值类型。此外,还可以使用`where T : IComparable<T>`这样的接口约束,确保`T`实现了特定接口。 3. **泛型列表...
Net程序员面试题 面向对象 引用类型和值类型区别 泛型特点,为什么要使用泛型? ..............................
- **为什么要使用泛型?** 泛型的主要优点在于避免了类型转换的麻烦,同时通过在编译时进行类型检查,提高了代码的安全性。 - **创建和使用泛型** 泛型可以应用于类、接口、方法等,如`List<T>`就是一种常见的泛型...
- **为什么要使用泛型?** 泛型的主要目的是提供更高效的数据结构和方法,同时保持类型安全,避免了不必要的类型转换。 - **创建和使用泛型** 包括定义泛型类、接口和方法,以及如何实例化和使用它们。 - **类型...
### 为什么要使用泛型集合? 在C# 2.0之前的版本,创建集合主要有两种方式: 1. **ArrayList**:允许直接存储各种类型的对象,但由于集合内的元素都是Object类型,每次访问都需要进行类型转换,这不仅增加了代码的...
#### 为什么要使用泛型? 在C# 2.0之前,C#并不支持泛型,这导致了代码复用性不高且存在类型安全问题。例如,当需要实现一个数据结构如列表时,为了适应不同类型的元素,可能需要为每种类型都实现一个列表类,这...
泛型使得我们可以指定容器存储的数据类型,从而避免了不必要的类型转换和可能的ClassCastException。 8. 野指针异常 在泛型中,如果尝试向泛型容器添加不匹配的类型,编译器会报错。然而,通过反射或构造旧版本API...
#### 什么是泛型? 泛型是Java SE 5.0中引入的一项新特性,它允许在程序设计中使用参数化类型,从而实现类型安全的集合类和其他容器类。在使用泛型之前,Java程序员通常依赖于Object类型作为通用容器的数据类型,...
使用泛型集合维护有继承关系的电影票对象 座位数可以在系统中指定 点击座位可购买不同类型的电影票 使用简单工厂模式创建不同类型的电影票 使用多态实现不同类型电影票的价格计算 使用接口实现电影票的打印 使用序列...
下面我们将深入探讨Java泛型方法的概念、语法以及使用示例。 **一、泛型方法概念** 泛型方法是一种具有类型参数的方法,这些类型参数可以在方法声明时指定,并在方法体内部使用。与类的泛型类似,它们提供了编译时...
1. **为什么要使用泛型?** 泛型的主要目标是提高代码的类型安全性和性能。通过泛型,开发者可以创建通用的容器类(如List),这些容器可以在运行时容纳任何类型的对象,同时保证类型检查在编译时完成,避免了不必要...
- **19.1.1 为什么要使用泛型?** 使用泛型可以避免编写重复代码,提高代码的复用性和可读性。此外,泛型还提供了类型安全性,减少了强制类型转换的需求。 - **19.1.2 创建和使用泛型** 泛型可以通过在类、接口...