当前位置: 首页 > news >正文

JAVA基础:HashMap底层数组容量控制,TreeMap底层存取机制,位运算符,原码反码补码

List常用实现类

  • List集合常用的实现类有3个 , ArrayList , LinkedList , Vector

ArrayList

  • 类似于我们之前的ArrayBox

  • 底层使用数组存储元素, 插入删除的效率低,检索的效率高

  • 当底层数组存储容量不足时,会进行扩容,每次扩容都是原数组的1.5倍

  • 使用无参构造器创建ArrayList集合时,首先提供的是一个默认0长度的数组(防止内存资源浪费)

    第一次add元素的时候,会创建一个10个长度的数组

Vector

  • 我们在开发中已经不怎么使用了

  • Vector是ArrayList早期版本1.0

  • 起初有自己的一套api方法,后来与ArrayList实现了相同的List接口

  • Vector也是使用数组的结构存储

  • Vector早期版本,线程同步的,安全性高,但效率低。

    ArrayList是非线程同步的,安全性低,但效率高

    (想到StringBuilder和StringBuffer)

  • Vector底层数组容量不足时,会按照原容量的2倍扩容

    ArrayList按照原容量1.5扩容

  • Vector在创建对象时,可以指定扩容增量,每次扩容时会按照指定的增量扩容

    ArrayList只能1.5倍扩容

LinkedList

  • 类似于我们之前封装的LinkedBox

  • 底层使用双向链表存储元素,插入删除效率高,检索效率低

  • LinkedList还提供了一组与头尾元素操作相关的方法。

2 Set常用实现类

  • 常用的实现类有2个 , HashSet , TreeSet

  • 底层用来存储元素的都是对应的Map,相当于只存储了map中的key,而没有value

HashSet

  • 底层使用的是HashMap存储

  • 向set中存储一个元素,就相当于向map中存储了一个key

  • 这个value是什么呢?

    • 是 null 么。 不是 。 底层的map,如果key重复,会返回之前的value。 如果key不重复,返回null

      对于set而言,返回null,说明key不重复,说明保存成功。

      反之说明key重复,保存失败

      如果用null作为value,就无法判断保存是否成功

    • 需要让value值是一个具体的对象。 但每次创建一个新对象,又浪费空间。所以用唯一的对象作为value值

public boolean add(E e) {return map.put(e, PRESENT)==null;
}

 

TreeSet

  • TreeSet底层是通过TreeMap集合存储。

  • 存储的key是自然有序的(有大小顺序)

  • 集合存储的类对象的,这就要求存入TreeSet中的元素(TreeMap中的key),要可以比较大小

    自身比较(Comparable),或提供第三方比较器(Comparator)

  • 使用无参构造器创建TreeSet时,要求存储的元素必须自身可以比较

    要么就使用可以传递第三方比较器的Tree构造方法

public TreeSet(Comparator comparator){}
  • TreeSet在存储第一个元素的时候,也需要保证元素是可以比较大小的。

3 Map常用实现类

  • Map常用的实现类有3个, HashMap ,TreeMap , Hashtable

HashMap

  • 底层使用数组 + 链表结构存储元素。jdk1.8之后,变成了数组 + 链表 or 红黑树 存储元素

  • hash的存储特点是单一元素的 快存 和 快取

  • 每次存储的元素(key),都会通过一个hash算法,计算出该元素在数组中存储的位置

    有可能第1个元素,被计算出存储在5位置。 第2个元素,被计算出存储2位置。使得存储位置是零散的

  • 是根据元素的什么信息进行位置的hash计算

    是根据元素的hashcode值。 每一个元素都会有一个hashcode值,我们通过调用hashcode方法来获得。

    Object中有一个hashcode方法,是一个native修饰的方法,表示该方法是由jvm(c/c++)语言实现的

    是由jvm为对象提供的一个尽可能唯一的数值。可以当成地址使用,不是内存地址

    对象的hashcode不是与生俱来的,当第一次调用hashcode方法时,才会生成hashcode(JOL验证)

    hashcode值的作用,就是为hash存储结构来计算存储位置的。

    hashcode方法可以被重写,重写原则是, equals相等, hashcode也相等。反之无所谓。

  • hash碰撞 ? 两个不同的对象,通过hash计算后,算出了相同的存储位置

    两个对象都需要存储,会在当前位置形成一个链表

