JavaSE14-数组

news2024/11/15 13:54:37

目录

1.数组基本用法

1.1.什么是数组

1.2.数组声明

1.3.数组的创建与初始化

1.3.1.基本类型数组

1.3.2.对象数组

1.4.数组的使用

1.4.1.获取长度 & 访问元素

1.4.2.遍历数组

2.数组作为方法的参数

2.1.基本用法

2.2.内存

2.3.引用

2.4.初识 JVM 内存区域划分

2.5.参数传内置类型&参数传数组类型

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

4.数组转字符串

4.1.使用JDK提供的工具类Arrays中的toString()方法

4.2.实现自己版本的数组转字符串

5.数组拷贝

5.1.全拷贝

5.1.1.使用JDK提供的类Arrays中的copyOf()方法

5.1.2.实现自己版本的数组全拷贝

5.2.部分拷贝

5.2.1.使用JDK提供的类Arrays中的copyOfRange()方法

5.2.2.实现自己版本的数组部分拷贝

6.多维数组

6.1.多维数组的声明

6.2.多维数组的实例化

6.2.1.直接为每一维分配内存,创建规则数组。

6.2.2.从最高维起,分别为每一维分配空间,创建不规则数组。

6.3.二维数组


1.数组基本用法

1.1.什么是数组

数组是一种顺序表数据结构。它用一组连续的内存空间,是用于储存多个相同类型数据的集合。

数组的长度在创建时确定,并且在创建后不变。

如果需要建立存储不同类型数据的集合,或者要求这种数据存储结构的长度可以动态变化,可以使用java.util包中的各种集合(collection)类,如Vector等。

1.2.数组声明

可以声明任何类型的数组,包括基本类型(char[] s)和类类型(Point[] p)的数组。

数组声明语句有两种格式:

  • 将表示数组的[]跟随在数组变量之后:int arr[];(C,C++,Java的标准格式)
  • 将表示数组的[]跟随在数组类型之后:int[] arr;(Java特有,推荐使用)

数组声明中不指定数组长度。

1.3.数组的创建与初始化

1.3.1.基本类型数组

//动态初始化
数据类型[] 数组名称 = new 数据类型 [元素个数]; //定好其长度,此时数组中元素都是默认值0(只定义不初始化)

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

//静态初始化
数据类型[] 数组名称 = { 初始化数据 };

int[] arr = new int[3];
int[] arr = new int[]{1, 2, 3};
int[] arr = {1, 2, 3}; //是上一行的简化,二者实际是一样的

1.3.2.对象数组

Point p[];
//创建包含10个Point类型对象引用的数组
p = new Point[10];
//创建10个Point对象
for(int i = 0; i < 10; i++){
   p[i] = new Point(i, i + 1);
}

数组创建后,其元素被系统自动进行初始化,不同类型的数组被初始化为不同的默认值。

1.4.数组的使用

1.4.1.获取长度 & 访问元素

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

注:
1).使用 arr.length 获取数组的长度。("."这个操作为成员访问操作符,后面在面向对象中会经常用到)

2).使用 [ ] 按下标取数组元素。需要注意,下标从 0 开始计数。(数组索引(偏移量)从0开始,表示与数组中第一个元素的相对位置)

3).使用 [ ] 操作既能读取数据,也能修改数据。

4).下标访问操作不能超出有效范围 [0, length - 1] , 如果超出有效范围, 会出现下标越界异常(抛出了 java.lang.ArrayIndexOutOfBoundsException 异常)

1.4.2.遍历数组

int[] arr = {1, 2, 3};

//使用for循环
for (int i = 0; i < arr.length; i++) {
   System.out.println(arr[i]);
} 

//使用for-each循环
for (int x : arr) {
   System.out.println(x);
} 

// 执行结果
1
2
3

2.数组作为方法的参数

2.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.2.内存

内存可直观理解成一个宿舍楼。有一个长长的大走廊,上面有很多房间。

每个房间的大小是 1 Byte (如果计算机有 8G 内存, 则相当于有 80亿个这样的房间)。

每个房间上面又有一个门牌号,这个门牌号称为地址。

2.3.引用

引用相当于一个 "别名",也可以理解成一个指针。

