[转载]Java数组扩容算法及Java对它的应用
原文链接:http://www.cnblogs.com/gw811/archive/2012/10/07/2714252.html
Java数组扩容的原理
1)Java数组对象的大小是固定不变的,数组对象是不可扩容的。
2)利用数组复制方法可以变通的实现数组扩容。
3)System.arraycopy()可以复制数组。
4)Arrays.copyOf()可以简便的创建数组副本。
5)创建数组副本的同时将数组长度增加就变通的实现了数组的扩容。
源码展示:

1 public class Arrays {
2 /**
3 * @param original: the array to be copied
4 * @param newLength: the length of the copy to be returned
5 * @return a copy of the original array, truncated or padded with zeros
6 * to obtain the specified length
7 */
8 public static int[] copyOf(int[] original, int newLength) {
9 int[] copy = new int[newLength];
10 System.arraycopy(original, 0, copy, 0,
11 Math.min(original.length, newLength));
12 return copy;
13 }
14 /**
15 * @param original the array from which a range is to be copied
16 * @param from the initial index of the range to be copied, inclusive
17 * @param to the final index of the range to be copied, exclusive.
18 * (This index may lie outside the array.)
19 * @return a new array containing the specified range from the original array,
20 * truncated or padded with zeros to obtain the required length
21 */
22 public static int[] copyOfRange(int[] original, int from, int to) {
23 int newLength = to - from;
24 if (newLength < 0)
25 throw new IllegalArgumentException(from + " > " + to);
26 int[] copy = new int[newLength];
27 System.arraycopy(original, from, copy, 0,
28 Math.min(original.length - from, newLength));
29 return copy;
30 }
31 }

示例说明:

1 import java.util.Arrays;
2
3 /** 数组变长算法!
4 * 数组对象长度不可改变
5 * 但是很多实际应用需要长度可变的数组
6 * 可以采用复制为容量更大的新数组, 替换原数组, 实现变长操作
7 * */
8 public class ArrayExpand {
9 public static void main(String[] args) {
10 //数组变长(扩容)算法!
11 int[] ary={1,2,3};
12 ary=Arrays.copyOf(ary, ary.length+1);
13 ary[ary.length-1]=4;
14 System.out.println(Arrays.toString(ary));//[1, 2, 3, 4]
15 //字符串连接原理
16 char[] chs = { '中', '国' };
17 chs = Arrays.copyOf(chs, chs.length + 1);
18 chs[chs.length - 1] = '北';
19 chs = Arrays.copyOf(chs, chs.length + 1);
20 chs[chs.length - 1] = '京';
21 //字符数组按照字符串打印
22 System.out.println(chs);//中国北京
23 //其他数组按照对象打印
24 System.out.println(ary);//[I@4f1d0d
25 }
26 }

实现案例:
案例1 : 统计一个字符在字符串中的所有位置.
字符串: 统计一个字符在字符串中的所有位置
字符: '字'
返回: {4,7}

1 public class CountCharDemo {
2 public static void main(String[] args) {
3 char key = '字';
4 String str = "统计一个字符在字符串中的所有位置";
5 int[] count=count(str,key);
6 System.out.println(Arrays.toString(count));//[4, 7]
7 }
8 public static int[] count(String str,char key){
9 int[] count={};
10 for(int i=0;i<str.length();i++){
11 char c=str.charAt(i);
12 if(c==key){
13 //扩展数组
14 count=Arrays.copyOf(count, count.length+1);
15 //添加序号i
16 count[count.length-1]=i;
17 }
18 }
19 return count;
20 }
21 }

char[]、String、StringBuilder
char[]:字符序列, 只有字符数据, 没有操作, 如果算法优秀, 性能最好。
String: char[] + 方法(操作, API功能)
StringBuilder: char[] + 方法(操作char[] 的内容)
String:内部包含内容不可变的char[],表现为String对象不可变。String包含操作(API方法),是对char[]操作,但不改变原对象经常返回新的对象,很多String API提供了复杂的性能优化算法,如:静态字符串池。
StringBuilder:内部也是一个char[],但是这个数组内容是可变的,并且自动维护扩容算法,因为数据内容可变,所以叫:可变字符串。StringBuilder API方法,是动态维护char[]内容,都可以改变char[]内容。

1 public abstract class AbstractStringBuilder {
2 /** The value is used for character storage.*/
3 char value[];
4 /** The count is the number of characters used.*/
5 int count;
6 /** Returns the length (character count).*/
7 public int length() {
8 return count;
9 }
10
11 public AbstractStringBuilder append(String str) {
12 if (str == null)
13 str = "null";
14 int len = str.length();
15 if (len == 0)
16 return this;
17 int newCount = count + len;
18 if (newCount > value.length)
19 expandCapacity(newCount);
20 str.getChars(0, len, value, count);
21 count = newCount;
22 return this;
23 }
24
25 /**
26 * 自动实现Java数组扩容
27 */
28 void expandCapacity(int minimumCapacity) {
29 int newCapacity = (value.length + 1) * 2;
30 if (newCapacity < 0) {
31 newCapacity = Integer.MAX_VALUE;
32 } else if (minimumCapacity > newCapacity) {
33 newCapacity = minimumCapacity;
34 }
35 value = Arrays.copyOf(value, newCapacity);
36 }
37 }

字符串数组与String类的原理

