黑马程序员 Java集合框架 Map和泛型


------- android培训java培训、期待与您交流! ----------

Map集合

1.特点:该集合存储键值对。一对一对往里存。而且要保证键的唯一性。

2.常用方法:

   1.添加:v  put(k key , v value); 将数据以键值对的方式存进集合中。

          void putAll(Map<? extends k,? extends v> m) ;从指定映射中将所有映射关系复制到此映射中

   2.删除:void clear();  清空集合

          v remove(Object key); 将集合中的元素以键的形式移除。

 

   3.判断:containsKey(Object key) 判断集合中是否有key。如果有,则返回true

          contaninValue(Object value) 判断集合中是否有value

   4.查询:get(Object key);

          size();

          values();

          entrySet();

         keySet();

 Map有三个常见子类:

   |---Hashtable:此类实现一个哈希表,该哈希表将映射到相应的值。任何非null对象都可以用作键或值。该集合是线程同步的。 Jdk1.0 效率低

   |--HashMap:底层是哈希表数据结构,允许使用null值和null键,该集合是不同步的。 Jdk1.2 效率高

   |--TreeMap:底层是二叉树数据结构。线程不同步。可以用于给map集合中的键进行排序。

 

Set很像。

其实Set底层就是使用了Map集合。

 

注:如果添加相同的键,那么后添加的值会覆盖原来键对应值。

map集合的两种取出方式:

1.keySet:将map中所有的键存入到Set集合。因为set具备迭代器。所以可以用迭代的方式取出所有的键,在根据get方法。获取每一个键对应的值。

 Map集合的取出原理:将map集合转成set集合。再通过迭代器取出。

2.Set<Map.Entry<k,v>>  entrySet:将map集合中德映射关系存入到了set集合中,而这个关系的数据类型就是:map.entry.

 

Map.Entry  其实Entry也是一个接口,它是Map接口中的一个内部接口。

为何要定义在其内部呢?

原因:

a.Map集合中的是映射关系这样的两个数据,是先有Map这个集合,才可有映射关系的存在,而且此类关系是集合的内部事务。

b.并且这个映射关系可以直接访问Map集合中的内部成员,所以定义在内部。

package Collection;

import java.util.HashMap;
import java.util.Iterator;

import java.util.Set;
import java.util.Map.Entry;

public class HashMapDemo {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		entrySetMethod();
         
	}
	//利用keySet这个方法取出map集合中的值。
	public static void keySetMethod(){
		//创建一个HashMap集合,并添加了泛型。键值都是String类型的。
        HashMap<String,String> hm = new HashMap<String,String>();
        hm.put("01","zhangsan01");
        hm.put("02","zhangsan02");
        hm.put("03","zhangsan03");
        hm.put("04","zhangsan04");
        Set<String> keyset = hm.keySet();
        for(Iterator<String> it =keyset.iterator();it.hasNext();){
       	    String key =it.next();
       	    String value = hm.get(key);
       	    System.out.println("key: "+key+" "+"value:"+value);
        }
	}
	public static void entrySetMethod(){
		HashMap<String,String> hm = new HashMap<String,String>();
        hm.put("01","zhangsan01");
        hm.put("02","zhangsan02");
        hm.put("03","zhangsan03");
        hm.put("04","zhangsan04");
        Set<Entry<String, String>> entry=hm.entrySet();
        for(Iterator<Entry<String, String>> it =entry.iterator();it.hasNext();){
        	Entry<String, String> en = it.next();
        	System.out.println(en.getKey()+"  "+en.getValue());
        }
	}
	

}

Map练习

1.描述学生。

2.定义map容器。将学生作为键,地址作为值。存入。

3.获取map集合中的元素。

package Collection;

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

