`
wangxiaohigh
  • 浏览: 1464724 次
文章分类
社区版块
存档分类
最新评论

浅谈Java集合中Array(数组)的应用

 
阅读更多
我们都知道,由于Array(数组)通常意义上讲只是一个单纯的线性序列,又基于Native,凭此它的效率历来便号称Java中最高。所以通常我们也都承认Java中效率最高的存储方式就是使用数组。但是,由于数组初始化后大小固定,索引不能超出下标,缺少灵活的扩展功能等原因,使得很多人放弃了数组的使用, 转而使用Collection,List,Map,Set等接口处理集合操作。

诚然在Java中使用集合类可以极大的简化我们的代码编写量。但是,有时明明只是缓存一些线性数据,还偏偏有人要使用HashMap保存,系统为此付出了不必要的内存损耗。如果是通常的程序还没有什么,当应用在高并发或者加载高内存消耗对象时(如Image图像)无意义的频繁使用集合类将极易引发OutOfMemoryException。

我们很清楚,以List接口实现的集合类中,ArrayList内部运算是基于Array的,所以他继承了Array的优势,非常适合索引取值和存储线性数据(Vector虽然也是基于Array的,但毁在大量的synchronized上……所以很多情况下等于废了……)。但它不适合插入数据和删除数据,因为每插入或删除一次就会产生一次大量数组内容Copy的操作。而LinkedList正好与ArrayList相反,它比较适合与插入删除操作,不适合于索引取值,因为它不可以像数组一样根据索引值直接就可以定位元素的地址,而需要从头至尾一个一个的来数位置。

其实这也是当然的,但凡接触过数据结构的都会知道,任何存储结构都是有其局限性的,没有也不可能有所有方面都完美的存储方式。我们所能做的,只是尽可能使用特定范围内最有效的解决方案,而对此什么解决方法最有效呢?

一般情况下,考虑到效率与类型检查等问题,应该尽可能使用数组,所以我个人比较推荐的方式就是根据需要基于数组定制集合类。

说到这里可能很多人以开发周期及稳定、通用性为借口而直接使用JDK或第三方集合类(或者COPY代码|||)。其实就我个人认为,这有些因噎废食了,确实有时存储对象比较复杂,自己的集合类性能无法保障。但在大多数项目中,这种情况并不存在,我们完全有能力根据需求构造集合以避免不必要的资源占用及进行相应优化。而某些人往往只是见Hibernate等框架返回个List便有样学样的自己也List和Map到底,干脆忘了Arrays.asList等方法是为什么而存在的。

本来JDK提供给我们Arrays.asList方法和Collection.toArray方法就是为了集合类和数组优势互补之用,以此成为数组和Collection等集合类间的桥梁,只要有这两种方法Array和Collection就可以相互转换。那么,我们有什么理由枉费可以利用的资源而不用呢?

事实上,我们只要基本掌握Arrays类和Reflect这两个有力的武器,操作数组处理持久对象根本就是张飞吃豆芽——小菜一碟。

下面,我抛砖引玉的写些代码举例:

TestBean.java(测试用实体类)

packageorg.loon.framework.db.test.util;

/***//**
*<p>
*Title:LoonFramework
*</p>
*<p>
*Description:
*</p>
*<p>
*Copyright:Copyright(c)2007
*</p>
*<p>
*Company:LoonFramework
*</p>
*
*
@authorchenpeng
*@email:ceponline@yahoo.com.cn
*
@version0.1
*/

publicclassTestBean...{

Stringname;

intid;

publicintgetId()...{
returnid;
}


publicvoidsetId(intid)...{
this.id=id;
}


publicStringgetName()...{
returnname;
}


publicvoidsetName(Stringname)...{
this.name=name;
}


}



ArrayUtil.java(用于Array的增、删、改、查等操作)
packageorg.loon.framework.db.test.util;

importjava.io.Serializable;
importjava.lang.reflect.Array;
importjava.util.Arrays;
importjava.util.Collection;
importjava.util.List;
importjava.util.Random;


/***//**
*<p>
*Title:LoonFramework
*</p>
*<p>
*Description:ArrayUtil,数组操作工具类
*</p>
*<p>
*Copyright:Copyright(c)2007
*</p>
*<p>
*Company:LoonFramework
*</p>
*
*
@authorchenpeng
*@email:ceponline@yahoo.com.cn
*
@version0.1
*/

publicclassArrayUtilimplementsSerializable...{
/***//**
*
*/

privatestaticfinallongserialVersionUID=8057374625909011982L;

//缓存数组对象
privateObjectobjArray;

//数组长度
privateintsize=0;

//缓存数组对象实体类型
privateStringobjType;

finalstaticprivateRandomrand=newRandom();

privatestaticArrayUtilinstance=null;

/***//**
*直接注入Collection
*
*
@paramcollection
*
@return
*/

publicstaticArrayUtilgetInstance(Collectioncollection)...{
returngetInstance(collection.toArray());
}

/***//**
*直接注入对象数组
*
*
@paramarray
*/

publicstaticArrayUtilgetInstance(Objectarray)...{
if(instance==null)...{
instance
=newArrayUtil(array);
}

returninstance;
}

/***//**
*注入类产生指定大小对象数组
*
*
@paramclazz
*
@parammaxSize
*/

publicstaticArrayUtilgetInstance(Classclazz,intmaxSize)...{
if(instance==null)...{
instance
=newArrayUtil(clazz,maxSize);
}

returninstance;
}


privateArrayUtil()...{

}


/***//**
*注入对象数组产生指定大小对象数组
*
*
@paramclazz
*
@parammaxSize
*/

privateArrayUtil(Classclazz,intmaxSize)...{
//转为指定大小对象数组
Objectarray=(Object[])Array.newInstance(clazz,
maxSize);
//初始化
init(array);
}


/***//**
*直接注入对象数组
*
*
@paramarray
*/

privateArrayUtil(Objectarray)...{
init(array);
}


privatevoidinit(Objectarray)...{
//检查是否数组对象
if(!(arrayinstanceofObject[]))...{
thrownewIndexOutOfBoundsException("Notobjectarrays!");
}

//缓存数组对象
objArray=array;
//缓存实体类型
objType=array.getClass().getComponentType().getSimpleName();
//缓存数组长度
size=Array.getLength(objArray);
}


/***//**
*返回指定对象索引位置
*
*
@paramobj
*
@return
*/

publicintget(Objectobj)...{
//检查是否合法对象
checkObject(obj);
Object[]object
=(Object[])objArray;
for(inti=0;i<size;i++)
if(object[i]==obj)...{
returni;
}

return-1;
}


/***//**
*返回指定索引位置对象
*
*
@paramindex
*
@return
*/

publicObjectget(intindex)...{
checkIndex(index);
returngetObjectArray()[index];
}


/***//**
*加载对象在指定位置
*
*
@paramobj
*
@paramindex
*/

publicvoidadd(Objectobj,intindex)...{
//检查索引是否越界
checkIndex(index);
//检查是否合法对象
checkObject(obj);
Object[]objTemp
=(Object[])objArray;
objTemp[index]
=obj;
//copy临时数组到objArray
System.arraycopy(objTemp,0,objArray,0,objTemp.length);
}


/***//**
*加载对象
*
*
@paramobj
*/

publicvoidadd(Objectobj)...{
//类型检查
checkObject(obj);
//累加
next();
//临时缓存旧数组数组
Object[]objTemp=newObject[size];
//加载对象
objTemp[size-1]=obj;
//copy
System.arraycopy(objArray,0,objTemp,0,Array.getLength(objArray));
//转换
objArray=objTemp;
}


/***//**
*删除指定索引位置数组数据
*
*
@paramindex
*
@return
*/

publicObjectremove(intindex)...{
//检查索引是否越界
checkIndex(index);
Object[]objTemp
=(Object[])objArray;

//重新构建objArray
intj;
if((j=size-index-1)>0)...{
System.arraycopy(objTemp,index
+1,objTemp,index,j);
}

//减少size
back();

returnobjTemp[index];
}


publicbooleancontains(Objectobj)...{
Object[]objTemp
=(Object[])objArray;
for(inti=0;i<size;i++)...{
if(hash(objTemp[i])==hash(obj))...{
returntrue;
}

}

returnfalse;
}


publicObject[]sub(intstartIndex,intendIndex)...{
//验证索引范围
checkIndex(startIndex);
checkIndex(endIndex);
intover=endIndex-startIndex;
if(over<0)...{
thrownewIndexOutOfBoundsException("Indexbeyondtheendoftheborder!");
}

Object[]objTemp
=(Object[])objArray;
Object[]objs
=(Object[])Array.newInstance(objArray.getClass().getComponentType(),
over);
for(inti=startIndex;i<endIndex;i++)...{
objs[i
-1]=objTemp[i-1];
}

returnobjs;
}


publicvoidclear()...{
Object[]objTemp
=(Object[])objArray;
//清空数据
for(inti=0;i<size;i++)...{
objTemp[i]
=null;
size
=0;
}

}


/***//**
*删除指定的对象实体
*
*
@paramobj
*
@return
*/

publicbooleanremove(Objectobj)...{
//检查是否合法对象
checkObject(obj);
Object[]object
=(Object[])objArray;
for(inti=0;i<size;i++)
if(object[i]==obj)...{
remove(i);
returntrue;
}

returnfalse;
}


/***//**
*混淆数组元素
*
*
@return
*/

publicvoidmixElements()...{
mixElements(objArray);
}


/***//**
*检查数组内元素是否为空
*
*
@return
*/

publicbooleanisEmpty()...{
return(size==0);
}


/***//**
*转为list
*
*
@return
*/

publicListgetList()...{
returnArrays.asList((Object[])objArray);
}


/***//**
*减少size
*
*/

privatevoidback()...{
size
--;
}


/***//**
*增加size
*
*/

privatevoidnext()...{
size
++;
}


/***//**
*检查索引是否溢出
*
*
@paramindex
*/

privatevoidcheckIndex(intindex)...{

if(index>=size||index<0)...{
thrownewIndexOutOfBoundsException("Index"+index
+"outofbounds!");
}

}


/***//**
*检查对象类型
*
*
@paramobj
*/

privatevoidcheckObject(Objectobj)...{
if(objinstanceofObject[])...{
thrownewIndexOutOfBoundsException("Notloadingarrays!");
}

Stringtype;
if(!objType.equals(type=obj.getClass().getSimpleName()))...{
thrownewIndexOutOfBoundsException("Notthis"+type
+"typeofloading!");
}

}


/***//**
*扩充数组对象
*
*
@paramobj
*
@parami
*
@paramflag
*
@return
*/

staticpublicObjectexpand(Objectobj,inti,booleanflag)...{
intj=Array.getLength(obj);
Objectobj1
=Array.newInstance(obj.getClass().getComponentType(),j
+i);
System.arraycopy(obj,
0,obj1,flag?0:i,j);
returnobj1;
}


/***//**
*扩充数组对象
*
*
@paramobj
*
@parami
*
@paramflag
*
@return
*/

staticpublicObjectexpand(Objectobj,inti)...{
returnexpand(obj,i,true);
}


/***//**
*随机返回数组内容
*
*
@paramobj
*/

staticpublicvoidmixElements(Objectobj)...{
inti=Array.getLength(obj);
for(intk=0;k<i;k++)...{
intj=getRandom(k,i-1);
Objectobj1
=Array.get(obj,j);
Array.set(obj,j,Array.get(obj,k));
Array.set(obj,k,obj1);
}


}


staticpublicRandomgetRandomObject()...{
returnrand;
}


staticpublicintgetRandom(inti,intj)...{
returni+rand.nextInt((j-i)+1);
}


privateinthash(Objectobj)...{
inth=obj.hashCode();
h
+=~(h<<9);
h
^=(h>>>14);
h
+=(h<<4);
h
^=(h>>>10);
returnh;
}


publicinthashCode()...{
returnhash(objArray.getClass());
}


publicintsize()...{
returnsize;
}


/***//**
*反回当前数组对象
*
*
@return
*/

publicObject[]getObjectArray()...{
return(Object[])objArray;
}


publicstaticvoidmain(String[]args)...{

/**//*TestBean[]tb=newTestBean[3];
for(inti=0;i<tb.length;i++){
tb[i]=newTestBean();
tb[i].setName("name"+i);
tb[i].setId(i);
}
//直接载入已有数组对象
ArrayUtilarrayUtil=ArrayUtil.getInstance(tb);
TestBeantb1=newTestBean();
//arrayUtil.add(tb[0]);
arrayUtil.remove(tb[0]);
//arrayUtil.remove(tb[2]);
System.out.println(arrayUtil.contains(tb1));
System.out.println(((TestBean)arrayUtil.get(0)).getName());
System.out.println(arrayUtil.size());
//打乱数组
arrayUtil.mixElements();
for(inti=0;i<arrayUtil.size();i++){
System.out.println(((TestBean)arrayUtil.get(i)).getName());
}
*/

//生成TestBean的数组实例,初始容量为5
ArrayUtilarrayUtil=ArrayUtil.getInstance(TestBean.class,5);
TestBeant
=newTestBean();
t.setName(
"test");
//在数组载入t的实例
arrayUtil.add(t,0);
TestBeant1
=newTestBean();
t1.setName(
"test1");
arrayUtil.add(t1,
1);
arrayUtil.add(t,
2);
arrayUtil.add(t,
3);
arrayUtil.add(t,
4);
//会自动增加数组容量
arrayUtil.add(t);
//显示索引5数据
System.out.println(((TestBean)arrayUtil.get(5)).getName());
//截取索引1-3,显示1数据
System.out.println(((TestBean)arrayUtil.sub(1,3)[1]).getName());

}


}



通过例子我们可以发现,只是简单的几行代码,就令Array发挥出我们需要借助他人集合类才能完成的功能。而这些,却本来是我们也可以轻易做到的。

所以本人在此呼吁,坚持尽量使用数组,从我做起,从每个项目做起~~~
分享到:
评论

相关推荐

    数组和集合

    下面将详细介绍Java中数组和集合的使用、特点和常见知识点。 首先,数组在Java中的特性与其他语言有所不同。Java中的数组是一个类,这使得它在Java语言中具有特别的地位。数组可以用来存储基本数据类型(如int, ...

    Java array 数组用法源码集.rar

    Java array 数组用法源码集,比如测试数组、划分成绩等级、从方法中返回数组、二分查找法、求二维数组的长度、锯齿数组、处理二维数组等,相信通过这些基础的数组操作实例,会对你了解Java中的数组有很好的帮助作用...

    Unity3D教程:Array数组类的使用(二)1

    在Unity3D中,数组是一种基础数据结构,用于存储同类型的数据集合。本教程主要关注Array数组类在JavaScript中的使用。 Array 类是Unity3D中JavaScript的特定实现,它提供了丰富的函数和属性来操作数组。数组的长度...

    C#中的数组 Array数组

    学习 C# 中的 数组 Array数组

    TIA博途中数组Array【】的基本使用方法详解.docx

    ### TIA博途中数组Array【】的基本使用方法详解 #### 一、数组Array【】概述 在TIA博途(Totally Integrated Automation Portal)中,数组(Array)是一种重要的数据结构,它由固定数量且相同数据类型的元素组成。...

    java 反射中操作数组

    总结来说,Java反射提供了对数组的动态操作能力,包括创建数组、设置和获取数组元素、以及处理类中声明的数组字段。这种能力使得代码更加灵活,但也增加了潜在的错误风险,因此在使用时应谨慎并充分测试。

    java中传递数组参数

    ### Java中传递数组参数 #### 一、传递数组到方法 在Java中,传递数组到方法的方式相对简单直观。可以通过以下几种方式实现: 1. **直接传递数组:** 在Java中,当我们想要将一个数组作为参数传递给某个方法时,...

    java数组与方法数组的定义及使用

    数组在Java中定义时需要指定类型,因为所有数组元素必须是同一类型。例如,如果我们要定义一个整型数组,可以写成: ```java int[] numbers; ``` 这声明了一个名为`numbers`的整型数组,但并未创建实际的数组对象。 ...

    java 数组的合并

    本篇文章将详细探讨Java中如何实现数组的合并。 首先,我们要了解Java中的数组。数组是由相同类型的数据元素构成的有序集合,可以通过索引来访问每个元素。例如,我们可以声明一个整型数组: ```java int[] ...

    java数组应用

    多维数组在Java中被视为数组的数组。最常见的形式是二维数组,它实质上是一维数组,其中的每个元素又是一个一维数组。 ##### 2.1 二维数组的定义 二维数组的定义如下: ```java type arrayName[][]; ``` 例如: ``...

    java中两个byte数组实现合并的示例

    本文将深入探讨如何在Java中实现两个`byte`数组的合并,以及一些相关的操作,如数组反转和加密。 首先,我们来看一下标题和描述中提到的示例代码。这个简单的函数`addBytes`接收两个`byte`类型的数组`data1`和`data...

    java数组基础与典型应用

    数组是Java中最基本的数据结构之一,用于存储固定大小的同类型元素集合。它允许程序员在单个变量中存储多个相同类型的数据项,这极大地提高了数据处理的效率和程序的可读性。 在Java中,数组可以通过以下方式声明:...

    Java中数组的各种应用

    本文将深入探讨Java中数组的各种应用,旨在帮助初学者理解并掌握数组的使用。 一、数组定义与初始化 在Java中,数组可以看作是同类型元素的有序集合。例如,你可以创建一个整型数组来存储一系列的整数。数组的定义...

    Array数组工具包

    Array数组工具包 1 1 1 1

    5-Java数组与方法

    首先,数组是Java中的基本类型之一,它是由相同类型的数据项组成的有序集合。每个数据项都有一个唯一的索引,从0开始,到数组长度减1。例如,一个整型数组int[] numbers可以存储一系列的整数,如numbers[0], numbers...

    C_Array Array数组

    数组是C#编程中基本的数据结构,用于存储同一类型的数据集合。数组的概念基于其元素的类型和固定的容量,它允许程序员以统一的方式管理和操作一组数据。数组的声明通常包括元素类型和数组变量名,如`string[] Mystr;...

    hibernate array 数组映射

    在Java的持久化框架Hibernate中,数组映射是一种常见的数据模型转换方式,它允许我们将数据库中的数据以数组的形式存储在Java对象中。本篇将详细探讨`hibernate array 数组映射`的相关知识点,包括其原理、配置、...

    Java基础教程:数组和集合

    Java 基础教程:数组和集合 Java 中的数组和集合是两个非常重要的概念,它们用于存储和操作数据...在实际编程中,根据具体需求选择合适的数据结构是非常重要的,希望本文对你理解和应用 Java 中的数组和集合有所帮助。

    C、C#、C++、Java 中定义以为数组的方式

    Java中的数组定义方式与C#相似,但也存在一些区别。Java支持两种主要的数组定义方式: #### 一维数组 ```java int[] intArray = new int[5]; // 定义一个整型数组intArray,有5个元素 int[] intArray = {1, 2, 3};...

    java-对象数组转换为json

    下面将深入探讨如何在Java中实现对象数组向JSON的转换,并反过来将JSON字符串转换为Java对象数组,同时也会提及一些相关技术和代码示例。 ### Java对象数组转JSON 首先,我们来看如何将一个Java对象数组转换成JSON...

Global site tag (gtag.js) - Google Analytics