Java面向对象系列[v1.0.0][数组详解]

理解数组

  • Java的数组要求所有的数组元素具有相同的数据类型,换句话说一个数组里只能存储一种数据类型的数据
  • 一旦数组的的初始化完成,数组在内存中所占的空间将被固定下来,因此数组的长度将不可改变,即使把某个数组元素的数据清空,但它所占的空间依然保留,依然属于该数组,数组的长度依然不变
  • Java的数组既可以存储基本类型的数据,也可以存储引用类型的数据,只要所有的数据属于相同的数据类型即可
  • 数组也是一种数据类型,它本身是一种引用类型,因此用它定义一个变量时,仅仅表示定义了一个引用变量(也就是一个指针),这个引用变量还未指定任何内存

数组的定义和初始化

Java提供了两种定义数组的语法

type[] arrayName;
type arrayName[];

数组是个引用类型,因此用它定义一个变量时,仅仅表示定义了一个引用变量(也就是一个指针),这个时候引用变量还未指定任何内存,因此定义数组时不能指定数组的长度,并且没有指定内存就没有内存空间来存储数组元素,因此这个数组也就不能直接使用,只有对数组进行初始化后才可以使用。

数组初始化

所谓初始化就是给数组分配内存空间,并为数组元素赋初始值,一旦为数组的每个数组元素分配了内存,每个内存空间里存储的内容就是该数组元素的值,即使这个内存空间存储的内容是空,这个空也是一个值(null),只要为数组元素分配了内存空间,数组元素就具有了初始值,初始值的的获得有两种方式:一种是系统自动分配,另一种由程序员指定。

  • 静态初始化:初始化时由程序员显示指定每个数组元素的初始值,由系统决定数组长度
  • 动态初始化:初始化时程序员只指定数组长度,由系统为数组元素分配初始值

静态初始化

arrayName = new type[] {element1, element2, element3, element4...};
public class ArrayTest
{
    public static void main(String[] args)
    {
        // 定义一个int数组类型的变量,变量名为intArr.
        int[] intArr;
        // 使用静态初始化,初始化数组时只指定数组元素的初始值,不指定数组长度。
        intArr = new int[] {5, 6, 8, 20};
        // 定义一个Object数组类型的变量,变量名为objArr.
        Object[] objArr;
        // 使用静态初始化,初始化数组时数组元素的类型是
        // 定义数组时所指定的数组元素类型的子类
        objArr = new String[] {"Java", "李刚"};
        Object[] objArr2;
        // 使用静态初始化
        objArr2 = new Object[] {"Java", "李刚"};
    }
}

使用Jshell操练数组

C:\Users\Administrator>jshell
|  Welcome to JShell -- Version 11.0.6
|  For an introduction type: /help intro

jshell> Object[] objArr;
objArr ==> null

jshell> objArr = new String[] {"java", "davieyang"}
objArr ==> String[2] { "java", "davieyang" }

jshell> Object[] objArr2;
objArr2 ==> null

jshell> objArr2 = new Object[] {"java", "davieyang"}
objArr2 ==> Object[2] { "java", "davieyang" }

jshell> int[] intArr;
intArr ==> null

jshell> intArr = new int[] {1,2,3,4,5}
intArr ==> int[5] { 1, 2, 3, 4, 5 }

在初始化数组的时候,数据类型必须与定义数组变量的时候使用的一样,也可以是定义数组时指定的数据类型的子类,例如代码中的Object,定义了Object类型的数组,初始化为String类型,因为String是Object的子类

静态初始化简化写法

type[] arrayName = {element1, element2, element3, element4...};
int[] arrA = {5,6,7,8};

动态初始化

arrayName = new type[length];

指定数组长度而不明确数组元素的值,参数length即是数组长度,也就是该数组容纳数组元素的个数

//数组的定义和初始化同时完成,使用动态初始化语法
int[] prices = new int[5];
// 数组的定义和初始化同时完成,初始化数组时元素的类型是定义数组时元素类型的子类
Object[] books = new String[4];

执行动态初始化时,程序员只需要指定数组的长度,即为每个数组元素指定所需的内存空间,系统负责为这些数组元素分配初始值,系统分配初始值规则如下

* 整数类型(byteshortintlong)数组元素值为0
* 浮点类型(floatdouble)数组元素值为0.0
* 字符类型(char)数组元素值为‘\u0000’
* 布尔类型(boolean)数组元素值为false
* 引用类型(类、接口和数组)则数组元素值为null

不要同时使用静态初始化和动态初始化,也就是说不要同时指定数组长度,又为每个数组元素分配初始值

