百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

Java集合类之Vector

liuian 2025-01-14 15:19 33 浏览

Vector源码解析

老样子,话不多说先上一张UML类图

  1. Vector的初始化构造方法

无参构造(默认调用初始化容量的构造方法,默认容量为10)

public Vector() {
	//这里会调用Vector带容量参数的构造方法默认容量为10
	//这里和ArrayList不同的是ArrayList在调用add方法才初始化容量
    this(10);
}

指定初始化容量大小

public Vector(int initialCapacity) {
	//这里调用下面指定初始化容量和增长系数的构造方法,默认增长系数为0
   	this(initialCapacity, 0);
}

指定初始化容量和增长系数

public Vector(int initialCapacity, int capacityIncrement) {
    super();
    //容量不能小于0
    if (initialCapacity < 0)
        throw new IllegalArgumentException("Illegal Capacity: "+
                                           initialCapacity);
    //为对象数组指定初始化容量
    this.elementData = new Object[initialCapacity];
    //设置增长系数
    this.capacityIncrement = capacityIncrement;
}

使用另外一个集合构造该集合

public Vector(Collection<? extends E> c) {
	//将传入集合转成数组
    elementData = c.toArray();
    //Vector的元素个数就是集合的长度
    elementCount = elementData.length;
    // c.toArray might (incorrectly) not return Object[] (see 6260652)
    //判断添加进的元素对象是否为对象数组类型
    if (elementData.getClass() != Object[].class)
    	//进行浅拷贝将类型转化为对象数组
        elementData = Arrays.copyOf(elementData, elementCount, Object[].class);
}
  1. add(E e)方法
public synchronized boolean add(E e) {
	//更新操作次数
    modCount++;
    //确保可以保存元素的容量,如有必要会进行扩容
    ensureCapacityHelper(elementCount + 1);
    elementData[elementCount++] = e;
    return true;
}
//如有必要,增加当前数组的容量,以确保至少可以保存minCapacity容量参数指定的元素个数
private void ensureCapacityHelper(int minCapacity) {
    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
    	//Vector的扩容方法
        grow(minCapacity);
}
//扩容方法
private void grow(int minCapacity) {
    // overflow-conscious code
    //原Vector容量值
    int oldCapacity = elementData.length;
    //如果有给capacityIncrement设置增长系数的话,就加上该系数值来扩容,否则将原先的数组容量变为2*oldCapacity
    int newCapacity = oldCapacity + ((capacityIncrement > 0) ?
                                     capacityIncrement : oldCapacity);
    //如果重新的设置的容量值还要小于最小要求的容量值得话                                 
    if (newCapacity - minCapacity < 0)
    	//就将最小的容量值赋值给新的容量
        newCapacity = minCapacity;
    //如果新容量值比限制的最大容量还要大的话
    if (newCapacity - MAX_ARRAY_SIZE > 0)
    	//重新设置大小
        newCapacity = hugeCapacity(minCapacity);
    //将原先数组的元素浅拷贝到一个新的数组
    elementData = Arrays.copyOf(elementData, newCapacity);
}

//获取最大容量
private static int hugeCapacity(int minCapacity) {
    if (minCapacity < 0) // overflow
        throw new OutOfMemoryError();
    return (minCapacity > MAX_ARRAY_SIZE) ?
        Integer.MAX_VALUE :
        MAX_ARRAY_SIZE;
}

add(int index, E element)方法(指定位置添加元素)

public void add(int index, E element) {
    insertElementAt(element, index);
}

public synchronized void insertElementAt(E obj, int index) {
    modCount++;
    if (index > elementCount) {
        throw new ArrayIndexOutOfBoundsException(index
                                                 + " > " + elementCount);
    }
    //如上确保容量
    ensureCapacityHelper(elementCount + 1);
    //将指定索引到末尾的元素分别往左移动一位
    System.arraycopy(elementData, index, elementData, index + 1, elementCount - index);
    elementData[index] = obj;
    elementCount++;
}

addAll(Collection<? extends E> c)方法