创建一个引用只是相当于创建了一个很小的变量, 这个变量保存了一个整数, 这个整数表示内存中的一个地址。

null 在 Java 中表示 "空引用" ,表示一个无效的内存位置,不能对这个内存进行任何读写操作。一旦尝试读写,就会抛出 NullPointerException异常。Java 中并没有约定 null 和 0 号地址的内存有任何关联。

int[] arr = null; //表示arr不再指向任何地址
System.out.println(arr[0]);
// 执行结果 空指针异常
Exception in thread "main" java.lang.NullPointerException at Test.main(Test.java:6)

2.4.初识 JVM 内存区域划分

程序计数器 (PC Register): 一个很小的空间,保存下一条执行的指令的地址。

虚拟机栈(JVM Stack): 主要存储局部变量和引用。

本地方法栈(Native Method Stack): 本地方法栈与虚拟机栈的作用类似,只不过保存的内容是Native方法的局部变量。在有些版本的 JVM 实现中(例如HotSpot),本地方法栈和虚拟机栈是一起的。空间比较小。

堆(Heap): 使用 new 创建的对象保存在堆上。是JVM所管理的最大内存区域。

方法区(Method Area): 用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。方法编译出的的字节码就是保存在这个区域。

运行时常量池(Runtime Constant Pool): 是方法区的一部分, 存放字面量(字符串常量)与符号引用。(从 JDK1.7 开始, 运行时常量池在堆上)

Native 方法:
JVM 是一个基于 C++ 实现的程序,在 Java 程序执行过程中,本质上也需要调用 C++ 提供的一些函数进行和操作系统底层进行一些交互。因此在 Java 开发中也会调用到一些 C++ 实现的函数。
这里的 Native 方法就是指这些 C++ 实现的,再由 Java 来调用的函数。

程序计数器,虚拟机栈,本地方法栈是每个线程具有一份;而堆,方法区,运行时常量池是整个 JVM 共享一个。

2.5.参数传内置类型&参数传数组类型

//参数传内置类型
public static void main(String[] args) {
    int num = 0;
    func(num);
    System.out.println("num = " + num);
} 

public static void func(int x) {
    x = 10;
    System.out.println("x = " + x);
} 

//执行结果:修改形参 x 的值,不影响实参的 num 值。
x = 10
num = 0

//参数传数组类型
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] = 100;
    System.out.println("a[0] = " + a[0]);
} 

//执行结果:在函数内部修改数组内容,函数外部也发生改变。
//此时数组名 arr 是一个 "引用",当传参的时候,是按照引用传参。
a[0] = 100
arr[0] = 100

 

针对 int[] arr = new int[]{1, 2, 3}:

a) new int[]{1, 2, 3} :在堆上创建了一块内存空间保存三个 int。(有new就有新空间)

b) int[] arr = new int[]{1, 2, 3} :在栈上创建了一个 int[] 变量arr。(引用类型,里面只保存了一个整数(数组的起始内存地址0x100))

c) int[] a = arr:进行传参,在栈上创建了一个 int[] 变量a。保存数组的起始内存地址0x100。

d) a[0] = 100:根据 0x100 地址找到对应的内存位置, 将值由1改成100。

此时已经将 0x100 地址的数据改成了 100 ,那么根据实参 arr 来获取数组内容 arr[0],本质上也是获取 0x100地址上的数据,也是 100。

总结: "引用" 本质上是存了一个地址,Java 将数组设定成引用类型,这样进行数组参数传参,就是将数组的地址传入到函数形参中。这样可以避免对整个数组的拷贝(数组可能比较长,那么拷贝开销就会很大)。

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

代码示例: 写一个方法, 将数组中的每个元素都 * 2

//直接修改原数组
class Test {
    public static void main(String[] args) {
         int[] arr = {1, 2, 3};
         transform(arr);
         printArray(arr);
    } 

    public static void printArray(int[] arr) {
         for (int i = 0; i < arr.length; i++) {
             System.out.println(arr[i]);
         }
    } 

    public static void transform(int[] arr) {
         for (int i = 0; i < arr.length; i++) {
             arr[i] = arr[i] * 2;
         }
    }
}

//返回一个新的数组,不破坏原数组
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;
    }
}