定义数组局部变量

在定义数组局部变量时,同样可以使用var来定义变量,但必须在定义变量时为它指定初始值,这样编译器就能推断出该变量的类型,但是使用静态初始化简化语法执行初始化的数组不能使用var定义数组变量。

// 编译器推断names变量的类型是String[]
var names = new String[] {"davieyang", "java"};
// 编译器推断weightArr变量的类型是double[]
var weightArr = new double[4];

使用数组元素

jshell> Object[] objarr
objarr ==> null

jshell> var names =
objarr

<press tab again to see all possible completions; total possible completions: 540>

jshell> var names = new String[]{"davieyang","java", "davieyang"}
names ==> String[3] { "davieyang", "java", "davieyang" }

jshell> System.out.println(names[1])
java

jshell> for(var i = 0; i<names.length; i++)
   ...> {
   ...> System.out.println(names[i]);
   ...> }
davieyang
java
davieyang
jshell> Object[] objarr
objarr ==> null

jshell> objarr = new Object[3]
objarr ==> Object[3] { null, null, null }

jshell> objarr[0]="davieyang"
$6 ==> "davieyang"

jshell> objarr[1]="java"
$7 ==> "java"
jshell> for(var i=0; i<objarr.length;i++)
   ...> {
   ...> System.out.println(objarr[i]);
   ...> }
davieyang
java
null

foreach循环

for(type variableName : array | collection)
{
    //variableName自动迭代访问每个元素...
}
public class ForEachTest
{
    public static void main(String[] args)
    {
        String[] books = {"轻量级Java EE企业应用实战",
            "疯狂Java讲义",
            "疯狂Android讲义"};
        // 使用foreach循环来遍历数组元素,
        // 其中book将会自动迭代每个数组元素
        for (String book : books)
        {
            System.out.println(book);
        }
        for (var book : books)
        {
            System.out.println(book);
        }
    }
}
public class ForEachTest
{
    public static void main(String[] args)
    {
        String[] books = {"轻量级Java EE企业应用实战",
            "疯狂Java讲义",
            "疯狂Android讲义"};
        // 使用foreach循环来遍历数组元素,
        // 其中book将会自动迭代每个数组元素
        for (var book : books)
        {
            System.out.println(book);
        }
    }
}

错误用法,不能对循环变量进行赋值,不但没有实际意义,而会导致循环出错

public class ForEachErrorTest
{
    public static void main(String[] args)
    {
        String[] books = {"轻量级Java EE企业应用实战",
            "疯狂Java讲义",
            "疯狂Android讲义"};
        // 使用foreach循环来遍历数组元素,其中book将会自动迭代每个数组元素
        for (var book : books)
        {
            book = "疯狂Ajax讲义";
            System.out.println(book);
        }
        System.out.println(books[0]);
    }
}

执行结果为:

“疯狂Ajax讲义”
“疯狂Ajax讲义”
“疯狂Ajax讲义”
“轻量级Java EE企业应用实战”

数组作为引用类型的运行机制

数组引用变量只是一个引用,这个引用变量可以指向任何有效的内存地址,只有当该引用指向有效的内存地址后,才可以通过该数组变量来访问数组元素。
与所有引用变量相同的是,引用变量是访问真实对象的根本方式,如果希望在程序中访问数组对象本身,则只能通过这个数组的引用变量来访问。
数组对象被存储在堆(heap)内存中,如果引用该数组对象的数组引用变量是一个局部变量,那么它被存储在栈(stack)内存中。
Java面向对象系列[v1.0.0][数组详解]_第1张图片
正如我们访问数组里的具体某个元素一样,需要使用p[index]的形式实现
为什么会有堆内存还有栈内存

  • 当一个方法执行时,每个方法都会建立自己的内存栈,在这个方法内定义的变量将会逐个放入这块栈内存里,随着方法的执行结束,这个方法的内存栈也自然销毁,因此所有在方法中定义的局部变量都是放在栈内存中的
  • 在程序中创建一个对象时,这个对象将被保存到运行时数据区中,以便反复利用,这个运行时数据区就是堆内存,堆内存中的对象不会随方法的结束而销毁,它还可能被另一个引用变量所引用,只有当一个对象没有任何引用变量引用它时,系统的垃圾回收器才会在合适的时候回收它
  • 为了切断数组引用变量和实际数组之间的引用关系,让实际数组成为垃圾从而其所占用的空间被垃圾回收器回收,可以直接将该数组的引用变量赋值null

只要类型相互兼容,就可以让一个数组变量指向另一个实际的数组,如下代码所示:

