【190天】黑马程序员27天视频学习笔记【Day16-下】

叨逼叨两句

  1. 放纵身体和感情,只能获得短期的快乐,长期这样,那种空虚感会把你逐渐吞没
  2. 唯有自律才能降低不确定性,减缓焦虑
  3. 也唯有自律才能保持足够的精力,用于抵消学习过程中的挫败感,获得技能。
  4. 技能带来竞争壁垒,让你获得人群中的优势地位
  5. 技能也带来确定性,能让你减缓焦虑
  6. 基于以上,拥有技能,可以让你获得长期的快乐。

16-6:泛型(generic)概述和基本使用

泛型概述

  1. 用了泛型,则集合中便只能存储所规定的引用数据类型或其子类

泛型的好处

  1. 提高安全性(将运行期的错误在编译器就发现)
  2. 省去强制转换的麻烦

不用泛型的示例

package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo28 {
    public static void main(String[] args) {
        ArrayList list = new ArrayList();
        list.add(110);
        list.add(true);
        list.add(new Student("张三",23));
        
        Iterator it = list.iterator();
        while(it.hasNext()){
        //    System.out.println(it.next());  //遍历集合中的元素并打印没有问题,但是若想要使用元素中特有的方法,则必须强制转换
        
        //    Student stu = (Student)it.next();   //这样强制转换出错了,明明编译期没有问题啊?那是因为集合混入了除Student以外的其它类型
        //    System.out.println(stu.getName()+"   "+stu.getAge());    
                    
        }
    }
}

使用泛型的示例

package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo28 {
    public static void main(String[] args) {
        ArrayList<Student> list = new ArrayList<Student>();
        //list.add(110);
        //list.add(true);
        list.add(new Student("张三",23));
        list.add(new Student("张四",33));
        
        Iterator<Student> it = list.iterator();
        while(it.hasNext()){
            Student stu = it.next();
            System.out.println(stu.getName()+"   "+stu.getAge());
        }
    }
}

值得注意的细节

  1. 这样直接用it.next()调用方法,结果有问题。
package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo28 {
    public static void main(String[] args) {
        ArrayList<Student> list = new ArrayList<Student>();
        //list.add(110);
        //list.add(true);
        list.add(new Student("张三",23));
        list.add(new Student("张四",33));
        
        Iterator<Student> it = list.iterator();
        while(it.hasNext()){
            //Student stu = it.next();
            System.out.println(it.next().getName()+"   "+it.next().getAge());
        }
    }
}
  1. 前后泛型必须一致,或者后面的泛型可以省略不写。
  2. 泛型设置为Object类型无意义。
package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo28 {
    public static void main(String[] args) {
        
        //ArrayList<Object> list2 = new ArrayList<Student>();  //前后泛型类型不一致,不可以
        //ArrayList<Object> list2 = new ArrayList<>(); //菱形泛型,前面写了后面就不写了,默认与前面一致,JDK1.7以后的新特性
        ArrayList<Object> list2 = new ArrayList<>(); //设置泛型为Object没有意义,这样的集合
        list2.add("aaa");
        list2.add(true);
        
        demo01();
    }

16-7:ArrayList存储字符串和自定义对象并遍历的泛型版

  1. 字符串
package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo29 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("a");
        list.add("b");
        list.add("c");
        list.add("d");
        list.add("e");
        
        Iterator<String> it = list.iterator();
        for(;it.hasNext();){
            System.out.println(it.next());
        }
    }
}
  1. 自定义对象
package com.test.regex;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo29 {
    public static void main(String[] args) {
        ArrayList<Student> list = new ArrayList<>();
        list.add(new Student("kk5",65));
        list.add(new Student("kk4",55));
        list.add(new Student("kk3",35));
        list.add(new Student("kk2",45));
        list.add(new Student("kk1",15));
        
        Iterator<Student> it = list.iterator();
        for(;it.hasNext();){
            Student stu = it.next();
            System.out.println(stu.getName()+"   "+stu.getAge());
        }
    }
}

16-8:泛型的由来

  1. 通过Object转型问题引入
  2. Object类型可以接收任意的对象类型,在实际使用中,会有类型转换问题,存在安全隐患,故Java提供泛型来解决这一问题。
package com.test.demo30;