扩展:hashmap保存元素时的那些事

  1. 当发生hash碰撞时,会用新节点与当前位置链表中的每一个节点进行比较

    有相等的(hash相等,equals相等。同一个key),key去重,value覆盖

    不相等,就将新元素的节点添加到链表最后

  2. 对于hash碰撞时,当前的hashMap底层,使用的时链表的尾插法

    jdk1.7时,发生hash碰撞,采用的是头插法 (头超法在多线程时会有问题)

  3. jdk1.8之后,HashMap底层使用数组 + 链表 或 红黑树。为什么用到树呢?

    防止链表过长,影响hash效率。当链表长度超过8时,就会尝试转换成红黑树了。

    如果数组长度<64 , 会优先2倍扩容

    如果数组长度>64,就会实现链表到红黑树的转化

HashMap底层数组的长度问题

  • 尽管HashMap提供了有参构造器,可以指定数组的初始容量,但最终都会处理成的2的次幂

  • 也就是说, hashmap底层的数组,无论如何其长度都是2的次幂数

//cap = 10
static final int tableSizeFor(int cap) {//n = 9//16个0 1000 0000 0000 0000int n = cap - 1;//n = n | n>>>1 ;//n = 28个0 1001 | 28个0 0100//n = 28个0 1101//1000 0101 0010 1001 | 0100 0010 1001 0100//1100 0111 1011 1101n |= n >>> 1;//1101 | 0011//28个0 1111 == 10000=16//11111== 100000=32//1100 0111 1011 1101 | 0011 0001 1110 1111//1111 0111 1111 1111n |= n >>> 2;// 1111 | 0000//1111 0111 1111 1111 | 0000 1111 0111 1111 n |= n >>> 4;n |= n >>> 8;n |= n >>> 16;//x个0, y个1return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
}

 

hashmap底层有很多的计算都是基于位运算的

  1. 通过hash计算元素的存储位置

    jdk早期版本,通过除法取模计算最终的存储位置 hash=101,length=6 , 101%6=5

  2. 数组扩容

Hashtable

  • Hashtable 与HashMap作用相同,也是hash结构,适合快存, 快取

  • 区别

    1. Hashtable早期版本,线程同步,安全性高,效率低

      HashMap,非线程同步,安全性低,效率高

    2. Hashtable中的key和value不能为null

      HashMap中的key和value可以为null

TreeMap

  • 底层使用的是红黑树存储元素,是BST树的一种特殊形式

  • 我们暂时只需要了解BST

    • 是一个二叉树,一个节点有左右两个子节点,和一个父节点

    • 节点间存储在大小关系,左枝节点 < 父节点 < 右枝节点

  • 存储

    • 第一个节点必然就是最开始的节点,我们称为根节点

    • 第二个数据添加时,会从根节点开始,依次进行大小比较

      • 如果与根节点key相同,说明重复,key取重,value覆盖

      • 如果与根节点key不相同,根据大小,如果小于根节点的key,向左枝找位置,否则反之

      • 接着会与左枝节点比较,相等取重,不等继续比较大小

      • 直到某一个节点的左枝或右枝为null,说明找到了要存放的位置,完成节点链接即可。

  • 遍历

    • 二叉树有3种遍历方式 前序遍历,中序遍历,后序遍历

    • 以父节点为基准

      父 左 右 :前序

      左 父 右 : 中序

      左 右 父 : 后续

BST -> AVL -> 红黑树

4 位运算符

运算符都有哪些呢? 算数运算符 , 赋值运算符,比较运算符,逻辑运算符,条件运算符,位运算符

位运算符的特点:

  • 对二进制进行运算的

  • 按位运算的

  • 位运算符包括

    • &(按位与),|(按位或),^(按位异或),~(按位取反)

    • <<(左移) , >>(无符号右移) , >>>(带符号右移)

 