public synchronized boolean addAll(Collection<? extends E> c) {
	//更新操作次数
    modCount++;
    //将传入的集合转为数组
    Object[] a = c.toArray();
    //获取传入集合的长度
    int numNew = a.length;
    //如上确保容量
    ensureCapacityHelper(elementCount + numNew);
    //将指定索引到末尾的元素分别往左移动一位
    System.arraycopy(a, 0, elementData, elementCount, numNew);
    //集合的大小为旧集合的大小+新集合的大小
    elementCount += numNew;
    return numNew != 0;
}
  1. remove(int index)方法
public synchronized E remove(int index) {
    modCount++;
    //对index进行边界检查
    if (index >= elementCount)
        throw new ArrayIndexOutOfBoundsException(index);
    //获取要删除坐标的元素
    E oldValue = elementData(index);
    //计算指定索引与末尾元素的个数
    int numMoved = elementCount - index - 1;
    //如果个数大于0
    if (numMoved > 0)
    	//将指定索引到末尾的元素分别往左移动一位
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    //递减元素个数,并将末尾元素置空
    elementData[--elementCount] = null; // Let gc do its work

    return oldValue;
}

remove(Object o)方法

public boolean remove(Object o) {
	//调用removeElement(Object obj)方法
    return removeElement(o);
}

public synchronized boolean removeElement(Object obj) {
	//更新操作次数
    modCount++;
    //得到要删除元素的坐标
    int i = indexOf(obj);
    if (i >= 0) {
    	//根据坐标删除元素
        removeElementAt(i);
        return true;
    }
    return false;
}

public synchronized void removeElementAt(int index) {
    modCount++;
    //对index进行边界检查
    if (index >= elementCount) {
        throw new ArrayIndexOutOfBoundsException(index + " >= " +
                                                 elementCount);
    }
    else if (index < 0) {
        throw new ArrayIndexOutOfBoundsException(index);
    }
    //计算指定索引与末尾元素的个数
    int j = elementCount - index - 1;
    if (j > 0) {
    	//将指定索引到末尾的元素分别往左移动一位
        System.arraycopy(elementData, index + 1, elementData, index, j);
    }
    //减少集合容量
    elementCount--;
    //递减元素个数,并将末尾元素置空
    elementData[elementCount] = null; /* to let gc do its work */
}

removeAll(Collection<?> c)方法

//删除指定集合中的所有元素
public synchronized boolean removeAll(Collection<?> c) {
	//调用AbstractCollection中的removeAll的方法
    return super.removeAll(c);
}

retainAll(Collection<?> c)方法

//删除非集合c中的元素
public synchronized boolean retainAll(Collection<?> c) {
    return super.retainAll(c);
}

clear()方法

public void clear() {
	//清空集合方法调用上面removeAllElements方法
    removeAllElements();
}
  1. get(int index)方法
public synchronized E get(int index) {
	//对index进行边界检查
    if (index >= elementCount)
        throw new ArrayIndexOutOfBoundsException(index);
	//根据下标获取元素
    return elementData(index);
}
E elementData(int index) {
	//在数组中根据下标获取元素
    return (E) elementData[index];
}
  1. set(int index, E element)方法
public synchronized E set(int index, E element) {
	//对index进行边界检查
    if (index >= elementCount)
        throw new ArrayIndexOutOfBoundsException(index);
	//获取index坐标的元素
    E oldValue = elementData(index);
    //将set的元素放入数组指定下标
    elementData[index] = element;
    return oldValue;
}

总结:

  1. Vector是使用数组保存数据,和ArrayList一样
  2. 在无参的构造方法中,默认的初始容量为10,增长系数为0比ArrayList多了一个增长系数的概念
  3. Vector类是线程安全的List,其底层是通过Synchronized关键字实现的(同步方法),ArrayList是线程不安全的
  4. 扩容机制:如果增长系数不位 0 那么就是当前容量 + 增长系数,否则就是2倍,ArrayList为1.5倍左右

相关推荐

GANs为何引爆机器学习?这篇基于TensorFlow的实例教程为你解惑!

「机器人圈导览」:生成对抗网络无疑是机器学习领域近三年来最火爆的研究领域,相关论文层出不求,各种领域的应用层出不穷。那么,GAN到底如何实践?本文编译自Medium,该文作者以一朵玫瑰花为例,详细阐...

高丽大学等机构联合发布StarGAN:可自定义表情和面部特征