public class Demo30 {
    public static void main(String[] args) {
        Tool t = new Tool();
        t.setObj(new Student());
        Worker w = (Worker)t.getObj();  
        System.out.println(w);
    }
}
package com.test.demo30;

public class Tool {
    private Object obj;

    public Object getObj() {
        return obj;
    }

    public void setObj(Object obj) {
        this.obj = obj;
    }
    
}
package com.test.demo30;

public class Person {
    private String name;
    private int age;
    
    public Person() {
        super();
        // TODO Auto-generated constructor stub
    }

    public Person(String name, int age) {
        super();
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
    
}
package com.test.demo30;

public class Student extends Person {

    public Student() {
        // TODO Auto-generated constructor stub
    }

    public Student(String name, int age) {
        super(name, age);
        // TODO Auto-generated constructor stub
    }

}
package com.test.demo30;

public class Worker extends Person {

    public Worker() {
        // TODO Auto-generated constructor stub
    }

    public Worker(String name, int age) {
        super(name, age);
        // TODO Auto-generated constructor stub
    }

}

16-9:泛型类的概述及使用

  1. 在定义的类上加入泛型,可以将运行期的错误提前转移到编译期发现。
package com.test.demo30;

public class Tool<Q> {
    private Q obj;

    public Q getObj() {
        return obj;
    }

    public void setObj(Q obj) {
        this.obj = obj;
    }
    
}
package com.test.demo30;

public class Demo30 {
    public static void main(String[] args) {
        Tool<Student> t = new Tool<>();
        t.setObj(new Student());
        Worker w = (Worker)t.getObj();  
        System.out.println(w);
    }
}

16-10:泛型方法的概述和使用

  1. 最好确保泛型类的泛型与泛型方法的泛型一致,如果不一致,需要在方法上声明该泛型。
package com.test.demo30;

public class Tool<Q> {
    private Q q;

    public Q getObj() {
        return q;
    }

    public void setObj(Q q) {
        this.q = q;
    }
    
    public void show(Q q){
        System.out.println(q);
    }
    
    public <T> void show(T t){
        System.out.println(t);
    }
}
package com.test.demo30;

public class Demo30 {
    public static void main(String[] args) {
        Tool<String> t = new Tool();
        t.show("abc");
    }
}
  1. 泛型方法的格式'public <泛型类型> 返回类型 方法名(泛型类型 变量名)'

非静态泛型方法的使用

  1. 对于非静态方法,若泛型方法的泛型类型与泛型类的泛型类型不一致,则需要为方法自己声明一个泛型类型。一致的话就不用了。
package com.test.demo30;

public class Tool<Q> {
    private Q q;

    public Q getObj() {
        return q;
    }

    public void setObj(Q q) {
        this.q = q;
    }
    
    public<W> void show(W w){
        System.out.println(w);
    }
}
package com.test.demo30;

public class Demo30 {
    public static void main(String[] args) {
        Tool<String> t = new Tool();
        t.show(true);
        t.show(new Integer(111));
        t.show(111);
    }
}

静态泛型方法的使用

  1. 对于静态方法,不可使其泛型类型与泛型类的泛型类型一致,因为类加载时还没创建对象,泛型类的泛型类型没法确定。
  2. 对于静态方法,必须声明自己的泛型类型。
package com.test.demo30;

public class Tool<Q> {
    private Q q;

    public Q getObj() {
        return q;
    }

    public void setObj(Q q) {
        this.q = q;
    }
    
    public<W> void show(W w){
        System.out.println(w);
    }
    
    public static <Q> void print(Q q){
        System.out.println(q);
    }
    
}
package com.test.demo30;

public class Demo30 {
    public static void main(String[] args) {
        
        Tool.print("a");
        
    }
}

16-11:泛型接口的概述和使用

package com.test.regex;

public class Demo31 {
    public static void main(String[] args) {
        
    }
}

interface Inter<T> {
    public void show(T t);
}

//第一种实现接口的方式,推荐这种

/*class Demo001 implements Inter<String> {

    @Override
    public void show(String t) {
        System.out.println(t);
    }
}*/

//第二种实现接口的方式,不推荐这种,实现接口没必要让类也加上泛型
class Demo002<T> implements Inter<String>{