public class ArrayInRam
{
    public static void main(String[] args)
    {
        // 定义并初始化数组,使用静态初始化
        int[] a = {5, 7, 20};
        // 定义并初始化数组,使用动态初始化
        var b = new int[4];
        // 输出b数组的长度
        System.out.println("b数组的长度为:" + b.length);
        // 循环输出a数组的元素
        for (int i = 0, len = a.length; i < len; i++)
        {
            System.out.println(a[i]);
        }
        // 循环输出b数组的元素
        for (int i = 0, len = b.length; i < len; i++)
        {
            System.out.println(b[i]);
        }
        // 因为a是int[]类型,b也是int[]类型,所以可以将a的值赋给b。
        // 也就是让b引用指向a引用指向的数组
        b = a;
        // 再次输出b数组的长度
        System.out.println("b数组的长度为:" + b.length);
    }
}

Java面向对象系列[v1.0.0][数组详解]_第2张图片
Java面向对象系列[v1.0.0][数组详解]_第3张图片
看待数组一定要把数组看成两个部分,一个是数组引用,也就是代码里的数组引用变量,另一个是实际的数组对象。

基本类型数组的初始化

对于基本类型数组而言,数组元素的值直接存储在对应的的数组元素中,因此初始化数组时,先为该数组分配内存空间,然后直接将数组元素的值存入对应的数组元素中。

public class PrimitiveArrayTest
{
    public static void main(String[] args)
    {
        // 定义一个int[]类型的数组变量
        int[] iArr;
        // 动态初始化数组,数组长度为5
        iArr = new int[5];
        // 采用循环方式为每个数组元素赋值。
        for (var i = 0; i <iArr.length; i++)
        {
            iArr[i] = i + 10;
        }
    }
}

Java面向对象系列[v1.0.0][数组详解]_第4张图片
Java面向对象系列[v1.0.0][数组详解]_第5张图片
Java面向对象系列[v1.0.0][数组详解]_第6张图片

引用类型数组的初始化

引用类型数组的数组元素是引用,因此情况就相对复杂了很多,每个数组元素里存储的还是引用,它指向另一块内存,这块内存里存储了有效数据

class Person
{
    public int age; // 年龄
    public double height; // 身高
    // 定义一个info方法
    public void info()
    {
        System.out.println("我的年龄是:" + age + ",我的身高是:" + height);
    }
}
public class ReferenceArrayTest
{
    public static void main(String[] args)
    {
        // 定义一个students数组变量,其类型是Person[]
        Person[] students;
        // 执行动态初始化
        students = new Person[2];
        // 创建一个Person实例,并将这个Person实例赋给zhang变量
        var zhang = new Person();
        // 为zhang所引用的Person对象的age、height赋值
        zhang.age = 15;
        zhang.height = 158;
        // 创建一个Person实例,并将这个Person实例赋给lee变量
        var lee = new Person();
        // 为lee所引用的Person对象的age、height赋值
        lee.age = 16;
        lee.height = 161;
        // 将zhang变量的值赋给第一个数组元素
        students[0] = zhang;
        // 将lee变量的值赋给第二个数组元素
        students[1] = lee;
        // 下面两行代码的结果完全一样,因为lee
        // 和students[1]指向的是同一个Person实例。
        lee.info();
        students[1].info();
    }
}

Java面向对象系列[v1.0.0][数组详解]_第7张图片
Java面向对象系列[v1.0.0][数组详解]_第8张图片
Java面向对象系列[v1.0.0][数组详解]_第9张图片
Java面向对象系列[v1.0.0][数组详解]_第10张图片

多维数组的本质

通过上面的例子可以看出多维数组无非是数组元素是引用类型且引用的指向还是一个数组而已

type[][] arrName;

初始化该二维数组,它本质上还是一维数组,那么按照一维数组的初始化方法

arrName = new type[length][]
public class TwoDimensionTest
{
    public static void main(String[] args)
    {
        // 定义一个二维数组
        int[][] a;
        // 把a当成一维数组进行初始化,初始化a是一个长度为4的数组
        // a数组的数组元素又是引用类型
        a = new int[4][];
        // 把a数组当成一维数组,遍历a数组的每个数组元素
        for (int i = 0, len = a.length; i < len; i++)
        {
            System.out.println(a[i]);
        }
        // 初始化a数组的第一个元素
        a[0] = new int[2];
        // 访问a数组的第一个元素所指数组的第二个元素
        a[0][1] = 6;
        // a数组的第一个元素是一个一维数组,遍历这个一维数组
        for (int i = 0, len = a[0].length; i < len; i++)
        {
            System.out.println(a[0][i]);
        }
    }
}