原文来源:arXiv、GitHub作者:YunjeyChoi、MinjeChoi、MunyoungKim、Jung-WooHa、SungKim、JaegulChoo「雷克世界」编译:嗯~...

TensorFlow和PyTorch相继发布最新版,有何变化

原文来源:GitHub「机器人圈」编译:嗯~阿童木呀、多啦A亮Tensorflow主要特征和改进在Tensorflow库中添加封装评估量。所添加的评估量列表如下:1.深度神经网络分类器(DNNCl...

「2022 年」崔庆才 Python3 爬虫教程 - 深度学习识别滑动验证码缺口

上一节我们使用OpenCV识别了图形验证码躯壳欧。这时候就有朋友可能会说了,现在深度学习不是对图像识别很准吗?那深度学习可以用在识别滑动验证码缺口位置吗?当然也是可以的,本节我们就来了解下使用深度...

20K star!搞定 LLM 微调的开源利器

LLM(大语言模型)微调一直都是老大难问题,不仅因为微调需要大量的计算资源,而且微调的方法也很多,要去尝试每种方法的效果,需要安装大量的第三方库和依赖,甚至要接入一些框架,可能在还没开始微调就已经因为...

大模型DeepSeek本地部署后如何进行自定义调整?

1.理解模型架构a)查看深度求索官方文档或提供的源代码文件,了解模型的结构、输入输出格式以及支持的功能。模型是否为预训练权重?如果是,可以在预训练的基础上进行微调(Fine-tuning)。是否需要...

因配置不当,约5000个AI模型与数据集在公网暴露

除了可访问机器学习模型外,暴露的数据还可能包括训练数据集、超参数,甚至是用于构建模型的原始数据。前情回顾·人工智能安全动态向ChatGPT植入恶意“长期记忆”,持续窃取用户输入数据多模态大语言模型的致...

基于pytorch的深度学习人员重识别

基于pytorch的深度学习人员重识别Torchreid是一个库。基于pytorch的深度学习人员重识别。特点:支持多GPU训练支持图像的人员重识别与视频的人员重识别端到端的训练与评估简单的re...

DeepSeek本地部署:轻松训练你的AI模型

引言:为什么选择本地部署?在AI技术飞速发展的今天,越来越多的企业和个人希望将AI技术应用于实际场景中。然而,对于一些对数据隐私和计算资源有特殊需求的用户来说,云端部署可能并不是最佳选择。此时,本地部...

谷歌今天又开源了,这次是Sketch-RNN

前不久,谷歌公布了一项最新技术,可以教机器画画。今天,谷歌开源了代码。在我们研究其代码之前,首先先按要求设置Magenta环境。(https://github.com/tensorflow/magen...

Tensorflow 使用预训练模型训练的完整流程

前面已经介绍了深度学习框架Tensorflow的图像的标注和训练数据的准备工作,本文介绍一下使用预训练模型完成训练并导出训练的模型。1.选择预训练模型1.1下载预训练模型首先需要在Tensorf...

30天大模型调优学习计划(30分钟训练大模型)

30天大模型调优学习计划,结合Unsloth和Lora进行大模型微调,掌握大模型基础知识和调优方法,熟练应用。第1周:基础入门目标:了解大模型基础并熟悉Unsloth等工具的基本使用。Day1:大模...

python爬取喜马拉雅音频,json参数解析

一.抓包分析json,获取加密方式1.抓包获取音频界面f12打开抓包工具,播放一个(非vip)视频,点击“媒体”单击打开可以复制URL,发现就是我们要的音频。复制“CKwRIJEEXn-cABa0Tg...

五、JSONPath使用(Python)(json数据python)

1.安装方法pipinstalljsonpath2.jsonpath与Xpath下面表格是jsonpath语法与Xpath的完整概述和比较。Xpathjsonpath概述/$根节点.@当前节点...

Python网络爬虫的时候json=就是让你少写个json.dumps()

大家好,我是皮皮。一、前言前几天在Python白银交流群【空翼】问了一个Python网络爬虫的问题,提问截图如下:登录请求地址是这个:二、实现过程这里【甯同学】给了一个提示,如下所示:估计很多小伙伴和...