当前位置:   article > 正文

JAVA小白学习日记Day7

JAVA小白学习日记Day7

1.Set集合

Set继承自Collection接口,没有特殊的方法

1.1Set集合的特点

  1. Set中不包含重复元素(只能存储一个null)
  2. 不维护存储顺序
  3. Set没有索引,不能通过索引访问元素,所以不能使用普通for循环遍历
  4. 只能通过元素本身来访问

1.2 HashSet

HashSet概述:

  1. 此类实现Set接口
  2. 底层数据结构为HashMap哈希表
  3. 允许一个null元素
  4. 线程不安全

元素放入时是无序的,指的是内存空间不连续,因为元素在set中的位置是由该元素的HashCode决定的。

而且,HashMap每次扩容都可能导致元素重新排布,所以元素的位置是不确定的。

1.3TreeSet

TreeSet概述:

  1. TreeSet是SortedSet接口的实现类,TreeSet可以确保集合元素处于排序状态。
  2. TreeSet内部实现的是红黑树,默认整形排序为从小到大。
  3. 不允许放入null值

存入TreeSet集合中的元素要具备比较性,比较性要实现Comparable接口,重写该接口的compareTo方法。

TreeSet根据构造方法不同,分为自然排序(无参构造)和比较器排序(有参构造):

自然排序要求元素必须实现Compareable接口,并重写里面的compareTo()方法,元素通过比较返回的int值来判断排序序列,返回0说明两个对象相同,不需要存储。
比较器排需要在TreeSet初始化是时候传入一个实现Comparator接口的比较器对象,或者采用匿名内部类的方式new一个Comparator对象,重写里面的compare()方法。

1.4LinkedHashSet

LinkedHashSet底层数据结构是哈希表和链表,根据元素的hashCode值来决定元素的存储位置,但它同时使用链表维护元素的次序,这样使得元素看起来是以插入的顺序保存的,也就是说当遍历集合LinkedHashSet集合里的元素时,集合将会按元素的添加顺序来访问集合里的元素。

输出集合里的元素时,元素顺序总是与添加顺序一致。但是LinkedHashSet依然是HashSet,因此它不允许集合重复。

Set集合代码:

  1. package com.easy722;
  2. import com.text.TextA;
  3. import java.util.*;
  4. public class EasySet {
  5. //set集合
  6. //List集合是有序的,set集合是无序的
  7. //有序和无序 取出的顺序和添加的顺序是一样的
  8. public static void main(String[] args) {
  9. //实例化Set集合
  10. HashSet set=new HashSet();
  11. set.add("23");
  12. set.remove("123");
  13. set.size();
  14. set.add("23");
  15. int size=set.size();//1
  16. System.out.println(size);
  17. //set中不能存储相同的数据
  18. set.add(new String("456"));
  19. set.add(new String("456"));
  20. size=set.size();//2
  21. System.out.println(size);
  22. //可以存储null值
  23. set.add(null);
  24. size=set.size();//3
  25. System.out.println(size);
  26. for (Object item:set){
  27. System.out.println(item);
  28. }
  29. //TreeSet 红黑树
  30. TreeSet test=new TreeSet();
  31. //存入数据
  32. test.add(2);
  33. test.add(200);
  34. test.add(3);
  35. test.add(88);
  36. test.add(45);
  37. test.add(72);
  38. test.add(1);
  39. test.add(99);
  40. //同样一个TreeSet存储的内容都应该是可比较的
  41. //默认情况下不能存储不同类型
  42. //test.add("234");
  43. //不能存null值
  44. //test.add(null);
  45. //无序的
  46. //遍历
  47. for (Object item:test){
  48. System.out.print(item+",");
  49. }
  50. System.out.println();
  51. //先序遍历 中序遍历 后序遍历
  52. //TreeSet遍历方式是中序方式
  53. //TreeSet 内部使用二叉树 内部节点是可以比较大小的
  54. Comparator<Student> com=(a,b)-> {
  55. if(a.score==b.score){
  56. return 0;
  57. }
  58. return a.score > b.score ? 1 : -1;
  59. };
  60. TreeSet<Student> ttset=new TreeSet<>(com);
  61. Student stua=new Student();stua.score=89;
  62. Student stub=new Student();stub.score=80;
  63. Student stuc=new Student();stuc.score=99;
  64. Student stud=new Student();stud.score=100;
  65. ttset.add(stua);
  66. ttset.add(stub);
  67. ttset.add(stuc);
  68. ttset.add(stud);
  69. for(Student stu:ttset){
  70. System.out.println(stu.score);
  71. }
  72. //LinkedHashSet 是有序的集合Set
  73. }
  74. }
  75. class Student implements Comparable{
  76. int score;
  77. @Override
  78. public int compareTo(Object o) {
  79. if (o instanceof Student){
  80. Student item=(Student)o;
  81. if(this.score==item.score)
  82. return 0;
  83. return this.score>item.score?1:-1;
  84. }else{
  85. //如果o对象不是Student 就无法比较 这是程序运行时出现的特殊情况
  86. //异常情况 我们的方法处理不了这种情况,就要抛出一个异常对象
  87. //告知调用此方法发代码
  88. throw new RuntimeException("传入对象不可比较");
  89. }
  90. }
  91. }

