- 浏览: 188345 次
- 性别:
- 来自: 上海
文章分类
最新评论
一、Vector简介
Vector可以实现可增长的对象数组。与数组一样,它包含可以使用整数索引进行访问的组件。不过,Vector的大小是可以增加或者减小的,以便适应创建Vector后进行添加或者删除操作。
Vector实现List接口,继承AbstractList类,所以我们可以将其看做队列,支持相关的添加、删除、修改、遍历等功能。
Vector实现RandmoAccess接口,即提供了随机访问功能,提供提供快速访问功能。在Vector我们可以直接访问元素。
Vector 实现了Cloneable接口,支持clone()方法,可以被克隆。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
public class Vector<E>
extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable
Vector提供了四个构造函数:
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
/**
* 构造一个空向量,使其内部数据数组的大小为 10,其标准容量增量为零。
*/
public Vector() {
this(10);
}
/**
* 构造一个包含指定 collection 中的元素的向量,这些元素按其 collection 的迭代器返回元素的顺序排列。
*/
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);
}
/**
* 使用指定的初始容量和等于零的容量增量构造一个空向量。
*/
public Vector(int initialCapacity) {
this(initialCapacity, 0);
}
/**
* 使用指定的初始容量和容量增量构造一个空的向量。
*/
public Vector(int initialCapacity, int capacityIncrement) {
super();
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
this.elementData = new Object[initialCapacity];
this.capacityIncrement = capacityIncrement;
}
在成员变量方面,Vector提供了elementData , elementCount, capacityIncrement三个成员变量。其中
elementData :"Object[]类型的数组",它保存了Vector中的元素。按照Vector的设计elementData为一个动态数组,可以随着元素的增加而动态的增长,其具体的增加方式后面提到(ensureCapacity方法)。如果在初始化Vector时没有指定容器大小,则使用默认大小为10.
elementCount:Vector 对象中的有效组件数。
capacityIncrement:向量的大小大于其容量时,容量自动增加的量。如果在创建Vector时,指定了capacityIncrement的大小;则,每次当Vector中动态数组容量增加时>,增加的大小都是capacityIncrement。如果容量的增量小于等于零,则每次需要增大容量时,向量的容量将增大一倍。
同时Vector是线程安全的!
二、源码解析
对于源码的解析,LZ在这里只就增加(add)删除(remove)两个方法进行讲解。
2.1增加:add(E e)
add(E e):将指定元素添加到此向量的末尾。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
public synchronized boolean add(E e) {
modCount++;
ensureCapacityHelper(elementCount + 1); //确认容器大小,如果操作容量则扩容操作
elementData[elementCount++] = e; //将e元素添加至末尾
return true;
}
这个方法相对而言比较简单,具体过程就是先确认容器的大小,看是否需要进行扩容操作,然后将E元素添加到此向量的末尾。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
private void ensureCapacityHelper(int minCapacity) {
//如果
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
/**
* 进行扩容操作
* 如果此向量的当前容量小于minCapacity,则通过将其内部数组替换为一个较大的数组俩增加其容量。
* 新数据数组的大小姜维原来的大小 + capacityIncrement,
* 除非 capacityIncrement 的值小于等于零,在后一种情况下,新的容量将为原来容量的两倍,不过,如果此大小仍然小于 minCapacity,则新容量将为 minCapacity。
*/
private void grow(int minCapacity) {
int oldCapacity = elementData.length; //当前容器大小
/*
* 新容器大小
* 若容量增量系数(capacityIncrement) > 0,则将容器大小增加到capacityIncrement
* 否则将容量增加一倍
*/
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);
}
/**
* 判断是否超出最大范围
* MAX_ARRAY_SIZE:private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
*/
private static int hugeCapacity(int minCapacity) {
if (minCapacity < 0)
throw new OutOfMemoryError();
return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE;
}
对于Vector整个的扩容过程,就是根据capacityIncrement确认扩容大小的,若capacityIncrement <= 0 则扩大一倍,否则扩大至capacityIncrement 。当然这个容量的最大范围为Integer.MAX_VALUE即,2^32 - 1,所以Vector并不是可以无限扩充的。
2.2、remove(Object o)
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
/**
* 从Vector容器中移除指定元素E
*/
public boolean remove(Object o) {
return removeElement(o);
}
public synchronized boolean removeElement(Object obj) {
modCount++;
int i = indexOf(obj); //计算obj在Vector容器中位置
if (i >= 0) {
removeElementAt(i); //移除
return true;
}
return false;
}
public synchronized void removeElementAt(int index) {
modCount++; //修改次数+1
if (index >= elementCount) { //删除位置大于容器有效大小
throw new ArrayIndexOutOfBoundsException(index + " >= " + elementCount);
}
else if (index < 0) { //位置小于 < 0
throw new ArrayIndexOutOfBoundsException(index);
}
int j = elementCount - index - 1;
if (j > 0) {
//从指定源数组中复制一个数组,复制从指定的位置开始,到目标数组的指定位置结束。
//也就是数组元素从j位置往前移
System.arraycopy(elementData, index + 1, elementData, index, j);
}
elementCount--; //容器中有效组件个数 - 1
elementData[elementCount] = null; //将向量的末尾位置设置为null
}
因为Vector底层是使用数组实现的,所以它的操作都是对数组进行操作,只不过其是可以随着元素的增加而动态的改变容量大小,其实现方法是是使用Arrays.copyOf方法将旧数据拷贝到一个新的大容量数组中。Vector的整个内部实现都比较简单,这里就不在重述了。
三、Vector遍历
Vector支持4种遍历方式。
3.1、随机访问
因为Vector实现了RandmoAccess接口,可以通过下标来进行随机访问。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
for(int i = 0 ; i < vec.size() ; i++){
value = vec.get(i);
}
3.2、迭代器
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
Iterator it = vec.iterator();
while(it.hasNext()){
value = it.next();
//do something
}
3.2、for循环
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
for(Integer value:vec){
//do something
}
3.4、Enumeration循环
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
Vector vec = new Vector<>();
Enumeration enu = vec.elements();
while (enu.hasMoreElements()) {
value = (Integer)enu.nextElement();
}
以上转自http://blog.csdn.net/chenssy/article/details/37520981
Vector可以实现可增长的对象数组。与数组一样,它包含可以使用整数索引进行访问的组件。不过,Vector的大小是可以增加或者减小的,以便适应创建Vector后进行添加或者删除操作。
Vector实现List接口,继承AbstractList类,所以我们可以将其看做队列,支持相关的添加、删除、修改、遍历等功能。
Vector实现RandmoAccess接口,即提供了随机访问功能,提供提供快速访问功能。在Vector我们可以直接访问元素。
Vector 实现了Cloneable接口,支持clone()方法,可以被克隆。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
public class Vector<E>
extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable
Vector提供了四个构造函数:
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
/**
* 构造一个空向量,使其内部数据数组的大小为 10,其标准容量增量为零。
*/
public Vector() {
this(10);
}
/**
* 构造一个包含指定 collection 中的元素的向量,这些元素按其 collection 的迭代器返回元素的顺序排列。
*/
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);
}
/**
* 使用指定的初始容量和等于零的容量增量构造一个空向量。
*/
public Vector(int initialCapacity) {
this(initialCapacity, 0);
}
/**
* 使用指定的初始容量和容量增量构造一个空的向量。
*/
public Vector(int initialCapacity, int capacityIncrement) {
super();
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
this.elementData = new Object[initialCapacity];
this.capacityIncrement = capacityIncrement;
}
在成员变量方面,Vector提供了elementData , elementCount, capacityIncrement三个成员变量。其中
elementData :"Object[]类型的数组",它保存了Vector中的元素。按照Vector的设计elementData为一个动态数组,可以随着元素的增加而动态的增长,其具体的增加方式后面提到(ensureCapacity方法)。如果在初始化Vector时没有指定容器大小,则使用默认大小为10.
elementCount:Vector 对象中的有效组件数。
capacityIncrement:向量的大小大于其容量时,容量自动增加的量。如果在创建Vector时,指定了capacityIncrement的大小;则,每次当Vector中动态数组容量增加时>,增加的大小都是capacityIncrement。如果容量的增量小于等于零,则每次需要增大容量时,向量的容量将增大一倍。
同时Vector是线程安全的!
二、源码解析
对于源码的解析,LZ在这里只就增加(add)删除(remove)两个方法进行讲解。
2.1增加:add(E e)
add(E e):将指定元素添加到此向量的末尾。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
public synchronized boolean add(E e) {
modCount++;
ensureCapacityHelper(elementCount + 1); //确认容器大小,如果操作容量则扩容操作
elementData[elementCount++] = e; //将e元素添加至末尾
return true;
}
这个方法相对而言比较简单,具体过程就是先确认容器的大小,看是否需要进行扩容操作,然后将E元素添加到此向量的末尾。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
private void ensureCapacityHelper(int minCapacity) {
//如果
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
/**
* 进行扩容操作
* 如果此向量的当前容量小于minCapacity,则通过将其内部数组替换为一个较大的数组俩增加其容量。
* 新数据数组的大小姜维原来的大小 + capacityIncrement,
* 除非 capacityIncrement 的值小于等于零,在后一种情况下,新的容量将为原来容量的两倍,不过,如果此大小仍然小于 minCapacity,则新容量将为 minCapacity。
*/
private void grow(int minCapacity) {
int oldCapacity = elementData.length; //当前容器大小
/*
* 新容器大小
* 若容量增量系数(capacityIncrement) > 0,则将容器大小增加到capacityIncrement
* 否则将容量增加一倍
*/
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);
}
/**
* 判断是否超出最大范围
* MAX_ARRAY_SIZE:private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
*/
private static int hugeCapacity(int minCapacity) {
if (minCapacity < 0)
throw new OutOfMemoryError();
return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE;
}
对于Vector整个的扩容过程,就是根据capacityIncrement确认扩容大小的,若capacityIncrement <= 0 则扩大一倍,否则扩大至capacityIncrement 。当然这个容量的最大范围为Integer.MAX_VALUE即,2^32 - 1,所以Vector并不是可以无限扩充的。
2.2、remove(Object o)
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
/**
* 从Vector容器中移除指定元素E
*/
public boolean remove(Object o) {
return removeElement(o);
}
public synchronized boolean removeElement(Object obj) {
modCount++;
int i = indexOf(obj); //计算obj在Vector容器中位置
if (i >= 0) {
removeElementAt(i); //移除
return true;
}
return false;
}
public synchronized void removeElementAt(int index) {
modCount++; //修改次数+1
if (index >= elementCount) { //删除位置大于容器有效大小
throw new ArrayIndexOutOfBoundsException(index + " >= " + elementCount);
}
else if (index < 0) { //位置小于 < 0
throw new ArrayIndexOutOfBoundsException(index);
}
int j = elementCount - index - 1;
if (j > 0) {
//从指定源数组中复制一个数组,复制从指定的位置开始,到目标数组的指定位置结束。
//也就是数组元素从j位置往前移
System.arraycopy(elementData, index + 1, elementData, index, j);
}
elementCount--; //容器中有效组件个数 - 1
elementData[elementCount] = null; //将向量的末尾位置设置为null
}
因为Vector底层是使用数组实现的,所以它的操作都是对数组进行操作,只不过其是可以随着元素的增加而动态的改变容量大小,其实现方法是是使用Arrays.copyOf方法将旧数据拷贝到一个新的大容量数组中。Vector的整个内部实现都比较简单,这里就不在重述了。
三、Vector遍历
Vector支持4种遍历方式。
3.1、随机访问
因为Vector实现了RandmoAccess接口,可以通过下标来进行随机访问。
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
for(int i = 0 ; i < vec.size() ; i++){
value = vec.get(i);
}
3.2、迭代器
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
Iterator it = vec.iterator();
while(it.hasNext()){
value = it.next();
//do something
}
3.2、for循环
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
for(Integer value:vec){
//do something
}
3.4、Enumeration循环
[java] view plaincopyprint?在CODE上查看代码片派生到我的代码片
Vector vec = new Vector<>();
Enumeration enu = vec.elements();
while (enu.hasMoreElements()) {
value = (Integer)enu.nextElement();
}
以上转自http://blog.csdn.net/chenssy/article/details/37520981
发表评论
文章已被作者锁定,不允许评论。
-
ReentrantLock与Condition
2017-03-17 14:25 526多线程和并发性并不是什么新内容,但是 Java 语言设计中的创 ... -
java linux监控
2017-03-13 17:49 483http://agapple.iteye.com/blog/1 ... -
transient和volatile两个关键字
2017-02-16 09:47 572transient和volatile两个关 ... -
java 锁机制
2016-12-09 13:43 465一段synchronized的代码被 ... -
java 正则表达式
2016-12-02 10:28 516众所周知,在程序开发中,难免会遇到需要匹配、查找、替换、判断字 ... -
java ClassNotFoundException和NoClassDefFoundException的差别
2016-08-17 19:47 907首先从名字上可以看出一类是异常,一类属于错误。异常可以通过异常 ... -
ThreadLocal
2016-07-19 11:10 327ThreadLocal是什么 Thre ... -
java CAS
2016-07-10 14:55 333cas 乐观锁每次不锁定整个线程,在操作之前进行判断。悲观锁独 ... -
concurrenthashmap
2016-07-10 11:11 422hash table虽然性能上不如 ... -
java 线程池的使用
2016-07-10 09:52 3721. 引言 合理利用线程池能够带来三个好处。第一:降低资源消 ... -
java.util.concurrent
2016-07-03 16:24 409我们都知道,在JDK1.5之 ... -
JVM 配置 以及垃圾收集器的选择
2016-04-15 12:36 728JVM监控的关键指标说明: a) FGC的环比增加次数。Zab ... -
jvm实时监控工具
2016-04-09 09:35 461 -
哈希 、一致性哈希、余数式哈希
2016-04-07 16:10 861什么是Hash Hash,一 ... -
jvm dump 相关
2016-03-22 17:22 681http://www.cnblogs.com/edwardla ... -
深入剖析volatile关键字
2016-03-21 16:02 534深入剖析volatile关键字 ... -
java线程安全问题之静态变量、实例变量、局部变量
2016-03-08 12:52 571java多线程编程中,存在很多线程安全问题,至于什么是线程安全 ... -
有状态的bean和无状态的bean的区别
2016-03-08 11:23 1493有状态会话bean :每个用户有自己特有的一个实例,在用户的生 ... -
Java nio详解
2016-01-20 16:30 551http://www.ibm.com/developerwor ... -
java 不定长数组
2015-11-24 15:00 768在调用某个方法时,若是方法的参数个数事先无法确定该如何处理 ...
相关推荐
然而,对于学习和理解C++内存管理和容器实现原理,尝试自己编写一个类似`std::vector`的数据结构是很有价值的。在这个自己实现的`vector`中,我们将探讨其核心功能以及优化的动态内存管理。 1. **基本结构**:一个...
在软件开发领域,尤其是汽车电子软件的开发过程...通过深入学习和实践Vector Cast Train资料,汽车电子软件的开发者可以提升其单元测试能力,确保软件的质量和可靠性,进而满足汽车行业对软件的高安全性和稳定性要求。
在C++编程语言中,`std::vector`是标准模板库(STL)中的一种容器,它提供...在`studyworkspace`这个文件夹中,可能包含了关于`vector`实现的代码示例和练习,通过学习和实践这些内容,你将更深入地掌握`vector`的使用。
本文将深入讲解这两种容器的基本概念、工作原理以及常见操作,以帮助初学者更好地理解和运用。 **一、std::vector简介** `std::vector`是一种动态数组,可以容纳任何类型的元素。它的最大特点是可以在运行时改变...
通过深入理解这个驱动的源代码,我们可以学习到如何与硬件接口交互,如何处理XCP报文,以及如何集成到VECTOR的标定环境中。 总的来说,"XCP Basic Driver.rar"提供的资源对于理解XCP协议和在VECTOR平台上进行标定...
《使用Vector实现学生管理系统》 在C++编程中,`std::vector` 是一个非常重要的容器,它...通过这个项目,我们可以深入理解vector的使用,掌握如何用C++实现简单的数据管理,同时也能提升对C++标准模板库的运用能力。
对于经验丰富的开发者,它也能作为复习和深入学习的资源,确保他们在复杂的项目中保持最佳实践。通过观看和学习,你可以更好地掌握代码测试的关键技术和工具,从而在实际工作中更高效地保证软件的质量和可靠性。
这篇总结将深入探讨在Vector Cast环境下进行UT工程时可能遇到的问题以及解决方案。 1. **Vector Cast介绍**:Vector Cast 是一套全面的静态和动态测试工具,包括单元测试、覆盖率分析、静态分析等功能。它提供了一...
通过这个"Labview Vector CAN"示例,用户不仅可以学习到如何在Labview环境中配置和控制Vector硬件,还能深入理解CAN和LIN通信协议的实现细节。对于开发和测试涉及CAN或LIN的嵌入式系统来说,这是一个非常有价值的...
标题“Startup_Vector_SLP4.rar”暗示了这是一个关于使用Vector工具进行系统启动配置的资源包,特别是...通过深入学习和实践,不仅可以提高工作效率,还能增强对AUTOSAR体系结构的理解,有助于解决实际项目中的问题。
首先,让我们深入了解`std::vector`的基本概念。`std::vector`是C++标准库中的一个模板类,它提供了一个动态大小的数组,可以容纳任意类型的元素。它允许我们在运行时添加、删除元素,且提供了下标访问、迭代器遍历...
在"Vector_XCP_Basic"这个压缩包中,我们能深入学习XCP的基础概念和应用。 **1. XCP协议介绍** XCP是基于CAN(Controller Area Network)的通信协议,但不仅限于CAN,还可以扩展到其他网络如FlexRay或Ethernet。XCP...
总的来说,"Vector-XCP 源代码"对于深入理解XCP协议、开发自己的ECU标定工具或者增强现有工具的功能具有重要意义。通过学习和使用这些源代码,开发者可以提升ECU软件开发的效率,确保项目的质量和兼容性,并且能够...
### Java中的Vector定义与用法详解 #### 一、Vector简介 `Vector`是Java集合框架中的一个类,它提供了一种线程安全的动态数组实现。...理解`Vector`的特性和用法对于深入学习Java集合框架是非常有益的。
C++中的模板类`std::vector`是标准模板库(STL)的一部分,它提供了一个动态数组的功能,允许程序员在运行时改变数组的大小。`vector`类在内存管理上...通过深入学习和实践,开发者可以充分利用其特性,解决实际问题。
《深入学习C++的好书》是一本专注于C++编程语言深度探索的专业书籍,它涵盖了C++的核心概念和技术,包括函数、类、模板以及标准库等多个关键领域。这本书以英文为载体,面向那些希望通过深入理解C++来提升自己编程...
下面我们将深入探讨`vector`的基础知识,包括其基本操作、迭代器的使用以及常见用法。 1. **`vector`的创建与初始化** - 创建空向量:`std::vector<int> vec;` - 初始化指定大小的向量:`std::vector<int> vec(10...
《HexView(Vector)V1.09.01:深入理解十六进制查看与编辑工具》 在计算机科学和IT领域,理解数据的底层表示是至关重要的,尤其是在进行二进制分析、软件逆向工程或者数据恢复时。HexView(Vector)V1.09.01是一款...
通过实践和学习这个文档,你可以更深入地了解`std::vector`如何在实际项目中发挥作用。 总之,`std::vector`是C++中常用的动态数组,提供了一套完整的方法来处理和操作数据。在VC++6.0或其他C++环境中,通过编写和...
在C++编程中,`std::vector`是一个非常重要的动态数组容器,它允许高效地进行元素的添加和删除。...通过这个简化的版本,我们可以理解`std::vector`的核心机制,这有助于深入学习C++的容器类和模板元编程。