public class MapTest {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		entrySet();
	}
	//第一种取出方式keySet();
    public static void keySet(){
    	HashMap<Students,String> hm   = new HashMap<Students,String>();
    	hm.put(new Students("lisi1",21), "beijing");
    	hm.put(new Students("lisi1",21), "tianjing");
    	hm.put(new Students("lisi4",33), "nanjing");
    	hm.put(new Students("lisi8",44), "ganzhou");
    	hm.put(new Students("lisi5",80), "hengyang");
    	Set<Students> keyset  =hm.keySet();
    	for(Iterator<Students> it = keyset.iterator();it.hasNext();){
    		  Students key =it.next();
    		  String addrv =hm.get(key);
    		  System.out.println(key+":::"+addrv);
    		  
    	}
    }
    public static void entrySet(){
    	HashMap<Students,String> hm   = new HashMap<Students,String>();
    	hm.put(new Students("lisi1",21), "beijing");
    	hm.put(new Students("lisi1",21), "tianjing");
    	hm.put(new Students("lisi4",33), "nanjing");
    	hm.put(new Students("lisi8",44), "ganzhou");
    	hm.put(new Students("lisi5",80), "hengyang");
    	Set<Map.Entry<Students, String>> entryset = hm.entrySet();
    	for(Iterator<Map.Entry<Students, String>> it =entryset.iterator();it.hasNext();){
    		 Map.Entry<Students, String> entry = it.next();
    		 Students s = entry.getKey();
    		 String addr = entry.getValue();
    		 System.out.println(s+":::"+addr);
    	}
    }
	
	

}

泛型

泛型概述

在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。

类的后面跟着<E>这样的标识,如ArrayList<E>,这个<E>就代表泛型,E就是泛型类型,表示ArrayList集合里面只能存储E类型的元素。
其本质就是实现参数化类型,也就是说所操作的数据类型被指定为一种类型。

好处:
1、将运行时期出现的问题ClassCastException转移到了编译时期,方便程序员解决,让运行时期问题减少。
2、避免了强制转换麻烦。

示例
import java.util.*;
class GenericDemo{
    public static void main(String[] args){
        //定义了一个String类型的数组列表
        ArrayList<String> al = new ArrayList<String>();
        al.add("abc002");
        al.add("abc013");
        al.add("abc256");
        al.add("abc028");
        
        //al.add(4);//al.add(new Integer(5));
        
        //迭代器要使用泛型
        Iterator<String> it = al.iterator();
        while(it.hasNext()){
            String s = it.next();
            System.out.println(s);
        }
    }
}
/*
结果
abc002
abc013
abc256
abc028
*/

泛型使用

泛型格式:通过<>来定义要操作的引用数据类型。
在使用java提供的对象时,什么时候使用泛型呢?
通常在集合框架中创建,只要见到<>就要定义泛型。

其实<>就是用来接收类型的,当使用集合时,
将集合中要存储的数据类型作为参数传递到<>中即可。

示例
import java.util.*;
class TreeSetTest{
    public static void main(String[] args){
        TreeSet<String> ts = new TreeSet<String>(new LenComparator());
        ts.add("abcd");
        ts.add("cc");
        ts.add("cba");
        ts.add("z");
        ts.add("a");
        ts.add("sdcvrt");

        Iterator<String> it = ts.iterator();
        while(it.hasNext()){
            String s = it.next();
            System.out.println(s);
        }
    }
}
//自定义比较器,实现Comparator接口并使用泛型
class LenComparator implements Comparator<String> {
    public int compare(String s1,String s2){
        /*
        //升序
        int num = new Integer(s1.length()).compareTo(new Integer(s2.length()));
        if(num==0)        
            return s1.compareTo(s2);
        return num;
        */
        //降序
        int num = new Integer(s2.length()).compareTo(new Integer(s1.length()));
        if(num==0)        
            return s2.compareTo(s1);
        return num;
    }
}
/*
结果
sdcvrt
abcd
cba
cc
z
a
*/

泛型类

什么时候定义泛型类?
当类中要操作的引用数据类型不确定的时候,
早期定义Object来完成扩展。
现在定义泛型来扩展。

class Worker{}
class Student{}
//泛型以前的做法
class Tool{
    private Object obj;
    public void setObject(Object obj){
        this.obj = obj;
    }
    public Object getObject(){
        return obj;
    }
}
//泛型类
class Utils<QQ>{
    private QQ q;
    public void setObject(QQ q){
        this.q = q;
    }
    public QQ getObject(){
        return q;
    }
}
class GenericDemo{
    public static void main(String[] args){
        Utils<Worker> u = new Utils<Worker>();
        u.setObject(new Worker());
        Worker w = u.getObject();
        /*
        Tool t = new Tool();
        t.setObject(new Worker());
        Worker w = (Worker)t.getObject();
        */
    }
}

泛型方法

为了让不同方法可以操作不同类型,而且类型还不确定,那么可以将
泛型定义在方法上。

class Demo{
    //类型定义在方法上
    public <T> void show(T t){
        System.out.println("show:"+t);
    }
    public <Q> void print(Q q){
        System.out.println("print:"+q);
    }
}
class  GenericDemo{
    public static void main(String[] args){
        Demo d = new Demo();
        d.show(5);
        d.show(new Integer(5));
        d.show("haha");
        
        d.print(8);
        d.print(new Integer(5));
        d.print("heihei");
    }
}
/*
结果
show:5
show:5
show:haha
print:8
print:5
print:heihei
*/