相关文章:

JAVA基础:HashMap底层数组容量控制,TreeMap底层存取机制,位运算符,原码反码补码

List常用实现类 List集合常用的实现类有3个 &#xff0c; ArrayList , LinkedList , Vector ArrayList 类似于我们之前的ArrayBox 底层使用数组存储元素&#xff0c; 插入删除的效率低&#xff0c;检索的效率高 当底层数组存储容量不足时&#xff0c;会进行扩容&#xff0c;…...

【Redis】Redis 缓存设计:抗住百万并发量的最佳实践

目录 1. Redis 缓存设计原则1.1 高可用性1.2 数据一致性1.3 读写分离 2. 缓存策略2.1 常用缓存策略2.1.1 缓存穿透2.1.2 缓存雪崩2.1.3 缓存击穿 2.2 额外缓存策略2.2.1 更新策略2.2.2 预热策略2.2.3 侧写缓存 3. Redis 架构设计3.1 单机 vs 集群3.2 Redis 集群示例架构 4. 性能…...

【hot100-java】【缺失的第一个正数】

R9-普通数组篇 class Solution {public int firstMissingPositive(int[] nums) {int nnums.length;for (int i0;i<n;i){while(nums[i]>0&&nums[i]<n&&nums[nums[i]-1]!nums[i]){//交换nums[i]和nums[nums[i]-1]int temp nums[nums[i]-1];nums[nums[i]…...

独立站新手教程转化篇:如何做好移动端优化?

随着移动设备在全球范围内的普及&#xff0c;越来越多消费者选择通过手机或平板电脑&#xff0c;来进行线上购物。因此移动端优化&#xff0c;因此移动端优化&#xff0c;也成为独立站卖家必须重视的一个关键环节。那么独立站移动端需要做好哪些优化工作呢&#xff1f; 选择响…...

Mybatis Plus分页查询返回total为0问题

Mybatis Plus分页查询返回total为0问题 一日&#xff0c;乌云密布&#xff0c;本人看着mybatis plus的官方文档&#xff0c;随手写了个分页查询&#xff0c;如下 Page<Question> questionPage questionService.page(new Page<>(current, size),questionService.g…...

VulnHub-Narak靶机笔记

Narak靶机笔记 概述 Narak是一台Vulnhub的靶机&#xff0c;其中有简单的tftp和webdav的利用&#xff0c;以及motd文件的一些知识 靶机地址&#xff1a; https://pan.baidu.com/s/1PbPrGJQHxsvGYrAN1k1New?pwda7kv 提取码: a7kv 当然你也可以去Vulnhub官网下载 一、nmap扫…...

查看和升级pytorch到指定版本

文章目录 查看和升级pytorch到指定版本查看pytorch的版本python 命令查看pytorch的版本使用pip 命令查看当前安装的PyTorch版本升级PyTorch到指定版本 升级到特定的版本 查看和升级pytorch到指定版本 查看pytorch的版本 python 命令查看pytorch的版本 通过Python的包管理工具…...

Maya---机械模型制作

材质效果&#xff08;4&#xff09;_哔哩哔哩_bilibili 三角面 四边面 多边面 *游戏允许出现三角面和四边面 游戏中一般是低模&#xff08;几千个面&#xff09; 动漫及影视是高模 机械由单独零件组合而成&#xff0c;需独立制作 低面模型到高面模型 卡线是为了将模型保…...

请不要在TS中使用Function类型

在 TypeScript 中&#xff0c;避免使用 Function 作为类型。Function 代表的是“任意类型的函数”&#xff0c;这会带来类型安全问题。对于绝大多数情况&#xff0c;你可能更希望明确地指定函数的参数和返回值类型。 如果你确实想表达一个可以接收任意数量参数并返回任意类型的…...

关于UVM仿真error数量达到指定值就退出仿真的设置

1. 问题描述 在某项目调试过程中&#xff0c;发现通过tc_base.sv中new函数里的set_report_max_quit_count()设置最大error数量不生效&#xff0c;uvm_error数量仍旧是达到10个&#xff08;默认&#xff09;就会退出仿真。 2. 设置uvm_error到达一定数量结束仿真的方式 由白皮…...

chatGPT问答知识合集【二】

Redis 架构说明 Redis 是一个开源的内存数据库&#xff0c;它也可以持久化到磁盘。以下是 Redis 的典型架构说明&#xff1a;### Redis 架构组件&#xff1a;1. **客户端**&#xff1a;与 Redis 服务器进行通信的应用程序或客户端库。2. **Redis 服务器**&#xff1a;执行实际…...

不靠学历,不拼年资,怎么才能月入2W?

之前统计局发布了《2023年城镇单位就业人员年平均工资情况》&#xff0c;2023年全国城镇非私营单位和私营单位就业人员年平均工资分别为120698元和68340元。也就是说在去年非私营单位就业人员平均月薪1W&#xff0c;而私营单位就业人员平均月薪只有5.7K左右。 图源&#xff1a;…...

【软考】多核CPU

目录 1. 说明 1. 说明 1.核心又称为内核&#xff0c;是 CPU 最重要的组成部分。2.CPU 中心那块隆起的芯片就是核心&#xff0c;是由单品硅以一定的生产工艺制造出来的&#xff0c;CPU 所有的计算、接收/存储命令、处理数据都由核心执行。3.各种 CPU 核心都具有固定的逻辑结构&…...

制作炫酷个人网页:用 HTML 和 CSS3 展现你的风格

你是否觉得自己的网站应该看起来更炫酷&#xff1f;今天我将教你如何使用 HTML 和 CSS3 制作一个拥有炫酷动画和现代设计风格的个人网页&#xff0c;让它在任何设备上看起来都无敌酷炫&#xff01; 哈哈哈哈哈哈哈哈,我感觉自己有点中二哈哈哈哈~ 目录 炫酷设计理念构建 HTML …...

WinCC中归档数据片段的时间和尺寸设置

1&#xff0e;归档数据片段介绍工控人加入PLC工业自动化精英社群 1.1 概述 WinCC V6.2 开始的后台数据库采用了MS SQL Server 2005 &#xff0c;所以归档方式与V5 有所不同&#xff0c;它的运行数据存放在数据片段&#xff08;segment&#xff09;当中&#xff0c;工程师可以…...

kubernetes网络(二)之bird实现节点间BGP互联的实验

摘要 上一篇文章中我们学习了calico的原理&#xff0c;kubernetes中的node节点&#xff0c;利用 calico 的 bird 程序相互学习路由&#xff0c;为了加深对 bird 程序的认识&#xff0c;本文我们将使用bird进行实验&#xff0c;实验中实现了BGP FULL MESH模式让宿主相互学习到对…...

动态语言? 静态语言? ------区别何在?java,js,c,c++,python分给是静态or动态语言?

JavaScript 被称为动态语言&#xff0c;而 Java 被称为静态语言 这主要与它们在类型系统、编译执行方式以及运行时行为等方面的不同特性有关。详细差异如下&#xff1a; JavaScript (动态语言) 动态类型&#xff1a; 在JavaScript中&#xff0c;变量的类型是在运行时确定的。这…...

计算机网络17——IM聊天系统——客户端核心处理类框架搭建

目的 拆开客户端和服务端&#xff0c;使用Qt实现客户端&#xff0c;VS实现服务端 Qt创建项目 Qt文件类型 .pro文件&#xff1a;配置文件&#xff0c;决定了哪些文件参与编译&#xff0c;怎样参与编译 .h .cpp .ui&#xff1a;画图文件 Qt编码方式 Qt使用utf-8作为编码方…...

C/C++面试题

关键字 1."#"&#xff0c;"##"的用法 #是字符串转换符&#xff0c;##是字符串连接符&#xff1b;发生在预处理阶段&#xff1b; 2.volatile的含义 防止编译器优化&#xff0c;告诉编译器每次都去真实地址中读取&#xff0c;而不是从寄存器或者缓存中&a…...

[3]Opengl ES着色器

术语&#xff1a; VertexShader&#xff1a;顶点着色器&#xff0c;用来描述图形图像位置的顶点坐标&#xff1b; FragmentShader&#xff1a;片元着色器&#xff0c;用来给顶点指定的区域进行着色&#xff1b; Vertex&#xff1a;顶点 Texture&#xff1a;纹理…...

Spring Boot 中实现任务后台处理的几种常见方式

​ 博客主页: 南来_北往 系列专栏&#xff1a;Spring Boot实战 前言 在现代应用程序中&#xff0c;后台处理对于处理发送电子邮件、处理文件、生成报告等任务至关重要。 Spring Boot 提供了多种机制来高效地实现后台任务。本文探讨了在 Spring Boot 中处理后台处理的各…...

部署--UmiJS

默认方案 umi2 默认对新手友好&#xff0c;所以默认不做按需加载处理&#xff0c;umi build 后输出 index.html、umi.js 和 umi.css 三个文件。 不输出 html 文件 某些场景 html 文件交给后端输出&#xff0c;前端构建并不需要输出 html 文件&#xff0c;可配置环境变量 HTM…...

python自学笔记

python部分总结 主要记录的是python与之前学的语言的不同之处 函数总结 首字母大写: name.title() 删除右边空格&#xff08;暂时&#xff09;:name.rstrip() 删除左边空格&#xff08;暂时&#xff09;:name.lstrip() 删除前缀&#xff08;暂时&#xff09;:name.removeprefi…...

Ubuntu磁盘不足扩容

1.问题 Ubuntu磁盘不足扩容 2.解决方法 安装一下 sudo apt-get install gpartedsudo gparted...

【ROS2】spin、spinOnce、spin_some、spin_until_future_complete

1、简述 spinOnce仅处理一个回调函数(ROS1); spin_some类似于ROS1的spinOnce,但处理多个任务,然后返回(ROS2); spin会持续处理回调函数直到无任务,然后阻塞(ROS1、ROS2); 注意: 只有消息推送(publisher)功能的程序,不需要使用spin_some(),因为它不执行任何回…...

化繁为简:中介者模式如何管理复杂对象交互

化繁为简&#xff1a;中介者模式如何管理复杂对象交互 中介者模式 是一种行为型设计模式&#xff0c;定义了一个中介者对象&#xff0c;来封装一组对象之间的交互。中介者模式通过将对象之间的交互行为从多个对象中抽离出来&#xff0c;集中封装在一个中介者对象中&#xff0c;…...

控制STM32蜂鸣器示例代码(江科大)

以下代码来源于本人学习江科大的课程&#xff0c;这是一个简单的STM32微控制器程序&#xff0c;用于控制连接到GPIOB第12号引脚的蜂鸣器。程序通过GPIOB的第12号引脚输出PWM波形来控制蜂鸣器的频率&#xff0c;从而产生声音。 #include "stm32f10x.h" …...

Java基础知识扫盲

目录 Arrays.sort的底层实现 BigDecimal(double)和BigDecimal(String)有什么区别 Char可以存储一个汉字吗 Java中的Timer定时调度任务是咋实现的 Java中的序列化机制是咋实现的 Java中的注解是干嘛的 Arrays.sort的底层实现 Arrays.sort是Java中提供的对数组进行排序的…...

ZLMediaKit Windows编译以及使用

1.运行ZLMediaKit 2.通过ffmpeg把视频源推流给ZLMediaKit 执行以下命令&#xff0c;将本地视频通过RTSP协议推流给ZLMediaKit。 ffmpeg -re -stream_loop -1 -i "D:\workplace\armgb\public\1.fileh264" -vcodec h264 -f rtsp rtsp://127.0.0.1/live/test 若想将本…...

基于YOLOv5s的无人机航拍输电线瓷瓶检测(附数据集与操作步骤)

本文主要内容:详细介绍了无人机航拍输电线瓷瓶检测的整个过程&#xff0c;从创建数据集到训练模型再到预测结果全部可视化操作与分析。 文末有数据集获取方式&#xff0c;请先看检测效果 现状 输电线路绝缘瓷瓶的检测主要依赖人工巡检。巡检人员需携带专业设备&#xff0c;攀…...