    @Override
    public void show(String t) {
        // TODO Auto-generated method stub
        
    }
    
}

16-12:泛型高级之通配符

  1. 泛型通配符<?>:任意类型,若不明确,代表Object以及任意的Java类。
package com.test.regex;

import java.util.ArrayList;
import java.util.List;

public class Demo32 {
    public static void main(String[] args) {
        List<?> list = new ArrayList<Integer>(); // 当右边泛型不确定时,左边可以用?
        List<?> list2 = new ArrayList<String>();
        List<String> list3 = new ArrayList<String>();
        //List<E> list = new ArrayList<Integer>();   //编译不通过
        
    }
}
  1. ?extends E:向下限定,E及其子类
  2. ?super E:向上限定,E及其父类
package com.test.demo30;

import java.util.ArrayList;

public class Demo32 {
    public static void main(String[] args) {
        ArrayList<Person> list1 = new ArrayList<>();
        list1.add(new Person("张三",23));
        list1.add(new Person("张四",24));
        list1.add(new Person("张五",25));
        
        ArrayList<Student> list2 = new ArrayList<>();
        list2.add(new Student("赵四",24));
        list2.add(new Student("赵五",34));
        
        list1.addAll(list2);
        System.out.println(list1);
        
        //list2.addAll(list1); //编译不通过
        System.out.println(list2);
        
    }
}
package com.test.demo30;

public class Student extends Person {

    public Student() {
        // TODO Auto-generated constructor stub
    }

    public Student(String name, int age) {
        super(name, age);
        // TODO Auto-generated constructor stub
    }

}

16-13:增强for循环的概述和使用

  1. 增强for循环底层依赖的是迭代器(Iterator)
package com.test.demo30;

import java.util.ArrayList;

public class Demo33 {

    public static void main(String[] args) {
        int[] arr = {11,22,33,44,55};
        for(int i : arr){
            System.out.println(i);
        }
        
        ArrayList<String> list = new ArrayList<String>();
        list.add("a");
        list.add("b");
        list.add("c");
        list.add("d");
        list.add("e");
        list.add("f");
        for (String string : list) {
            System.out.println(string);
        }
        
    }

}

16-14:ArrayList存储自定义对象并遍历增强for版

package com.test.demo30;

import java.util.ArrayList;

public class Demo33 {

    public static void main(String[] args) {

        ArrayList<Student> list = new ArrayList<Student>();
        list.add(new Student("zzz",22));
        list.add(new Student("z1",22));
        list.add(new Student("z22z",22));
        list.add(new Student("z3",22));
        list.add(new Student("z4z",22));
        for (Student stu : list) {
            System.out.println(stu);
        }
        
    }

}

16-15:关于三种迭代能否删除的测试

  1. for循环
  2. 迭代器
  3. 增强for循环
package com.test.demo30;

import java.util.ArrayList;
import java.util.Iterator;

public class Demo33 {

    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("a");
        list.add("b");
        list.add("c");
        list.add("b");
        list.add("d");
        
        //1. 普通for循环,可以删除,但是要在删除后将游标减一
    /*    for(int i = 0; i < list.size(); i++){
            if("b".equals(list.get(i))){
                list.remove(i--);
            }
        }*/
        
        //    2. 迭代器,可以删除,但是不可使用list的删除方法,而应该使用Iterator的删除方法,否则将出现并发修改异常
        
        /*for(Iterator<String> it = list.iterator();it.hasNext();){
            if("b".equals(it.next())){
                //list.remove("b");
                it.remove();
            }
        }*/

        // 3. 增强for循环, 由于底层是迭代器,只可遍历,不可删除。
        for (String string : list) {
            if("b".equals(string)){
            //    list.remove("b");
            }
        }
            
        System.out.println(list);
    }
}

16-16:静态导入的概述和使用

  1. 导包:import 包名.类名
  2. 静态导入:import static 包名.类名.方法名

    1. jdk1.5后的新特性
    2. 导入类中的静态方法
    3. 开发一般不用,能看懂就行,若有多个同名静态方法,容易分不清。

16-17:可变参数的概述及使用

  1. 可变参数实际底层就是数组
  2. 可变参数可以不赋值,也可以赋无穷多个值
package com.test.demo30;

