JavaSE--数组以及Arrays工具类
一、数组
1、数组概述
数组是一种引用数据类型,数组的父类是Object
数组因为是引用类型,所以在堆内存中存放
2、数组的分类:
一维数组、二维数组、三维数组、多维数组...(一维数组较常用)
3、数组的长度
a)数组一旦创建,java中规定,长度不可变
b)获取数组长度:所有数组对象都有length属性,用来获取数组中的元素的个数
c)数组中每个元素都是有下标的,下标都从0开始,以1递增。最有一个元素的下标是 length - 1
4、数组中存放的数据
a)数组实际上是一个容器,可以同时容纳多个元素
b)数组当中可以存储“基本数据类型”的数据,也可以存储“引用数据类型”数据。数组当中如果存储的是java对象的话,实际上存储的是对象的引用
c)Java中的数组要求数组中元素的类型统一。例如:int类型数组只能存放int类型,Student类型数组只能存储Student对象
5、数组的内存地址
a)数组在内存方面存储的时候,数组中的元素的内存地址是连续的,(存储的每一个元素都有规则的挨着排列)
b)数组中首元素(第一个元素)的内存地址作为整个数组的内存地址
6、数组这种数据结构的优点:
查询/查找/检索某个下标上的元素时效率极高。可以说是查询效率最高的一个数据结构。
- 每一个元素的内存地址在空间存储上是连续的。
- 每一个元素类型相同,所以占用空间大小一样。
- 知道第一个元素内存地址,知道每一个元素占用空间的大小,又知道下标,所以通过一个数学表达式就可以计算出某个下标上元素的内存地址。直接通过内存地址定位元素,所以数组的检索效率是最高的
7、数组这种数据结构的缺点:
- 由于为了保证数组中每个元素的内存地址连续,所以在数组上随机删除或者增加元素的时候,效率较低,因为随机增删元素会涉及到后面元素统一向前或者向后位移的操作。
- 数组不能存储大数据量,因为很难在内存空间上找到一块特别大的连续的内存空间。
二、一维数组
1、一维数组语法格式
int[] array1;
double[] array2;
boolean[] array3;
String[] array4;
Object[] array5;
2、一维数组的初始化
// 两种方式:静态初始化、动态初始化
// 静态初始化
int[] array = {100,200,500};
// 动态初始化
// 初始化一个5个长度的int类型数组,每个元素默认值0
int[] array = new int[5]; // 这里的5表示数组的元素个数。
// 初始化6个长度的String类型数组,每个元素默认值null
String[] names = new String[6];
3、访问一维数组中的元素以及遍历
public class ArrayTest01 {
public static void main(String[] args) {
// 声明一个int类型的数组,使用静态初始化的方式
int[] a = {1, 100, 10, 20, 55, 689};
// 所有的数组对象都有length属性
System.out.println("数组中元素的个数" + a.length);
// 数组中每一个元素都有下标
// 通过下标对数组中的元素进行存和取。
// 取(读)
System.out.println("第一个元素 = " + a[0]);
System.out.println("最后一个元素 = " + a[5]);
System.out.println("最后一个元素 = " + a[a.length - 1]);
// 存(改)
// 把第一个元素修改为111
a[0] = 111;
// 把最后一个元素修改为0
a[a.length - 1] = 0;
System.out.println("第一个元素 = " + a[0]);
System.out.println("最后一个元素 = " + a[5]);
// 一维数组遍历
for(int i = 0; i < a.length; i++){
System.out.println(a[i]); // i是从0到5,是下标
}
// 下标越界了 异常
//System.out.println(a[6]); //ArrayIndexOutOfBoundsException
// 从最后一个元素遍历到第1个元素
for (int i = a.length - 1; i >= 0; i--) {
System.out.println("逆序输出-->" + a[i]);
}
}
}
4、动态初始化方式创建数组
public class ArrayTest02 {
public static void main(String[] args) {
// 声明/定义一个数组,采用动态初始化的方式创建
int[] a = new int[4]; // 创建长度为4的int数组,数组中每个元素的默认值是0
// 遍历数组
for (int i = 0; i < a.length; i++) {
System.out.println(i + "的元素:"+ a[i]);
}
// 后期赋值
a[0] = 1;
a[1] = 100;
a[2] = 111;
a[3] = 222; // 注意下标别越界
for (int i = 0; i < a.length; i++) {
System.out.println("数组中下标为" + i + "的元素是:" + a[i]);
}
// 初始化一个Object类型的数组,采用动态初始化方式
Object[] objs = new Object[3]; // 3个长度,动态初始化,所以每个元素默认值是null
for (int i = 0; i < objs.length; i++) {
System.out.println(objs[i]);
}
}
}
5、静态初始化与动态初始化什么时候使用
- 当你创建数组的时候,确定数组中存储哪些具体的元素时,采用静态初始化方式
- 当你创建数组的时候,不确定将来数组中存储哪些数据,你可以采用动态初始化的方式,预先分配内存空间
6、方法参数中直接传递一个静态数组
public class ArrayTest05{
public static void main(String[] args) {
// 静态初始化一维数组
int[] a = {1,2,3};
printArray(a);
// 传递一个静态初始化一维数组,语法必须这样
printArray(new int[]{1,2,3});
}
public static void printArray(int[] array){
for (int i = 0; i < array.length; i++) {
System.out.println(array[i]);
}
}
}
7、main方法中“String[] args”
JVM调用main方法的时候,会自动传一个String数组过来,例如dos命令中这样运行程序:java ArrayTest05 abc def xyz,那么这个时候JVM会自动将“abc def xyz”通过空格的方式进行分离,分离完成之后,自动放到“String[] args”数组当中,所以main方法上面的String[] args数组主要是用来接收用户输入参数的
idea中在run选项卡中,可以编辑参数
// "abc".equal不容易出现空指针异常,哪怕name和pass为空
// name.equal("abc")如果name为null,则会出现空指针异常,不太好
if("abc".equal(name) && "123".equal(pass)){
}
8、数组中存放引用数据类型
public class ArrayTest07 {
public static void main(String[] args) {
Animal[] animals = {new Animal(), new Animal()};
for (int i = 0; i < animals.length; i++) {
animals[i].move();
}
// 创建子类
Animal[] anis = {new Cat(), new Bird()};
for (int i = 0; i < anis.length; i++){
// 调用子对象特有方法,需要向下转型
if(anis[i] instanceof Cat){
Cat cat = (Cat)anis[i];
cat.catchMouse();
}else if(anis[i] instanceof Bird){
Bird bird = (Bird)anis[i];
bird.sing();
}
}
}
}
class Animal{
public void move(){
System.out.println("Animal move...");
}
}
// Cat是子类
class Cat extends Animal {
public void move(){
System.out.println("猫在走猫步!");
}
// 特有方法
public void catchMouse(){
System.out.println("猫抓老鼠!");
}
}
// Bird子类
class Bird extends Animal {
public void move(){
System.out.println("Bird Fly!!!");
}
// 特有的方法
public void sing(){
System.out.println("鸟儿在歌唱!!!");
}
}
9、一维数组的扩容
先新建一个大容量数组,然后将小容量的数组中的元素拷贝到大容量中,使用System.arraycopy(5个参数);
注意:数组扩容效率较低,所以尽可能不要使用,使用数组提前预估数组长度
public class ArrayTest08 {
public static void main(String[] args) {
// java中数组拷贝的方法arraycopy
//System.arraycopy(5个参数);
// 拷贝源(从这个数组中拷贝)
int[] src = {1, 11, 22, 3, 4};
// 拷贝目标(拷贝到这个目标数组上)
int[] dest = new int[20]; // 动态初始化一个长度为20的数组,每一个元素默认值0
// 调用JDK System类中的arraycopy方法,来完成数组的拷贝
//System.arraycopy(src, 1, dest, 3, 2);
// 拷贝全部元素
System.arraycopy(src, 0, dest, 0, src.length);
for (int i = 0; i < dest.length; i++) {
System.out.println(dest[i]);
}
// 拷贝引用数据被雷
Object[] objs = {new Object(), new Object(), new Object()};
Object[] newObjs = new Object[5];
// 这里拷贝的时候是拷贝对象的地址
System.arraycopy(objs, 0, newObjs, 0, objs.length);
for (int i = 0; i < newObjs.length; i++) {
System.out.println(newObjs[i]);
}
}
}
三、二维数组
1、关于二维数组
二维数组其实就是一个特殊的一维数组,特殊在在这个一维数组当中的每一个元素都是一个一维数组
2、二维数组静态初始化
public class ArrayTest10{
public static void main(String[] args){
int[][] a = {
{1,3},
{20,30,10},
{0}
};
System.out.println(a.length);// 3
System.out.println(a[0].length);// 2
System.out.println(a[1].length);// 3
System.out.println(a[2].length);// 1
}
}
3、二维数组元素的读和改
public class ArrayTest10 {
public static void main(String[] args) {
// 二维数组
int[][] a = {
{34,4,65},
{100,200,39,111},
{0}
};
// 读
System.out.println(a[0][0]);
System.out.println(a[1][2]);
System.out.println(a[2][0]);
// 改
a[2][0] = 11111;
System.out.println(a[2][0]);
}
}
4、二维数组的遍历
public class ArrayTest11 {
public static void main(String[] args) {
// 二维数组
String[][] array = {
{"mysql", "java", "c++", "python", "c#"},
{"张三", "李四", "王五"},
{"lucy", "jack", "rose"}
};
for(int i = 0; i < array.length; i++){
for(int j = 0; j < array[i].length; j++){
System.out.print(array[i][j] + " ");
}
System.out.println();// 换行
}
}
}
5、二维数组动态初始化
public class ArrayTest12 {
public static void main(String[] args) {
// 动态初始化
int[][] array = new int[3][4];
// 静态初始化
int[][] a = {{1,2,3,4},{4,5,6,76},{1,23,4}};
printArray(a);
// 传递一个静态初始化的二维数组
printArray(new int[][]{{1,2,3,4},{4,5,6,76},{1,23,4}});
}
public static void printArray(int[][] array){
// 遍历二维数组
for (int i = 0; i < array.length; i++) {
for (int j = 0; j < array[i].length; j++) {
System.out.print(array[i][j] + " ");
}
System.out.println();
}
}
}
三、数组工具类
SUN公司提供的数组工具类java.util.Arrays。其中方法都是静态的,直接用类名调用就行
1、给数组排序
int[] arr = {52,74,12,0,3,96};
Arrays.sort(arr);
for(int i = 0;i< arr.length,i++){
System.out.println(arr[i]);
}
2、二分法查找
int[] arr = {5,8,12,15,23,55};
int index = Arrays.binarySearch(arr,15);
System.out.println(index == -1 ? "该元素不存在" : "该元素的下标是" + index);