Java program idiom之 容器

Java常用的集合类型有 List Set Map ,容器一般也有线程安全与非线程安全之分, java.util 包下的线程安全类一般都是废弃不用的,应该使用 concurrent 包下面的线程安全类。

 

 

1.1.  容器的一般使用

A、 声明容器时要指定泛型的类型,可避免编译器警告和强类型转换。

 

 

B、 如果一个方法的返回一个容器,不要返回null ,应该返回空的容器以避免空指针异常。 Collections 工具类定义了一些静态的字段:EMPTY_LIST EMPTY_MAP EMPTY_SET ,分别表示不可变的空的 list map set,可使用这些静态字段,而不是自己实例化一个。

 

 

C、 声明容器类型时尽量使用接口:List Map Set ,而不是具体的实现类。 这样可以方便更改具体的实现类,特别是方法的参数类型和返回类型,如果使用具体的实现类,则几乎是不可能进行更改的,因为每个调用的地方都需要进行修改。如果使用接口,则只需在方法内部或方法调用的地方更改就行了,把改动的范围最小化。所谓接口隔离原则也就是这个意思。

 

 

D、 如果容器接受初始容量,都应该指定一个尽可能准确的初始容量,特别是容器是以数组作为底层存储结构的。

 

 

1.2.  Collections工具类

Collections工具类提供了很多使用的方法。可完成的功能有:查找、拷贝、排序、查找最大最小元素、反转列表、轮换列表元素、将容器封装为线程安全的或不可变的。务必要记住这个类,对容器进行操作时,现在 API 里查找此类是否已经有满足要求的方法。

 

 

1.3.  集合的交、差、并集

Collection接口定义了 retainAll removeAll addAll  方法用于生成两个集合的交、差、并集。

 

 

1.4.  数组与列表的互转换

数组到列表的转换可以使用:Arrays.asList  方法。

列表到数组的转换可以直接调用:toArray  方法。

public class InterfaceSep {
	public static void main(String[] args) {
		List<String> source = Arrays.asList("123", "456", "789", "abc");
		List<String> list = new LinkedList<String>(Arrays.asList("789", "xyz"));
		
		list.retainAll(source);
		System.out.println("retainAll :" + list);	// 交集
		
		list.addAll(source);		// 并集
		System.out.println("" + list);
		
		list.add("rest");
		list.removeAll(source);		// 差集
		System.out.println(list);
		
		ArrayList<String> arrayList = new ArrayList<String>(source);
		LinkedList<String> linkedList = new LinkedList<String>(source);
		
		concrete(arrayList);
//		concrete(linkedList);	// compile error
		
		abstrac(arrayList);
		abstrac(linkedList);
	}

	public static void concrete(ArrayList<String> list) {
		for (String string : list) {
			System.out.println(string);
		}
	}

	public static void abstrac(List<String> list) {
		for (String string : list) {
			System.out.println(string);
		}
	}
}
 

 

 

 

1.5.  Map

JDK Map  是一种 key value 的映射结构,提供根据 key 快速查找 value 的能力,一般以 Map.Entry 的结构存储。常用的 Map 实现有 HashMap LinkedHashMap

 

HashMap的底层存储结构是用数组(槽)  链表的形式,存储时根据 key 的值映射到数组的某个位置(槽位),如果有多个 key 映射到相同的槽位,则这些元素以链表的形式存放。

 

LinkedHashMap是采用双向链表,具有可预知的迭代顺序,一般就是插入顺序。

 

 

使用:

A、 在需要特定的迭代顺序,使用LinkedHashMap ;否则使用 HashMap

B、 迭代Map 时,如果通过 keySet 时,会有一个根据 key 查找 value 的开销。应该使用 entrySet

C、 使用Map 时,应该显式指定 key value 的类型,避免强类型转换。

D、 使用HashMap ,因为它底层使用了数组,如果可能,应该尽量给数组预分配合适的大小。

 

 

 

Map<String, Integer> map = new HashMap<String, Integer>(8);

for (Entry<String, Integer> entry : map.entrySet()) {
	System.out.println(entry.getKey() + "->" + entry.getValue());
}

for (String key : map.keySet()) {
	Integer value = map.get(key);
	System.out.println(key + "->" + value);
}

 

 

对于容器这种标准的类库,要熟悉其API ,避免自己蹩脚地实现已有的功能。

 

你可能感兴趣的:(java)