解析java中的数组

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 解析java中的数组

数组的定义

数组:本质上就是一组相同类型的变量。

二 数组的基本语法

1) 数据类型[] 数组名称 = new 数据类型 [] { 初始化数据 };
例:int[] arr = new int[]{1, 2, 3};
2) 数据类型[] 数组名称 = { 初始化数据 };
例: int[] arr = {1, 2, 3};
3) 数据类型[] 数组名称 = new int[定义数组长度];//这种定义数组里面全部初始化为零
例:int[] arr = new int[10];

三 数组的使用

1 数组获取长度与访问元素

int[] arr = {1, 2, 3};
// 获取数组长度
System.out.println("length: " + arr.length); // 执行结果: 3
// 访问数组中的元素
System.out.println(arr[1]); // 执行结果: 2

注意事项:

(1): 访问数组元素的下标是从0开始的

(2):下表访问的数不能超过【0,length-1】,否则会出现数组越界,这样程序会报错。

(3):利用[]可以对数组的内容进行更改与读取操作

2 数组的遍历

(1)for循环遍历

int[] arr = {1, 2, 3};
for (int i = 0; i < arr.length; i++) {
System.out.println(arr[i]);
}

(2)for-each遍历数组

int[] arr = {1, 2, 3};
for (int x : arr) {
System.out.println(x);
}

四 数组作为方法的参数

(1)基本用法

public static void main(String[] args) {
  int[] arr = {1, 2, 3};
  printArray(arr);
}
public static void printArray(int[] a) {
  for (int x : a) {
  System.out.println(x);
  }
}
// 执行结果
1
2
3

在以上的这个数组中,int[] a 是函数的形参, int[] arr 是函数实参.

(2)理解引用类型

参数传数组类型就是属于引用类型

public static void main(String[] args) {
   int[] arr = {1, 2, 3};
   func(arr);
    System.out.println("arr[0] = " + arr[0]);
}
public static void func(int[] a) {
        a[0] = 10;
       System.out.println("a[0] = " + a[0]);
}
// 执行结果
a[0] = 10
arr[0] = 10

我们发现, 在方法内部修改数组内容, 主函数数组内容也发生改变.

此时数组名 arr 是一个 “引用” . 当传参的时候, 是按照引用传参.

这个就需要我们从内存这个知识去解释了,在内存中,局部变量创建是放在JVM的栈上

,对象是在堆上。在上面这个代码中,首先进入main函数,为arr开辟一个内存空间,放到栈上,里面存放的是对象的地址(0x789),而这里的对象就是对数组的内容(1,2,3)会被放到堆上地址为(0x789),通过地址,arr就能够访问到对象,而之后用到方法,arr会把它的地址传给a,此时JVM的栈上有开辟了一个空间(0x789),通过a,也可以访问到arr的对象,可以进行改写与读操作。

图例:

总结:

所谓的 “引用” 本质上只是存了一个地址. Java 将数组设定成引用类型

五 认识null

null 在 Java 中表示 “空引用” , 也就是一个无效的引用

例题

int[] arr = null;
System.out.println(arr[0]);
// 执行结果
Exception in thread "main" java.lang.NullPointerException
at Test.main(Test.java:6)

null 的作用类似于 C 语言中的 NULL (空指针), 都是表示一个无效的内存位置. 因此不能对这个内存进行任何读写操作. 一旦尝试读写, 就会抛出 NullPointerException.

注意:

Java 中并没有约定 null 和 0 号地址的内存有任何关联

六. 数组作为方法的返回值

// 返回一个新的数组
class Test {
public static void main(String[] args) {
  int[] arr = {1, 2, 3};
  int[] output = transform(arr);
  printArray(output);
}
public static void printArray(int[] arr) {
  for (int i = 0; i < arr.length; i++) {
 System.out.println(arr[i]);
 }
}
public static int[] transform(int[] arr) {
      int[] ret = new int[arr.length];
      for (int i = 0; i < arr.length; i++) {
      ret[i] = arr[i] * 2;
      }
   return ret;
  }
}