public class Demo34 {
    public static void main(String[] args) {
        int[] arr = {11,22,33,44,55};
        //print(arr);
        print(11,22,33,44,55,66);
        System.out.println("----------------");
        print();
        
    }
    
    /*public static void print(int[] arr){
        for (int i = 0; i < arr.length; i++) {
            System.out.println(arr[i]);
        }
    }*/
    
    public static void print(int ... arr){    //可变参数其实就是一个数组
        for (int i = 0; i < arr.length; i++) {
            System.out.println(arr[i]);
        }
    }
}
  1. 如果一个方法有多个参数,那么可变参数必须放在最后。
package com.test.demo30;

public class Demo34 {
    public static void main(String[] args) {
        print(11,22,33,44,55,66);
    }
    
    //public static void print(int ... arr,int x){   //这样设置参数是不行的
    public static void print(int x,int ... arr){    //可变参数其实就是一个数组
        for (int i = 0; i < arr.length; i++) {
            System.out.println(arr[i]);
        }
    }
}

16-18:ArrayList工具类的asList()方法的使用

  1. 数组转集合
package com.test.demo30;

import java.util.Arrays;
import java.util.List;

public class Demo35 {
    /**
     * @param args
     * 数组转换为集合
     */
    public static void main(String[] args) {
        String[] arr = {"a","b","c"};
        List<String> list = Arrays.asList(arr);
        list.add("d");
        System.out.println(list);    //数组转换成集合虽然不能增加或减少元素,但是可以用集合的思想操作数组,也就是说可以使用其他集合中的方法
    }
}
  1. 基本数据类型的数组与引用数据类型的数组,用asList处理后打印结果不一样
  2. 基本数据类型的数组转换成集合,会将整个数组当作一个元素
package com.test.demo30;

import java.util.Arrays;
import java.util.List;

public class Demo35 {
    /**
     * @param args
     * 数组转换为集合
     */
    public static void main(String[] args) {
        //demo01();
        int[] arr = {11,22,33,44,55};
        List<int[]> list = Arrays.asList(arr);
        System.out.println(list);
        
    }

    public static void demo01() {
        String[] arr = {"a","b","c"};
        List<String> list = Arrays.asList(arr);
        //list.add("d");
        System.out.println(list);    //数组转换成集合虽然不能增加或减少元素,但是可以用集合的思想操作数组,也就是说可以使用其他集合中的方法
    }
}
  1. 想将数组转换成集合,数组必须是引用数据类型【基本数据类型你就使用它们的包装类】
public static void main(String[] args) {
        Integer[] arr = {11,22,33,44,55};
        List<Integer> list = Arrays.asList(arr);
        System.out.println(list);    
    }
  1. 集合转数组【加泛型】

    1. 当集合转换数组时,数组长度如果是小于等于集合的size,转换后的数组长度等于数组长度。
    2. 如果数组的长度大于了size,分配的数组长度就和你指定的长度一样。
package com.test.demo30;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class Demo35 {
    /**
     * @param args
     * 数组转换为集合
     */
    public static void main(String[] args) {
        ArrayList<String> a = new ArrayList<>();
        a.add("a");
        a.add("b");
        a.add("c");
        a.add("d");
        
    //    String[] arr = a.toArray(new String[0]);
    //    String[] arr = a.toArray(new String[1]);
        String[] arr = a.toArray(new String[10]);
        for (String string : arr) {
            System.out.println(string);
        }
        
        
    }

16-19:集合嵌套之ArrayList嵌套ArrayList

package com.test.demo30;

import java.util.ArrayList;

public class Demo36 {

    public static void main(String[] args) {
        ArrayList<ArrayList<Person>> list = new ArrayList<>();   //学科集合
        ArrayList<Person> first = new ArrayList<>();     //班级集合
        first.add(new Person("kk",30));      //每一个人
        first.add(new Person("j",30));
        first.add(new Person("gg",30));
        
        ArrayList<Person> second = new ArrayList<>();
        second.add(new Person("a",30));
        second.add(new Person("b",30));
        second.add(new Person("c",30));
    
        list.add(first);
        list.add(second);
        
        for (ArrayList<Person> a : list) {
            for(Person p: a){
                System.out.println(p);
            }
        }
        
    }

}

相关推荐