Java面向对象系列[v1.0.0][数组详解]_第11张图片
Java面向对象系列[v1.0.0][数组详解]_第12张图片

// 同时初始化二维数组的2个维数
int[][] b = new int[3][4];

Java面向对象系列[v1.0.0][数组详解]_第13张图片

// 使用静态初始化的语法来初始化一个二维数组
String[][] str1 = new String[][] {new String[3], new String[] {"hello"}};
// 使用简化的静态初始化语法来初始化二维数组
String[][] str2 = {new String[3], new String[] {"hello"}};
System.out.println(str1[1][0]);
System.out.println(str2[1][0]);

Java面向对象系列[v1.0.0][数组详解]_第14张图片

操作数组的工具类

Java提供的Arrays类里包含了一些static修饰的方法可以直接操作数组,这个Arrays类里包含了如下几个static修饰的方法(static修饰的方法可以直接通过类名调用)

  • *int binarySearch(type[] a, type key):*使用二分法查询key元素值为a数组中出现的索引;如果a数组不包含key元素值,则返回负数,调用该方法之前要求数组中元素已经按升序排列,否则得不到正确结果
  • int binarySearch(type[] a int fromIndex, int toIndex, type key):
  • type[] copyOf(type[] original, int length):
  • type[] copyOfRange(type[] original, int from, int to):
  • boolean equals(type[] a, type[] a2):
  • void fill(type[] a, type val):
  • void fill(type[] a, int from Index, int toIndex, type val):
  • void sort(type[] a):
  • void sort(type[] a, int fromIndex, int toIndex):
  • String to String(type[] a):
public class ArraysTest
{
    public static void main(String[] args)
    {
        // 定义一个a数组
        var a = new int[] {3, 4, 5, 6};
        // 定义一个a2数组
        var a2 = new int[] {3, 4, 5, 6};
        // a数组和a2数组的长度相等,每个元素依次相等,将输出true
        System.out.println("a数组和a2数组是否相等:"
            + Arrays.equals(a, a2));
        // 通过复制a数组,生成一个新的b数组
        var b = Arrays.copyOf(a, 6);
        System.out.println("a数组和b数组是否相等:"
            + Arrays.equals(a, b));
        // 输出b数组的元素,将输出[3, 4, 5, 6, 0, 0]
        System.out.println("b数组的元素为:"
            + Arrays.toString(b));
        // 将b数组的第3个元素(包括)到第5个元素(不包括)赋为1
        Arrays.fill(b, 2, 4, 1);
        // 输出b数组的元素,将输出[3, 4, 1, 1, 0, 0]
        System.out.println("b数组的元素为:"
            + Arrays.toString(b));
        // 对b数组进行排序
        Arrays.sort(b);
        // 输出b数组的元素,将输出[0, 0, 1, 1, 3, 4]
        System.out.println("b数组的元素为:"
            + Arrays.toString(b));
    }
}

Java8增强Arrays类功能

  • void parallelPrefix(type[] array, XxxBinaryOperator op):
  • void parallelPrefix(type[]array, int fromIndex, inttoIndex, XxxBinaryOperator op):
  • void setAll(type[] array, IntToXxxFunction generator):
  • void parallelSetAll(type[] array, IntToXxxFunction generator):
  • void parallelSort(type[] array)
  • void parallelSort(type[] array, int fromIndex, int toIndex):
  • Spliterator.OfXxx spliterator(type[] array):
  • Spliterator.OfXxx spliterator(type[] array, int startInclusive, int endExclusive):
  • XxxStream stream(type[] array):
  • XxxStream stream(type[] array, int startInclusive, int endExclusive):
public class ArraysTest2
{
    public static void main(String[] args)
    {
        var arr1 = new int[] {3, -4, 25, 16, 30, 18};
        // 对数组arr1进行并发排序
        Arrays.parallelSort(arr1);
        System.out.println(Arrays.toString(arr1));
        var arr2 = new int[] {3, -4, 25, 16, 30, 18};
        Arrays.parallelPrefix(arr2, new IntBinaryOperator()
        {
            // left代表新数组中前一个索引处的元素,right代表原数组中当前索引处的元素
            // 新数组的第一个元素总等于原数组第一个元素
            public int applyAsInt(int left, int right)
            {
                return left * right;
            }
        });
        System.out.println(Arrays.toString(arr2));
        var arr3 = new int[5];
        Arrays.parallelSetAll(arr3, new IntUnaryOperator()
        {
            // operand代表正在计算的元素索引
            public int applyAsInt(int operand)
            {
                return operand * 5;
            }
        });
        System.out.println(Arrays.toString(arr3));
    }
}