从以上代码分析可以得出,数组可以作为返回值类型,从而返回一个数组,特别注意一下这个ret数组,这个ret数组里面存放的地址不在是arr内存中的地址,此时的ret数组里面的内容会与arr的内容一致。

七 数组的拷贝

1 copyof

import java.util.Arrays;//这个是导入的包,在下面使用Arrays后自动导入

public class java1027 {
    public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] ret = Arrays.copyOf(arr,arr.length);//arr表示要进行拷贝的数组,arr.length表示拷贝的长度
    }

注意这里拷贝的长度可以是乘以一个实数,比如乘以2,那么拷贝后ret数组就变成{1,2,3,4,5,0,0,0,0,0}这个就是扩容处理

2 copyRange

import java.util.Arrays;

public class java1027 {
    public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] ret = Arrays.copyOfRange(arr,2,4);//arr表示原数组,2表示的是拷贝的起始位置,4表示拷贝结束的位置
    }

3 clone()

public class java1027 {
    public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] ret = arr.clone();
    }

4 System.arraycopy

import java.util.Arrays;

public class java1027 {
    public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] ret = new int[arr.length];
        System.arraycopy(arr,0,ret,0,arr.length);
        System.out.println(Arrays.toString(ret));//数组以字符串的形式输出
    }//arr表示的是原数组,0表示从原数组0下标开始拷贝,ret表示目的数组,0表示从下标0开始,arr.length表示结束下标

以上四种拷贝方式都是属于浅拷贝,至于深拷贝,后续的内容我们会详解

八. 二维数组

1 基本语法

1)数据类型[][] 数组名称 = new 数据类型 [行数][列数] { 初始化数据 };

2)数据类型[][] 数组名称 ={ 初始化数据 };