1 /** 字符串数组与String类的原理 */
2 public class CharArrayDemo {
3 public static void main(String[] args) {
4 /* Java 可以将char[]作为字符串处理 */
5 char[] ch1={'中','国','北','京'};
6 char[] ch2={'欢','迎','您'};
7 System.out.println(ch1);//中国北京
8 System.out.println(ch2);//欢迎您
9 /* char[]运算需要编程处理,如连接: */
10 char[] ch3=Arrays.copyOf(ch1, ch1.length+ch2.length);
11 System.arraycopy(ch2, 0, ch3, ch1.length, ch2.length);
12 System.out.println(ch3);//中国北京欢迎您
13 /* String API提供了简洁的连接运算: */
14 String str1="中国北京";
15 String str2="欢迎您";
16 String str3=str1.concat(str2);
17 System.out.println(str3);//中国北京欢迎您
18 /* 字符串转大写: */
19 char[] ch4={'A','a','c','f'};
20 char[] ch5=Arrays.copyOf(ch4, ch4.length);
21 for(int i=0;i<ch5.length;i++){
22 char c=ch5[i];
23 if(c>='a' && c<='z'){
24 ch5[i]=(char)(c+('A'-'a'));
25 }
26 }
27 System.out.println(ch5);//AACF, 原数组ch4不变
28 String str4="Aacf";
29 String str5=str4.toUpperCase();//原字符串str4保持不变
30 System.out.println(str5);//AACF
31 }
32 }

[转载]Java数组扩容算法及Java对它的应用的更多相关文章
- Java数组扩容算法及Java对它的应用
1)Java数组对象的大小是固定不变的,数组对象是不可扩容的.利用数组复制方法可以变通的实现数组扩容.System.arraycopy()可以复制数组.Arrays.copyOf()可以简便的创建数组 ...
- 使用泛型对java数组扩容
编写一个通用方法,其功能是将数组扩展到10%+10个元素(转载请注明出处) package cn.reflection; import java.lang.reflect.Array; public ...
- java数组扩容
有些时候使用数组代替栈,玩意数组容量不够需要扩容 则: 1.Array.toString();直接遍历打印数组 2.数组扩容采用Array.copyOf(),直接实现数组扩容功能,非常强大 (实际 ...
- 13-02 Java 数组高级算法,Arrays类
冒泡排序 冒泡排序原理 冒泡排序代码: package cn.itcast_01; /* * 数组排序之冒泡排序: * 相邻元素两两比较,大的往后放,第一次完毕,最大值出现在了最大索引处 */ pub ...
- Java 数组扩容
在添加数据到达数组的上限的时候数组进行扩容: public void resizeArrayCaptcity(){ if(size>=arr.length){ Emp [] arr2=new ...
- JAVA 数组算法(复制、查找、插入)
一.复制数组算法 //数组复制算法 public class Test{ public static void main(String[] args){ int[] arrA = {100,800,5 ...
- Java 数组基础,java.util.Arrays
定义数组 方式1(推荐,更能表明数组类型) 方式2(同C语言) 方式3定义时直接初始化 数组运用基础 数组长度 equals() 数组元素不为基本数据类型时 二维数组 二维数组基础 变长的二维数组 j ...
- 数据结构与算法【Java】03---栈
前言 数据 data 结构(structure)是一门 研究组织数据方式的学科,有了编程语言也就有了数据结构.学好数据结构才可以编写出更加漂亮,更加有效率的代码. 要学习好数据结构就要多多考虑如何将生 ...
- java数组与内存控制
1.1数组的初始化 数组是大多数编程语言都提供一种的复合结构,如果程序需要多个类型相同的变量时,就可以考虑定义一个数组.java语言的数组变量时引用类型的变量,因此具有java独有的特性. java数 ...
随机推荐
- 第32课 Qt中的文件操作
1. Qt的中IO操作 (1)Qt中IO操作的处理方式 ①Qt通过统一的接口简化了文件和外部设备的操作方式 ②Qt中的文件被看作一种特殊的外部设备 ③Qt中的文件操作与外部设备的操作相同 (2)IO操 ...
- (原创)JAVA多线程二线程池
一,线程池的介绍 线程池包括一下三种: 线程池名称 创建方法 特点 其他 固定大小线程池 ExecutorService threadpool = Executors.newFixedThreadPo ...
- java多线程系类:JUC线程池:03之线程池原理(二)(转)
概要 在前面一章"Java多线程系列--"JUC线程池"02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包 ...
- LeetCode "448. Find All Numbers Disappeared in an Array"
My first reaction is to have an unlimited length of bit-array, to mark existence. But if no extra me ...
- cf Round 603
A.Alternative Thinking(思维) 给出一个01串,你可以取反其中一个连续子串,问取反后的01子串的最长非连续010101串的长度是多少. 我们随便翻一个连续子串,显然翻完之后,对于 ...
- Dell xps 13 9350待机时总是关机的处理方法
现象: 年初买的dell xps13 9350,最近可能是由于win10做了一些更新,每次睡眠就自动关机了,重启很多次,修改电源选项都没用, 原因分析: 在网上搜了一下,有人发现xps15 9350也 ...
- 小巧数据库 Apache Derby 使用攻略
1. Derby 介绍 将目光放在小 Derby 的原因是纯绿色.轻巧.内存占用小,分分钟在你机子跑起来,自己做点需要连接数据库的代码实践非常方便. 虽然 Mysql 也可以,多一种选择,不是也挺好么 ...
- Sort using in VS
- mysql5.7.10 的源码安装
mysql 5.7.10的源码安装:http://fyduan.blog.51cto.com/4234935/1729873cmake . -DCMAKE_INSTALL_PREFIX=/usr/lo ...
- Linux 命令积累
1, su root 切换到root用户 su user 切换到普通用户 2, mkdir / touch 创建文件夹 /文件 3, vi 打开编辑文件 按insert进入编辑模式 编辑完成后 按es ...