数组的实际应用场景

public class Num2Rmb
{
    private String[] hanArr = {"零", "壹", "贰", "叁", "肆",
        "伍", "陆", "柒", "捌", "玖"};
    private String[] unitArr = {"十", "百", "千"};

    /**
     * 把一个浮点数分解成整数部分和小数部分字符串
     * @param num 需要被分解的浮点数
     * @return 分解出来的整数部分和小数部分。第一个数组元素是整数部分,第二个数组元素是小数部分。
     */
    private String[] divide(double num)
    {
        // 将一个浮点数强制类型转换为long,即得到它的整数部分
        var zheng = (long) num;
        // 浮点数减去整数部分,得到小数部分,小数部分乘以100后再取整得到2位小数
        var xiao = Math.round((num - zheng) * 100);
        // 下面用了2种方法把整数转换为字符串
        return new String[] {zheng + "", String.valueOf(xiao)};
    }

    /**
     * 把一个四位的数字字符串变成汉字字符串
     * @param numStr 需要被转换的四位的数字字符串
     * @return 四位的数字字符串被转换成的汉字字符串。
     */
    private String toHanStr(String numStr)
    {
        var result = "";
        int numLen = numStr.length();
        // 依次遍历数字字符串的每一位数字
        for (var i = 0; i < numLen; i++)
        {
            // 把char型数字转换成的int型数字,因为它们的ASCII码值恰好相差48
            // 因此把char型数字减去48得到int型数字,例如'4'被转换成4。
            var num = numStr.charAt(i) - 48;
            // 如果不是最后一位数字,而且数字不是零,则需要添加单位(千、百、十)
            if (i != numLen - 1 && num != 0)
            {
                result += hanArr[num] + unitArr[numLen - 2 - i];
            }
            // 否则不要添加单位
            else
            {
                result += hanArr[num];
            }
        }
        return result;
    }

    public static void main(String[] args)
    {
        var nr = new Num2Rmb();
        // 测试把一个浮点数分解成整数部分和小数部分
        System.out.println(Arrays.toString(nr.divide(236711125.123)));
        // 测试把一个四位的数字字符串变成汉字字符串
        System.out.println(nr.toHanStr("609"));
    }
}
public class Gobang
{
    // 定义棋盘的大小
    private static int BOARD_SIZE = 15;
    // 定义一个二维数组来充当棋盘
    private String[][] board;
    public void initBoard()
    {
        // 初始化棋盘数组
        board = new String[BOARD_SIZE][BOARD_SIZE];
        // 把每个元素赋为"╋",用于在控制台画出棋盘
        for (var i = 0; i < BOARD_SIZE; i++)
        {
            for (var j = 0; j < BOARD_SIZE; j++)
            {
                board[i][j] = "╋";
            }
        }
    }
    // 在控制台输出棋盘的方法
    public void printBoard()
    {
        // 打印每个数组元素
        for (var i = 0; i < BOARD_SIZE; i++)
        {
            for (var j = 0; j < BOARD_SIZE; j++)
            {
                // 打印数组元素后不换行
                System.out.print(board[i][j]);
            }
            // 每打印完一行数组元素后输出一个换行符
            System.out.print("\n");
        }
    }
    public static void main(String[] args) throws Exception
    {
        var gb = new Gobang();
        gb.initBoard();
        gb.printBoard();
        // 这是用于获取键盘输入的方法
        var br = new BufferedReader(new InputStreamReader(System.in));
        String inputStr = null;
        // br.readLine():每当在键盘上输入一行内容按回车,用户刚输入的内容将被br读取到。
        while ((inputStr = br.readLine()) != null)
        {
            // 将用户输入的字符串以逗号(,)作为分隔符,分隔成2个字符串
            String[] posStrArr = inputStr.split(",");
            // 将2个字符串转换成用户下棋的座标
            var xPos = Integer.parseInt(posStrArr[0]);
            var yPos = Integer.parseInt(posStrArr[1]);
            // 把对应的数组元素赋为"●"。
            gb.board[yPos - 1][xPos - 1] = "●";
            /*
             电脑随机生成2个整数,作为电脑下棋的座标,赋给board数组。
             还涉及
                1.座标的有效性,只能是数字,不能超出棋盘范围
                2.如果下的棋的点,不能重复下棋。
                3.每次下棋后,需要扫描谁赢了
             */
            gb.printBoard();
            System.out.println("请输入您下棋的座标,应以x,y的格式:");
        }
    }
}

你可能感兴趣的:(Java基础即高端)