아, 알겠습니다. 소 아저씨 말처럼 얕지도 않고, 꼬마 다람쥐 말처럼 깊지도 않은데, 직접 가보셔야 알 수 있어요! 소문은 항상 현상을 보여줄 수 있습니다. 직접 시도해야만 그 깊이를 알 수 있습니다! ! ! ! !
배열이란? 배열이란 무엇입니까? 내 생각에 배열은 다음과 같아야 합니다. new 키워드를 통해 배열을 만들고 조합하고, 정수 인덱스 값을 사용하여 해당 요소에 액세스하고, 크기는 변경할 수 없습니다!
하지만 이건 배열의 표면일 뿐입니다! 더 깊게? 그게 전부입니다. 배열은 정렬된 데이터의 모음입니다. 배열의 각 데이터는 범위를 벗어나지 않는 첨자 값과 함께 고유하게 사용됩니다. 배열의 요소.
더 깊은 것이 있습니다. 즉, 배열은 특별한 개체입니다! ! (저는 이 LZ를 잘 이해하지 못하고 JVM을 읽지 않았기 때문에 통찰력이 제한적입니다.) 다음 참고자료: //m.sbmmt.com/, //m.sbmmt.com/
무엇, Java에서는 객체입니다. 다소 특별한 개체입니다.
public class Test { public static void main(String[] args) { int[] array = new int[10]; System.out.println("array的父类是:" + array.getClass().getSuperclass()); System.out.println("array的类名是:" + array.getClass().getName()); } } -------Output: array的父类是:class java.lang.Object array的类名是:[I
위의 예에서 볼 수 있듯이 배열은 Object의 직접적인 하위 클래스입니다. ""First Class Object"에 속하지만 일반 Java 개체와는 매우 다릅니다. 클래스 이름에서 알 수 있습니다. [나, 이게 뭐죠? ? JDK에서 이 클래스를 찾지 못했습니다. 이 "[I"는 법적 식별자가 아니라고 합니다. 카테고리로 정의하는 방법은 무엇입니까? 그래서 SUM의 천재들이 배열의 맨 아래 레이어에 특별한 처리를 했음이 틀림없다고 생각합니다.
다음 예를 살펴보겠습니다.
public class Test { public static void main(String[] args) { int[] array_00 = new int[10]; System.out.println("一维数组:" + array_00.getClass().getName()); int[][] array_01 = new int[10][10]; System.out.println("二维数组:" + array_01.getClass().getName()); int[][][] array_02 = new int[10][10][10]; System.out.println("三维数组:" + array_02.getClass().getName()); } } -----------------Output: 一维数组:[I 二维数组:[[I 三维数组:[[[I
이 예를 통해 우리는 다음을 알 수 있습니다. [는 배열의 차원을 나타내고, 하나의 [는 1차원을 나타내고, 두 개의 [는 2차원을 나타냅니다. 간단히 말해서 배열의 클래스 이름은 여러 개의 '['와 배열 요소 유형의 내부 이름으로 구성됩니다. 확실하지 않다면 다시 살펴보겠습니다.
아아아아
. 동시에 배열은 일반 Java 클래스와 다르다는 것을 알 수 있습니다. 일반 Java 클래스는 정규화된 경로 이름 + 클래스 이름을 유일한 식별자로 사용하는 반면 배열은 여러 [+L+array 요소 클래스 전체 이름을 정의합니다. 경로 + 클래스를 사용하여 고유하게 식별합니다. 이러한 차이점은 배열과 일반 Java 클래스의 구현에 큰 차이가 있음을 어느 정도 설명할 수 있습니다. 아마도 이 차이는 JVM이 배열과 일반 Java 클래스를 처리할 때 구별하는 데 사용될 수 있습니다.
이건 뭐 [나, 누가 선언했는지, 어떻게 선언했는지 (지금은 모르겠다!) 확인할 수 있음: 런타임 시 결정됩니다. 먼저 다음을 살펴보세요.
public class Test {
public static void main(String[] args) {
System.out.println("Object[]:" + Object[].class);
System.out.println("Object[][]:" + Object[][].class);
System.err.println("Object[][][]:" + Object[][][].class);
System.out.println("Object:" + Object.class);
}
}
---------Output:
Object[]:class [Ljava.lang.Object;
Object[][]:class [[Ljava.lang.Object;
Object[][][]:class [[[Ljava.lang.Object;
Object:class java.lang.Object
멤버 변수가 없습니다. 메소드, 생성자, 주석 또는 심지어 길이 멤버 변수입니다. 완전히 비어 있는 클래스입니다. 길이가 선언되지 않았는데, array.length를 사용할 때 컴파일러가 오류를 보고하지 않는 이유는 무엇입니까? 실제로 배열의 길이는 매우 특별한 멤버 변수입니다. 배열이 Object의 직접 클래스라는 것을 알고 있지만 Object에는 멤버 변수 length가 없으므로 length가 배열의 멤버 변수여야 하지만 위의 예를 보면 배열에 멤버 변수가 없다는 것을 알 수 있습니다. 전혀 모순되지 않나요?
public class Test { public static void main(String[] args) { int[] array = new int[10]; Class clazz = array.getClass(); System.out.println(clazz.getDeclaredFields().length); System.out.println(clazz.getDeclaredMethods().length); System.out.println(clazz.getDeclaredConstructors().length); System.out.println(clazz.getDeclaredAnnotations().length); System.out.println(clazz.getDeclaredClasses().length); } } ----------------Output: 0 0 0 0 0
클래스 파일을 열고 기본 메소드의 바이트코드를 가져옵니다.
0 iconst_2 //将int型常量2压入操作数栈 1 newarray 10 (int) //将2弹出操作数栈,作为长度,创建一个元素类型为int, 维度为1的数组,并将数组的引用压入操作数栈 3 astore_1 //将数组的引用从操作数栈中弹出,保存在索引为1的局部变量(即a)中 4 aload_1 //将索引为1的局部变量(即a)压入操作数栈 5 arraylength //从操作数栈弹出数组引用(即a),并获取其长度(JVM负责实现如何获取),并将长度压入操作数栈 6 istore_2 //将数组长度从操作数栈弹出,保存在索引为2的局部变量(即i)中 7 return //main方法返回
在这个字节码中我们还是没有看到length这个成员变量,但是看到了这个:arraylength ,这条指令是用来获取数组的长度的,所以说JVM对数组的长度做了特殊的处理,它是通过arraylength这条指令来实现的。
通过上面算是对数组是什么有了一个初步的认识,下面将简单介绍数组的使用方法。
数组的使用方法无非就是四个步骤:声明数组、分配空间、赋值、处理。
声明数组:就是告诉计算机数组的类型是什么。有两种形式:int[] array、int array[]。
分配空间:告诉计算机需要给该数组分配多少连续的空间,记住是连续的。array = new int[10];
赋值:赋值就是在已经分配的空间里面放入数据。array[0] = 1 、array[1] = 2……其实分配空间和赋值是一起进行的,也就是完成数组的初始化。有如下三种形式:
int a[] = new int[2]; //默认为0,如果是引用数据类型就为null int b[] = new int[] {1,2,3,4,5}; int c[] = {1,2,3,4,5};
处理:就是对数组元素进行操作。通过数组名+有效的下标来确认数据。
以上就是java提高篇(十八)-----数组之一:认识JAVA数组 的内容,更多相关内容请关注PHP中文网(m.sbmmt.com)!