由于数组是引用类型, 返回的时候只是将这个数组的首地址返回给函数调用者, 没有拷贝数组内容, 从而比较高效。

4.数组转字符串

4.1.使用JDK提供的工具类Arrays中的toString()方法

import java.util.Arrays; //包含了一些操作数组的常用方法

int[] arr = {1,2,3,4,5,6};

String newArr = Arrays.toString(arr);

System.out.println(newArr);
System.out.println(arr);

// 执行结果
[1, 2, 3, 4, 5, 6]
[I@1b6d3586] //表示输出的是一个数组对象的地址

什么是包?
程序开发不是从零开始,而是要站在巨人的肩膀上。
很多程序我们在写的过程中不必把所有的细节都自己实现,已经有大量的标准库(JDK提供好的代码)和海量的第三方库(其他机构组织提供的代码)供我们直接使用。这些代码就放在一个一个的 "包" 之中。

4.2.实现自己版本的数组转字符串

public static void main(String[] args) {
    int[] arr = {1,2,3,4,5,6};
    System.out.println(toString(arr));
} 

public static String toString(int[] arr) {
    String ret = "[";
    for (int i = 0; i < arr.length; i++) {
       // 借助 String += 进行拼接字符串
       ret += arr[i];
       // 除了最后一个元素之外, 其他元素后面都要加上 ", "
       if (i != arr.length - 1) {
          ret += ", ";
       }
    } 
    ret += "]";
    return ret;
}

5.数组拷贝

5.1.全拷贝

5.1.1.使用JDK提供的类Arrays中的copyOf()方法

import java.util.Arrays;

int[] arr = {1,2,3,4,5,6};
int[] newArr = Arrays.copyOf(arr, arr.length);

System.out.println("newArr: " + Arrays.toString(newArr));

arr[0] = 10;
System.out.println("arr: " + Arrays.toString(arr));
System.out.println("newArr: " + Arrays.toString(newArr));

//执行结果
newArr: [1, 2, 3, 4, 5, 6]
arr: [10, 2, 3, 4, 5, 6]
newArr: [1, 2, 3, 4, 5, 6]

新拷贝出来的数组和原数组指向的不是同一块内存空间。

相比于 newArr = arr 这样的赋值, copyOf 是将数组进行了深拷贝,即又创建了一个数组对象,拷贝原有数组中的所有元素到新数组中。因此,修改原数组,不会影响到新数组。

5.1.2.实现自己版本的数组全拷贝

public static int[] copyOf(int[] arr) {
    int[] ret = new int[arr.length];
    for (int i = 0; i < arr.length; i++) {
        ret[i] = arr[i];
    } 
    return ret;
}

5.2.部分拷贝

5.2.1.使用JDK提供的类Arrays中的copyOfRange()方法

import java.util.Arrays;

int[] arr = {1,2,3,4,5,6};

int[] newArr2 = Arrays.copyOfRange(arr, 2, 4); 
//拷贝某个范围:arr是原数组引用,2是原数组开始索引,4是结束索引(左闭右开区间)
System.out.println("newArr2: " + Arrays.toString(newArr2));

//执行结果
newArr2: [3, 4]

5.2.2.实现自己版本的数组部分拷贝

public static int[] copyOfRange(int[] arr, int startIndex, int endIndex){
    int[] ret = new int[endIndex - startIndex];
    for(int i = startIndex; i < endIndex; i++){
        //原数组和拷贝后的数组索引相差startIndex个索引
        ret[i - startIndex] = arr[i];
    }
    return ret;
}

6.多维数组

Java中的多维数组可看作是数组的数组,即n维数组是n-1维数组的数组。

6.1.多维数组的声明

要用多对[]表示数组的维数,一般n维数组要用n对[]。

6.2.多维数组的实例化

6.2.1.直接为每一维分配内存,创建规则数组。

int a[][];
a = new int[4][4];

6.2.2.从最高维起,分别为每一维分配空间,创建不规则数组。

int a[][] = new int[2][]; //只有最后维可以不给值,其他都要给
a[0] = new int[10];
a[1] = new int[5];

6.3.二维数组

二维数组本质上是一维数组,不过每个元素又是一个一维数组。

基本语法

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

代码示例

