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", "李刚"};
}
}
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];
执行动态初始化时,程序员只需要指定数组的长度,即为每个数组元素指定所需的内存空间,系统负责为这些数组元素分配初始值,系统分配初始值规则如下
* 整数类型(byte、short、int和long)数组元素值为0
* 浮点类型(float、double)数组元素值为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
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)内存中。
正如我们访问数组里的具体某个元素一样,需要使用p[index]的形式实现
为什么会有堆内存还有栈内存
只要类型相互兼容,就可以让一个数组变量指向另一个实际的数组,如下代码所示:
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);
}
}
看待数组一定要把数组看成两个部分,一个是数组引用,也就是代码里的数组引用变量,另一个是实际的数组对象。
对于基本类型数组而言,数组元素的值直接存储在对应的的数组元素中,因此初始化数组时,先为该数组分配内存空间,然后直接将数组元素的值存入对应的数组元素中。
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;
}
}
}
引用类型数组的数组元素是引用,因此情况就相对复杂了很多,每个数组元素里存储的还是引用,它指向另一块内存,这块内存里存储了有效数据
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();
}
}
通过上面的例子可以看出多维数组无非是数组元素是引用类型且引用的指向还是一个数组而已
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]);
}
}
}
// 同时初始化二维数组的2个维数
int[][] b = new int[3][4];
// 使用静态初始化的语法来初始化一个二维数组
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提供的Arrays类里包含了一些static修饰的方法可以直接操作数组,这个Arrays类里包含了如下几个static修饰的方法(static修饰的方法可以直接通过类名调用)
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类功能
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的格式:");
}
}
}