2.Map

Map   存储键值对  键   就是名字   值  存储的对象
Map接口是一个键值对的集合,它继承自Collection接口中的size()和isEmpty()等方法,同时还提供了根据键查找值的方法,以及添加、删除和更新键值对的方法。在Java中,Map接口有几个常见的实现类,每个实现类都具有不同的性能和用途。

HashMap:基于哈希表实现,具有快速的查找和插入操作,适用于需要快速查找键值对的场景。
TreeMap:基于红黑树实现,可以对键进行排序,并提供了一系列与排序相关的方法,适用于需要对键进行排序的场景。
LinkedHashMap:基于哈希表和链表实现,保持键值对的插入顺序,适用于需要保持插入顺序的场景。

2.1HashMap

底层实现:数组加列表,数组的默认容量是16,每次扩容2倍,扩容阈值是0.75,一个链达到8就对该链进行树化,一支树上的元素低于6个,这个树就会退化成链,最小树化容量阈值  64    就是数组容量最少64且链上最少是8个才树化,如果数组的长度没有达到64,优先扩容。

put(K key, V value): 将指定的键值对添加到HashMap中。
remove(Object key): 从HashMap中移除指定键的键值对。
get(Object key): 返回指定键对应的值。
containsKey(Object key): 检查HashMap中是否包含指定的键。
containsValue(Object value): 检查HashMap中是否包含指定的值。
size(): 返回HashMap中键值对的数量。

 
2.2TreeMap

在HashMap中key和value可以是null,但是key不能重复,而在Tree Map中,key应该可以比较,且不能是null值。Hash table的key和value都不能是null,Hashtable是线程安全的集合。

ConcurrentHashMap 线程安全且效率较高。