静态方法泛型

特殊之处:
静态方法不可以访问类上定义的泛型。
如果静态方法操作的引用数据类型不确定,可以将泛型定义在方法上。

class Demo<T>{
    //show()方法的类型由类的类型决定
    public void show(T t){
        System.out.println("show:"+t);
    }
    //print()方法的类型由自己决定
    public <Q> void print(Q q){
        System.out.println("print:"+q);
    }
    //静态方法泛型
    public static <W> void method(W w){
        System.out.println("methld:"+w);
    }
}
class  GenericDemo{
    public static void main(String[] args){
        Demo<String> d = new Demo<String>();
        d.print(8);
        d.print(new Integer(5));
        d.print("heihei");
        d.show("haha");
        //d.show(5);
        d.method("Hello");
    }
}
/*
结果
print:8
print:5
print:heihei
show:haha
methld:Hello
*/

泛型限定

上限:<? extends E> 可以接受E类型或者E的子类型。
下限:<? super E> 可以接收E类型或者E的父类型。

import java.util.*;
class Person{
    private String name;
    Person(String name){
        this.name=name;
    }
    public String getName(){
        return name;
    }
}
class Student extends Person{
    Student(String name){
        super(name);
    }
}
class GenericDemo{
    public static void main(String[] args){



        ArrayList<Person> arr1 = new ArrayList<Person>();
        arr1.add(new Person("Person01"));
        arr1.add(new Person("Person02"));
        arr1.add(new Person("Person03"));
        ArrayList<Student> arr2 = new ArrayList<Student>();
        arr2.add(new Student("Student01"));
        arr2.add(new Student("Student02"));
        arr2.add(new Student("Student03"));
        printColl(arr2);
    }
    public static void printColl(ArrayList<? extends Person> arr1){
        //泛型限定在Person或者Person的子类,就是继承Person类
        Iterator<? extends Person> it = arr1.iterator();
        while(it.hasNext()){
            System.out.println(it.next().getName());
        }
    }
}
/*
结果
Student01
Student02
Student03
*/

泛型扩展

泛型限定是用于泛型扩展用的。

import java.util.*;
class GenericDemo{
    public static void main(String[] args){
        //指定为Student类型,传递一个比较器
        TreeSet<Student> ts = new TreeSet<Student>(new Comp());
        ts.add(new Student("Student01"));
        ts.add(new Student("Student05"));
        ts.add(new Student("Student09"));
        ts.add(new Student("Student08"));
        Iterator<Student> it = ts.iterator();
        while(it.hasNext()){
            System.out.println(it.next().getName());
        }
        //指定为Worker类型,传递一个比较器
        TreeSet<Worker> wo = new TreeSet<Worker>(new Comp());
        wo.add(new Worker("Worker01"));
        wo.add(new Worker("Worker05"));
        wo.add(new Worker("Worker09"));
        wo.add(new Worker("Worker08"));
        wo.add(new Worker("Worker07"));
        Iterator<Worker> it1 = wo.iterator();
        while(it1.hasNext()){
            System.out.println(it1.next().getName());
        }
    }
}
/*
//学生比较器
class StudentComp implements Comparator<Student>{
    public int compare(Student s1,Student s2){
        return s1.getName().compareTo(s2.getName());
    }
}
//工人比较器
class WorkerComp implements Comparator<Worker>{
    public int compare(Worker s1,Worker s2){
        return s1.getName().compareTo(s2.getName());
    }
}
//对上面两个比较器进行优化如下,指定为父类型Person
*/
class Comp implements Comparator<Person>{
    public int compare(Person s1,Person s2){
        //return s1.getName().compareTo(s2.getName());//正序打印
        return s2.getName().compareTo(s1.getName());//倒序打印
    }
}
class Person{
    private String name;
    Person(String name){
        this.name = name;
    }
    public String getName(){
        return name;
    }
    public String toString(){
        return "person:"+name;
    }
}
class Student extends Person{
    Student(String name){
        super(name);
    }
}
class Worker extends Person{
    Worker(String name){
        super(name);
    }
}
/*
结果
Student09
Student08
Student05
Student01
Worker09
Worker08
Worker07
Worker05
Worker01
*/

------- android培训java培训、期待与您交流! ----------
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值