int[][] arr = {
    {1, 2, 3, 4},
    {5, 6, 7, 8},
    {9, 10, 11, 12}
};

for (int row = 0; row < arr.length; row++) {
   for (int col = 0; col < arr[row].length; col++) {
       System.out.printf("%d\t", arr[row][col]);
   } 
   System.out.println("");
}
 
// 执行结果
1   2   3   4
5   6   7   8
9   10  11  12


 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/108349.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

怎么做现货白银的心理障碍

克服投资的心理障碍&#xff0c;是怎么做现货白银投资的关键。很多时候&#xff0c;技术分析是很简单的。一根K线&#xff0c;一条均线&#xff0c;就能够让人获利。但是为什么使用同样工具的人&#xff0c;却不能获利呢&#xff1f;为什么他们还要去追求一些特别复杂的分析系统…

四平方和(蓝桥杯C/C++B组真题详解)(三种做法)

目录 题目详细&#xff1a;​编辑 题目思路&#xff1a; 暴力&#xff1a; 代码详解&#xff1a; 哈希&#xff1a; 二分&#xff1a; 题目详细&#xff1a; 题目思路&#xff1a; 这个题目大家可能马上就可以想到暴力做 例如这样 暴力&#xff1a; #include<iost…

TensorFlow和Keras应如何选择?

前些年&#xff0c;深度学习领域的研究人员、开发人员和工程师必须经常做出一些选择&#xff1a; 我应该选择易于使用但自定义困难的 Keras 库&#xff1f;还是应该使用难度更大的 TensorFlow API&#xff0c;编写大量代码&#xff1f;&#xff08;更不用说一个不那么容易使用…

pyspark之sparksql数据交互

在pyspark中&#xff0c;使用sparksql进行mysql数据的读写处理&#xff0c;将程序保存为test.py #-*- coding: UTF-8 -*- # 设置python的默认编码 import sys reload(sys) sys.setdefaultencoding(utf-8) # Spark 初始化 from pyspark.sql import SQLContext, SparkSession, …

【推荐】DDD领域驱动设计和中台实践资料合集

Domain Driven Design&#xff08;简称 DDD&#xff09;&#xff0c;又称为领域驱动设计&#xff0c;起源于杰出软件建模专家Eric Evans在2003年发表的书籍《DOMAIN-DRINEN DESIGN —TACKLING COMPLEXITY IN THE HEART OF SOFTWARE》&#xff08;中文译名《领域驱动设计—软件核…

卓海科技冲刺创业板:拟募资5.47亿 相宇阳控制52.9%股权

雷递网 雷建平 12月20日无锡卓海科技股份有限公司&#xff08;简称&#xff1a;“卓海科技”&#xff09;日前递交招股书&#xff0c;准备在深交所创业板上市。卓海科技计划募资5.47亿元&#xff0c;其中&#xff0c;1.04亿元用于半导体前道量检测设备扩产项目&#xff0c;1.84…

ev_api_server:大事件node接口项目开发

Headline 大事件后台 API 项目&#xff0c;API 接口文档请参考 https://www.showdoc.cc/escook?page_id3707158761215217 1. 初始化 1.1 创建项目 新建 api_server 文件夹作为项目根目录&#xff0c;并在项目根目录中运行如下的命令&#xff0c;初始化包管理配置文件&#x…

尚医通-前端Vue学习(十)

目录&#xff1a; &#xff08;1&#xff09;node.js介绍 &#xff08;2&#xff09;npm包管理工具 &#xff08;3&#xff09;es6模块化 &#xff08;4&#xff09;babel转码器 &#xff08;5&#xff09;webpack打包工具 &#xff08;1&#xff09;node.js介绍 浏览器的…

Python使用pandas导入xlsx格式的excel文件内容

Python使用pandas导入xlsx格式的excel文件内容1. 基本导入2. 列标题与数据对齐3. 指定导入某个sheet4. 指定行索引5. 指定列索引6. 指定导入列7. 指定导入的行数8. 更多的参数1. 基本导入 在 Python中使用pandas导入.xlsx文件的方法是read_excel()。 # codingutf-8 import pa…

Windows环境下在VScode中运行开源运动规划库(zhm-real / PathPlanning)的方法

