array(2) { ["docs"]=> array(10) { [0]=> array(10) { ["id"]=> string(9) "308660876" ["text"]=> string(45) "安全测试前置实践1-白盒&黑盒扫描" ["intro"]=> string(411) "本文我们将以围绕系统安全质量提升为目标,讲述在安全前置扫描上实践开展过程。希望通过此篇文章,帮助大家更深入、透彻地了解安全测试,能快速开展安全测试。 作者:京东物流 陈维 一、引言 G.J.Myers在《软件测试的艺术》中提出:从心理学角度来说,测试是一个为了寻找错误而运行程序的过程。 " ["username"]=> string(12) "jingdongkeji" ["tagsname"]=> string(39) "前端|安全|黑盒测试|白盒测试" ["tagsid"]=> string(29) "["160","2823","14120","5741"]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1681206002" ["_id"]=> string(9) "308660876" } [1]=> array(10) { ["id"]=> string(9) "308660875" ["text"]=> string(24) "vulnhub靶场之ORASI: 1" ["intro"]=> string(256) "准备: 攻击机:虚拟机kali、本机win10。 靶机:Orasi: 1,下载地址:https://download.vulnhub.com/orasi/Orasi.ova,下载后直接vbox打开即可。 知识点:hex编码、ida逆向、AndroidKiller逆向、ffuf爆破、ssti漏洞、s" ["username"]=> string(6) "upfine" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1681204802" ["_id"]=> string(9) "308660875" } [2]=> array(10) { ["id"]=> string(9) "308660874" ["text"]=> string(92) "C# Kafka重置到最新的偏移量,即从指定的Partition订阅消息使用Assign方法" ["intro"]=> string(428) "在使用Kafka的过程中,消费者断掉之后,再次开始消费时,消费者会从断掉时的位置重新开始消费。 场景再现:比如昨天消费者晚上断掉了,今天上午我们会发现kafka消费的数据不是最新的,而是昨天晚上的数据,由于数据量比较多,也不会及时的消费到今天上午的数据,这个时候就需要我们对偏移量进行重置为最新的,以" ["username"]=> string(15) "Poetwithapistol" ["tagsname"]=> string(10) ".NET|Kafka" ["tagsid"]=> string(13) "["300","440"]" ["catesname"]=> string(4) ".NET" ["catesid"]=> string(7) "["119"]" ["createtime"]=> string(10) "1681203303" ["_id"]=> string(9) "308660874" } [3]=> array(10) { ["id"]=> string(9) "308660873" ["text"]=> string(129) "迁移学习()《Attract, Perturb, and Explore: Learning a Feature Alignment Network for Semi-supervised Domain Adaptation》" ["intro"]=> string(194) "论文信息 论文标题:Attract, Perturb, and Explore: Learning a Feature Alignment Network for Semi-supervised Domain Adaptation论文作者:Taekyung Kim论文来源:2020 ECCV论文地" ["username"]=> string(12) "BlairGrowing" ["tagsname"]=> string(0) "" ["tagsid"]=> string(2) "[]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1681203302" ["_id"]=> string(9) "308660873" } [4]=> array(10) { ["id"]=> string(9) "308660872" ["text"]=> string(92) "C# Kafka重置到最新的偏移量,即从指定的Partition订阅消息使用Assign方法" ["intro"]=> string(428) "在使用Kafka的过程中,消费者断掉之后,再次开始消费时,消费者会从断掉时的位置重新开始消费。 场景再现:比如昨天消费者晚上断掉了,今天上午我们会发现kafka消费的数据不是最新的,而是昨天晚上的数据,由于数据量比较多,也不会及时的消费到今天上午的数据,这个时候就需要我们对偏移量进行重置为最新的,以" ["username"]=> string(10) "goodboydcc" ["tagsname"]=> string(10) ".NET|Kafka" ["tagsid"]=> string(13) "["300","440"]" ["catesname"]=> string(4) ".NET" ["catesid"]=> string(7) "["119"]" ["createtime"]=> string(10) "1681202402" ["_id"]=> string(9) "308660872" } [5]=> array(10) { ["id"]=> string(9) "308660870" ["text"]=> string(42) "Django怎么使用原生SQL查询数据库" ["intro"]=> string(392) "这篇文章主要介绍“Django怎么使用原生SQL查询数据库”,在日常操作中,相信很多人在Django怎么使用原生SQL查询数据库问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Django怎么使用原生SQL查询数据库”的疑惑有所帮助!接下来,请跟着小编一起来学习吧! D" ["username"]=> NULL ["tagsname"]=> string(20) "django|sql|数据库" ["tagsid"]=> NULL ["catesname"]=> string(0) "" ["catesid"]=> NULL ["createtime"]=> string(10) "1681201981" ["_id"]=> string(9) "308660870" } [6]=> array(10) { ["id"]=> string(9) "308660871" ["text"]=> string(37) "Express怎么实现定时发送邮件" ["intro"]=> string(432) "今天小编给大家分享一下Express怎么实现定时发送邮件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。 在开发中我们有时候需要每隔 一段时间发送一次电子邮件,或者在某个特定的时间进行发" ["username"]=> NULL ["tagsname"]=> string(7) "express" ["tagsid"]=> NULL ["catesname"]=> string(0) "" ["catesid"]=> NULL ["createtime"]=> string(10) "1681201981" ["_id"]=> string(9) "308660871" } [7]=> array(10) { ["id"]=> string(9) "308660869" ["text"]=> string(29) "mysql运维------分库分表" ["intro"]=> string(412) "1. 介绍 问题分析: 随着互联网以及移动互联网的发展,应用系统的数据量也是成指数式增长,若采用单数据库进行数据存储,存在以下性能瓶颈: IO瓶颈:热点数据太多,数据库缓存不足,产生大量磁盘IO,效率较低。请求数据太多,带宽不够,网络IO瓶颈。CPU瓶颈:排序、分组、连接查询、聚合统计等SQL会耗费" ["username"]=> string(13) "qds1401744017" ["tagsname"]=> string(5) "mysql" ["tagsid"]=> string(7) "["237"]" ["catesname"]=> string(0) "" ["catesid"]=> string(2) "[]" ["createtime"]=> string(10) "1681200304" ["_id"]=> string(9) "308660869" } [8]=> array(10) { ["id"]=> string(9) "308660868" ["text"]=> string(41) "ASP.NET Core - 缓存之内存缓存(下)" ["intro"]=> string(292) "话接上篇 [ASP.NET Core - 缓存之内存缓存(上)],所以这里的目录从 2.4 开始。 2.4 MemoryCacheEntryOptions MemoryCacheEntryOptions 是内存缓存配置类,可以通过它配置缓存相关的策略。除了上面讲到的过期时间,我们还能够设置下面这些" ["username"]=> string(6) "wewant" ["tagsname"]=> string(12) "asp.net core" ["tagsid"]=> string(7) "["179"]" ["catesname"]=> string(25) "APS.NET Core 系列总结" ["catesid"]=> string(9) "["15288"]" ["createtime"]=> string(10) "1681200302" ["_id"]=> string(9) "308660868" } [9]=> array(10) { ["id"]=> string(9) "308660867" ["text"]=> string(9) "SPI协议" ["intro"]=> string(334) "SPI协议是由摩托罗拉公司提出的通讯协议(Serial Peripheral Interface),即串行外设接口。广泛用在ADC、LCD等设备与MCU间,要求通讯速率较高的场合。区分它与I2C协议差异以及FLASH存储器与EEPROM存储器的区别。下面我们分别对SPI协议的物理层及协议层进行讲解。" ["username"]=> string(8) "Kaelthas" ["tagsname"]=> string(5) "STM32" ["tagsid"]=> string(8) "["1311"]" ["catesname"]=> string(5) "STM32" ["catesid"]=> string(8) "["1139"]" ["createtime"]=> string(10) "1681199702" ["_id"]=> string(9) "308660867" } } ["count"]=> int(5621682) } 源码分析笔记Vector - 爱码网

概述

继承抽象类AbStractList,实现接口List、RandomAccess、Cloneable以及序列化接口
默认容量大小为10,扩容增量为0,扩容为原容量的2倍
如设置的增量大于0,则扩容为(原容量+增量)
支持随机访问,添加删除元素较慢
线程安全的动态数组,方法加上了synchronized同步锁,故性能较低
源码

字段信息

public class Vector<E>
extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
/**
* 保存数据的数组
*/
protected Object[] elementData;

/**
* 元素个数
*/
protected int elementCount;

/**
* 容量增量,可以指定扩容时,容量扩大多少
*/
protected int capacityIncrement;

/** 序列化ID */
private static final long serialVersionUID = -2767605614048989439L;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
构造方法

/**
*主要的构造方法,内部也是调用该方法
*initialCapacity 初始容量
*capacityIncrement 增量,每次扩容时增加的容量大小
*/
public Vector(int initialCapacity, int capacityIncrement) {
super();
//判断初始容量的合法性
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
//创建initialCapacity大小的Object数组
this.elementData = new Object[initialCapacity];
//指定增量大小
this.capacityIncrement = capacityIncrement;
}

/**
*指定初始容量,但不指定增量
*也就是增量为0
*/
public Vector(int initialCapacity) {
//调用2个参数的构造方法
this(initialCapacity, 0);
}

/**
*无参构造方法,默认初始容量为10,增量为0
*/
public Vector() {
this(10);
}
/**
*传入另一集合,增量为传入集合的长度
*/
public Vector(Collection<? extends E> c) {
elementData = c.toArray();
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
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
添加元素

/**
*添加元素
*/
public synchronized boolean add(E e) {
//数组修改次数记录
modCount++;
//容量检查,判断是否需要扩容
ensureCapacityHelper(elementCount + 1);
//添加元素
elementData[elementCount++] = e;
return true;
}

/**
*在指定位置添加元素
*/
public void add(int index, E element) {
insertElementAt(element, index);
}

public synchronized void insertElementAt(E obj, int index) {
modCount++;
//下标合法性检查,index不能大于元素个数
if (index > elementCount) {
throw new ArrayIndexOutOfBoundsException(index
+ " > " + elementCount);
}
//容量检查
ensureCapacityHelper(elementCount + 1);
//数组拷贝
//将index开始,长度为elementCount-index的数组,往后移动一位
System.arraycopy(elementData, index, elementData, index + 1, elementCount - index);
elementData[index] = obj;
elementCount++;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
扩容操作

/**
*容量检查,判断是否需要扩容
*/
private void ensureCapacityHelper(int minCapacity) {
// overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}

/**
*最大容量门限,最大容量为Integer.MAX_VALUE
*/
private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

/**
*实际的扩容方法
*/
private void grow(int minCapacity) {
// 当前容量
int oldCapacity = elementData.length;
//计算新的容量
//如果没有传入增量大小,则扩容为当前容量的2倍(2*oldCapacity)
//如果有设置增量大小,如果增量小于0,则扩容2倍;如果大于0,则扩容为(oldCapacity+增量大小)
int newCapacity = oldCapacity + ((capacityIncrement > 0) ?
capacityIncrement : oldCapacity);
//判断新的容量的大小是否满足最小容量需要,如不满足,则直接扩容为所需要的容量大小
if (newCapacity - minCapacity < 0)
newCapacity = minCapacity;
//判断是否达到最大容量门限
if (newCapacity - MAX_ARRAY_SIZE > 0)
newCapacity = hugeCapacity(minCapacity);
//数组扩容,该方法代价较大,将数组扩容为newCapacity大小
elementData = Arrays.copyOf(elementData, newCapacity);
}

private static int hugeCapacity(int minCapacity) {
//判断所需的最小容量是否整数溢出
if (minCapacity < 0) // overflow
throw new OutOfMemoryError();
//扩容的最大容量大小为Integer.MAX_VALUE
return (minCapacity > MAX_ARRAY_SIZE) ?
Integer.MAX_VALUE :
MAX_ARRAY_SIZE;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
小结

默认扩容大小为原容量的2倍;设置了大于0的增量,扩容大小则为原容量+增量
最大容量为Integer.MAX_VALUE,门限为Integer.MAX_VALUE-8
添加元素顺序:传入所需的容量大小 -> 判断是否需要扩容 -> 执行扩容操作(计算新的容量大小、调用Arrays.copyOf(object[], newCapacity)) -> 添加元素
因Arrays.copyOf()代价较大,故在创建Vector时应尽量设置集合大小
删除元素

/**
*根据指定下标删除元素,返回删除元素的值
*/
public synchronized E remove(int index) {
modCount++;
if (index >= elementCount)
throw new ArrayIndexOutOfBoundsException(index);
//获取保存index下标的元素值
E oldValue = elementData(index);
//计算拷贝移动的数组长度
int numMoved = elementCount - index - 1;
//长度大于0,则将index下标后面的元素往前拷贝移动一个位置
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
//将新空出来的下标位置填充为null,让gc可以回收内存
elementData[--elementCount] = null; // Let gc do its work

return oldValue;
}
/**
*删除特定元素值,因为不确定集合中是否存在该元素,故返回boolean值,返回是否删除成功
*/
public boolean remove(Object o) {
return removeElement(o);
}

public synchronized boolean removeElement(Object obj) {
modCount++;
//获取指定元素值第一次出现的下标,没有找到则返回-1
int i = indexOf(obj);
if (i >= 0) {
//根据指定下标删除元素
removeElementAt(i);
return true;
}
//没有找到指定元素,返回false,删除失败
return false;
}

/**
*根据下标删除元素,无返回值
*/
public synchronized void removeElementAt(int index) {
modCount++;
//判断下标是否合法
if (index >= elementCount) {
throw new ArrayIndexOutOfBoundsException(index + " >= " +
elementCount);
}
else if (index < 0) {
throw new ArrayIndexOutOfBoundsException(index);
}
//同remove(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 */
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
小结

remove(int index) 与 removeElementAt(int index)区别为remove方法返回删除元素的值,而removeElementAt无返回值
根据元素内容删除不一定删除成功,可能集合中无目标元素
手动缩容

/**
*手动缩小内部数组容量
*/
public synchronized void trimToSize(http://www.amjmh.com) {
modCount++;
int oldCapacity = elementData.length;
//判断数组中元素个数与数组长度大小
if (elementCount < oldCapacity) {
elementData = Arrays.copyOf(elementData, elementCount);
}
}
1
2
3
4
5
6
7
8
9
10
11
手动扩容

/**
*传入一个容量大小,大于当前集合容量,将扩容
*/
public synchronized void ensureCapacity(int minCapacity) {
if (minCapacity > 0) {
modCount++;
//判断minCapacity是否大于数组容量,是否执行扩容操作
ensureCapacityHelper(minCapacity);
}
}
1
2
3
4
5
6
7
8
9
10
设置集合大小

/**
*设置容量大小
*如果大于当前数组容量,则扩容
*如果小于当前数组容量,则将操出的部分填充为null,让gc可以回收该部分内存
*/
public synchronized void setSize(int newSize) {
modCount++;
if (newSize > elementCount) {
ensureCapacityHelper(newSize);
} else {
for (int i = newSize ; i < elementCount ; i++) {
elementData[i] = null;
}
}
elementCount = newSize;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
与ArrayList比较

Vector是线程安全的,在方法上加了synchronize同步锁,故访问速度比ArrayList慢,性能较低;最好使用ArrayList,因为同步操作完全可以有我们自己来控制
ArrayList扩容是1.5倍(oldCapacity >> 1),Vector默认是2倍
总结

使用时最好指定容量大小
删除元素时不会自己缩容
---------------------

相关文章:

  • 2021-09-28
  • 2022-12-23
  • 2022-12-23
  • 2022-12-23
  • 2022-01-13
  • 2021-04-08
  • 2021-04-03
  • 2021-07-14
猜你喜欢
  • 2021-12-18
  • 2021-06-14
  • 2021-08-23
  • 2021-09-05
相关资源
相似解决方案