线程安全的Map  Hashtable  ConcurrentHashMap(性能优异,锁颗粒度比较小)

 代码:

  1. package com.easy722;
  2. import java.util.*;
  3. import java.util.concurrent.ConcurrentHashMap;
  4. public class EasyMap {
  5. //Map 存储键值对 键 就是名字 值 存储的对象
  6. public static void main(String[] args) {
  7. Map map= new HashMap();
  8. //存储数据
  9. map.put("A1","张三");
  10. //可以通过存入的key获取存入的对象
  11. Object obj=map.get("A1");//获取A1对应的value值
  12. System.out.println(obj);//张三
  13. //通过key删除键值对
  14. Object rem_obj=map.remove("A1");
  15. System.out.println(rem_obj);//张三 返回删除的value值
  16. //是否包含key
  17. map.containsKey("A1");
  18. //是否包含value
  19. map.containsValue("张三");
  20. Set setkey=map.keySet();//获取所有的key
  21. //获取所有的value
  22. Collection con=map.values();
  23. //key和value可以是null,但是key不能重复
  24. map.put(null,null);
  25. map.put("A1","张三");
  26. map.put("A1",null);
  27. System.out.println(map);//{null=null, A1=null}null将张三覆盖,key是唯一的
  28. //TreeMap
  29. //key应该是可比较的
  30. //key不能是null值
  31. Hashtable ht=new Hashtable();
  32. ht.put("t1","张三");
  33. //Hashtable key和value都不能是null
  34. //Hashtable是线程安全的集合
  35. LinkedHashMap lMap=new LinkedHashMap();
  36. lMap.put("","");
  37. //ConcurrentHashMap 线程安全 效率较高
  38. ConcurrentHashMap chm;
  39. //HashMap 底层实现 数组加链表
  40. HashMap hmap=new HashMap();
  41. hmap.put("","");
  42. //HashMap 的数组默认容量 16
  43. //每次扩容 2倍
  44. //扩容阈值 0.75
  45. //一个链达到8就对该链进行树化
  46. //一支树上的元素低于6个,这个树就会退化成链
  47. //最小树化容量阈值 64 就是数组容量最少64且链上最少是8个才树化
  48. //如果数组的长度没有达到64,优先扩容
  49. //线程安全的Map Hashtable ConcurrentHashMap(性能优异,锁颗粒度比较小)
  50. }
  51. }

3.泛型

泛型是广泛的数据类型,确保类型安全的一种途径。

当创建一个 Generic< T > 类对象时,会向尖括号 <> 中传入具体的数据类型。

方法签名中声明了< 泛型标识>的方法称为泛型方法。只有在方法签名中声明了的方法才是泛型方法,仅使用了泛型类定义的类型参数的方法并不是泛型方法。

1、泛型类中的静态方法和静态变量不可以使用泛型类所声明的类型参数。
2、泛型方法可以是静态的,这一点容易和上面的点混淆。
3、使用了泛型类中的泛型标识的方法不一定是泛型方法,泛型方法必须必须由< T>泛型标识声明。即使泛型类中定义的类型参数标识和泛型方法中定义的类型参数标识都为< T > ,但它们彼此之间是相互独立的。也就是说,泛型方法始终以自己声明的类型参数为准,所以,在泛型类中定义泛型方法时最好用不同的标识来区分。

 代码:

  1. package com.easy722;
  2. import java.util.*;
  3. public class EasyGenericity<J,M,N>{
  4. //泛型 广泛的数据类型
  5. //确保类型安全的一种途径
  6. //定义 类或方法上定义泛型
  7. private M m;
  8. public void test(M m,N n){
  9. }
  10. //在返回值前面声明
  11. public static <J extends List > J testA(J j){//就近原则,不是上面的J
  12. return j;
  13. }
  14. public static <A>void testAA (List<?super A>a){
  15. }
  16. public <E> E test(E e){
  17. //返回的类型和传入的类型是同一种
  18. return e;
  19. }
  20. public static void main(String[] args) {
  21. testA(new ArrayList());
  22. //testA(new HashSet());
  23. //类型安全
  24. List list=new ArrayList();
  25. list.add("123");
  26. //list.add(new Object());
  27. for (Object object:list){
  28. //类型转换的时候没有明确对象的数据类型进行强制数据类型转换
  29. //会抛出Class CaseException(类型转换异常)
  30. //类型不安全
  31. String str=(String)object;
  32. System.out.println(str);
  33. }
  34. List<String> listA=new ArrayList<>();
  35. listA.add("8");
  36. Map<String,Object> map=new HashMap<>();
  37. }
  38. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/秋刀鱼在做梦/article/detail/907323
推荐阅读
相关标签
  

闽ICP备14008679号