public class java1027 {
    public static void main(String[] args) {
    int[][] arr = {{1,2,4},{1,2,3}};
     System.out.println(arr.length);
     System.out.println(arr[0].length);
    }

运行结果

2

3

二维数组arr从这张图就可以看出二维数组是一个特殊的一维数组,结合上诉代码就可以看出arr.length代表的是整个行的长度,arr[0].length(或者arr[1].length)代表的是列的长度。知道这个之后就可以遍历二维数组的内容了,

2 二维数组的遍历

(1)for循环进行遍历


public class java1027 {
    public static void main(String[] args) {
    int[][] arr = {{1,2,4},{1,2,3}};
        for (int i = 0; i < arr.length; i++) {//访问行
            for (int j = 0; j < arr[i].length; j++) {//访问列
                System.out.print(arr[i][j]+" ");

            }
            System.out.println();
        }
    }

运行结果

1 2 4

1 2 3

(2)for-each循环遍历

public class java1027 {
    public static void main(String[] args) {
    int[][] arr = {{1,2,4},{1,2,3}};
        for (int[] ret:arr) {
            for (int x:ret) {
                System.out.print(x+" ");
            }
            System.out.println();
        }
    }

(3)调用Arrays

import java.util.Arrays;

public class java1027 {
    public static void main(String[] args) {
    int[][] arr = {{1,2,4},{1,2,3}};
   System.out.println(Arrays.deepToString(arr));
    }

运行结果

[[1, 2, 4], [1, 2, 3]]

3 不规则的二维数组

public class java1027 {
    public static void main(String[] args) {
    int[][] arr = new int[2][];
    arr[0]=new int[3];
    arr[1]=new int[2];
        for (int i = 0; i < arr.length; i++) {
            for (int j = 0; j < arr[i].length; j++) {
                System.out.print(arr[i][j]+" ");
            }
            System.out.println();
        }

    }

运行结果

0 0 0

0 0

目录
相关文章
|
3天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
15 2
|
7天前
|
Java
轻松上手Java字节码编辑:IDEA插件VisualClassBytes全方位解析
本插件VisualClassBytes可修改class字节码,包括class信息、字段信息、内部类,常量池和方法等。
49 6
|
4天前
|
存储 算法 Java
Java Set深度解析:为何它能成为“无重复”的代名词?
Java的集合框架中,Set接口以其“无重复”特性著称。本文解析了Set的实现原理,包括HashSet和TreeSet的不同数据结构和算法,以及如何通过示例代码实现最佳实践。选择合适的Set实现类和正确实现自定义对象的hashCode()和equals()方法是关键。
15 4
|
7天前
|
Java 编译器 数据库连接
Java中的异常处理机制深度解析####
本文深入探讨了Java编程语言中异常处理机制的核心原理、类型及其最佳实践,旨在帮助开发者更好地理解和应用这一关键特性。通过实例分析,揭示了try-catch-finally结构的重要性,以及如何利用自定义异常提升代码的健壮性和可读性。文章还讨论了异常处理在大型项目中的最佳实践,为提高软件质量提供指导。 ####
|
11天前
|
存储 分布式计算 Java
存算分离与计算向数据移动:深度解析与Java实现
【11月更文挑战第10天】随着大数据时代的到来,数据量的激增给传统的数据处理架构带来了巨大的挑战。传统的“存算一体”架构,即计算资源与存储资源紧密耦合,在处理海量数据时逐渐显露出其局限性。为了应对这些挑战,存算分离(Disaggregated Storage and Compute Architecture)和计算向数据移动(Compute Moves to Data)两种架构应运而生,成为大数据处理领域的热门技术。
32 2
|
11天前
|
设计模式 安全 Java
Java编程中的单例模式深入解析
【10月更文挑战第31天】在编程世界中,设计模式就像是建筑中的蓝图,它们定义了解决常见问题的最佳实践。本文将通过浅显易懂的语言带你深入了解Java中广泛应用的单例模式,并展示如何实现它。
|
10天前
|
存储 Java 开发者
Java中的集合框架深入解析
【10月更文挑战第32天】本文旨在为读者揭开Java集合框架的神秘面纱,通过深入浅出的方式介绍其内部结构与运作机制。我们将从集合框架的设计哲学出发,探讨其如何影响我们的编程实践,并配以代码示例,展示如何在真实场景中应用这些知识。无论你是Java新手还是资深开发者,这篇文章都将为你提供新的视角和实用技巧。
11 0
|
8天前
|
安全 Java 测试技术
Java并行流陷阱:为什么指定线程池可能是个坏主意
本文探讨了Java并行流的使用陷阱,尤其是指定线程池的问题。文章分析了并行流的设计思想,指出了指定线程池的弊端,并提供了使用CompletableFuture等替代方案。同时,介绍了Parallel Collector库在处理阻塞任务时的优势和特点。
|
4天前
|
安全 Java 开发者
深入解读JAVA多线程:wait()、notify()、notifyAll()的奥秘
在Java多线程编程中,`wait()`、`notify()`和`notifyAll()`方法是实现线程间通信和同步的关键机制。这些方法定义在`java.lang.Object`类中,每个Java对象都可以作为线程间通信的媒介。本文将详细解析这三个方法的使用方法和最佳实践,帮助开发者更高效地进行多线程编程。 示例代码展示了如何在同步方法中使用这些方法,确保线程安全和高效的通信。
23 9
|
7天前
|
存储 安全 Java
Java多线程编程的艺术:从基础到实践####
本文深入探讨了Java多线程编程的核心概念、应用场景及其实现方式,旨在帮助开发者理解并掌握多线程编程的基本技能。文章首先概述了多线程的重要性和常见挑战,随后详细介绍了Java中创建和管理线程的两种主要方式:继承Thread类与实现Runnable接口。通过实例代码,本文展示了如何正确启动、运行及同步线程,以及如何处理线程间的通信与协作问题。最后,文章总结了多线程编程的最佳实践,为读者在实际项目中应用多线程技术提供了宝贵的参考。 ####

推荐镜像

更多