本文主要介绍Windows环境下&#xff0c;在Vscode中运行zhm-real发布的开源运动规划库PathPlanning的实现方法&#xff0c;包括环境配置及运行开源包时常见错误解决方法。    一、环境配置 &#xff08;1&#xff09;VScode 下载及安装&#xff0c;官网如下&#xff1a; http…

Flowable工作流进阶使用教程(监听器+流程变量+网关)

一、任务分配和流程变量 1.任务分配 1.1 固定分配 固定分配就是我们前面介绍的&#xff0c;在绘制流程图或者直接在流程文件中通过Assignee来指定的方式 1.2 表达式分配 Flowable使用UEL进行表达式解析。UEL代表Unified Expression Language&#xff0c;是EE6规范的一部分…

【Python】用python将html转化为pdf

其实早在去年就有做过&#xff0c;一直没有写&#xff0c;先简单记录下 1、主要用到的工具【wkhtmltopdf】 【下载地址】wkhtmltopdf 根据系统选择安装包&#xff0c;速度有点慢&#xff0c;先挂着 2、下载Python库 pip install pdfkit pip install wkhtmltopdf 3、简单代码…

CAD教程:CAD自定义之基础设置的操作技巧

在使用国产CAD软件绘制CAD图纸的过程中&#xff0c;有些时候会需要CAD自定义设置&#xff0c;那么你知道浩辰CAD建筑软件中CAD自定义之基础设置怎么使用吗&#xff1f;不知道也没关系&#xff0c;接下来的CAD教程就让小编来给大家介绍一下国产CAD软件——浩辰CAD建筑软件中CAD自…

【1799. N 次操作后的最大分数和】

来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 描述&#xff1a; 给你 nums &#xff0c;它是一个大小为 2 * n 的正整数数组。你必须对这个数组执行 n 次操作。 在第 i 次操作时&#xff08;操作编号从 1 开始&#xff09;&#xff0c;你需要&#xff1a; 选择两个…

实验一 逻辑回归

一、实验目的 &#xff08;1&#xff09;学习并掌握常见的机器学习方法&#xff1b; &#xff08;2&#xff09;能够结合所学的python知识实现机器学习算法&#xff1b; &#xff08;3&#xff09;能够用所学的机器学习算法解决实际问题。 二、实验内容与要求 &#xff08…

设计模式之备忘录模式

Memento design pattern 备忘录模式的概念、备忘录模式的结构、备忘录模式的优缺点、备忘录模式的使用场景、备忘录模式的实现示例、备忘录模式的源码分析 1、备忘录模式的概念 备忘录模式&#xff0c;又称快照模式&#xff0c;即在不破坏封装的前提下&#xff0c;获取并保存一…

【数电】Simulation Test 模拟测试

一、 选择题&#xff1a;&#xff08;共20分&#xff0c;每小题2分&#xff09; 1、逻辑函数的所有最小项之和等于多少&#xff1f; A. 0 B. 1 C. 0或1 D. 任意值 2、与非门的多余输入端应如何处理&#xff1f;…

MySQL面试常问问题(基础) —— 赶快收藏

目录 1. 什么是内连接、外连接、交叉连接、笛卡尔积呢&#xff1f; 2. 那MySQL 的内连接、左连接、右连接有有什么区别&#xff1f; 3.说一下数据库的三大范式&#xff1f; 4.varchar与char的区别&#xff1f; 5.blob和text有什么区别&#xff1f; 6.DATETIME和TIMESTAMP…

SCSS学习笔记

文章目录1.安装scss2.选择器嵌套3.属性嵌套4.父选择器&5.变量5.1变量的规范5.2变量的作用域5.3给变量设置默认值(!default)6数据类型7.运算符8.插值语法9.流程控制语句9.1 条件语句9.2循环语句9.2.1for9.2.2each9.2.3while10import10.1引入scss不编译10.2嵌套引入scss11.mi…

【软件测试】概念篇

目录 一、需求 1.1用户需求 1.2软件需求 1.3需求的重要性 二、测试用例 三、BUG 3.1什么是BUG 3.2如何描述一个BUG 4.3BUG优先级 四、软件开发模型 4.1软件生命周期 4.2开发模型 定义&#xff1a;软件测试就是一系列活动&#xff0c;这些活动是为了评估一个程序或者…