【Java】/*数组的定义与使用*/

目录

一、数组的定义

1.1 为什么要使用数组

1.2 什么是数组

1.3 数组的初始化 

1.3.1 动态初始化

1.3.2 静态初始化

1.2.3 注意事项 

三、遍历数组

3.1 用循环的方式遍历数组

3.2 用 for each 的方式遍历数组

3.3 用 Arrays.toString 的方式遍历数组

3.4 一些其他的补充

四、如何理解数组名 

五、数组的存储

5.1 JVM内存分布

5.2 内置类型变量与引用类型变量的区别

5.3 再谈引用变量

5.4 认识 null

​编辑

六、数组作为函数的参数

七、数组作为函数的返回值

八、数组练习

8.1 数组转字符串

8.2 数组拷贝

8.3 求数组中元素的平均值

8.4 查找数组中指定元素的下标(顺序查找)

8.5 查找数组中指定元素的下标(二分查找)

8.6 数组排序(冒泡排序)

8.7 数组逆序

8.8 判断数组内容是否一致

8.9  给数组中连续几个元素填充为同一个值

8.10 总结Arrays当前已经学过的方法(9个)

九、二维数组


一、数组的定义

1.1 为什么要使用数组

假设我们要存储20个学生的数学成绩,根据之前所学过数据类型和变量的知识,我们就需要创建20个整型变量,如果学生的人数是100、200的话那我们就的创建100、200个整型变量,这显然是非常的不方便,由此引入了数组这个概念。

1.2 什么是数组

1. 概念:数组是相同类型元素的一个集合。

2. 特点:

    ① 数组中的各元素类型保持一致。

    ② 数组中每个元素的内存是连续的。

    ③ 和C语言一样,可以根据下标访问数组中的每个元素,下标也是从0开始的

1.3 数组的初始化 

1.3.1 动态初始化

1. 概念:在创建数组时,直接指定数组中元素的个数

2. 示例:int[] array = new int[10];

3. 动态初始化分步写,示例:

    int[] array1;

    array1 = new int[10];

4. 动态初始化,会自动为每个元素赋初始值,比如:数组中存储元素类型为byte、short、int、long类型,每个元素会被初始化为0;数组中存储元素类型为float类型,每个元素会被初始化为0.0f;数组中存储元素类型为double类型,每个元素会被初始化为0.0;数组中存储元素类型为char类型,每个元素会被初始化为/u0000;数组中存储元素类型为boolean类型,每个元素会被初始化为false;另外,如果数组中存储元素类型为引用类型,默认值为null。

1.3.2 静态初始化

1. 概念:在创建数组时不直接指定数组中元素的个数,编译器会根据{}中的元素个数确定

2. 示例1:int[] array1 = {1,2,3,4,5}; -->通常使用这种省略版本,编译时会还原成未省略的。

3. 示例2:int[] array2 = new int[]{1,2,3,4,5};

4. 静态初始化分步写,示例:

    int[] array;

    array = new int[]{1,2,3,4,5};

    //注意:省略版本不能采用分步初始化,否则编译失败

    //int[] array;

    //array = {1,2,3,4,5};//err

1.2.3 注意事项 

数组类型int[]中的[]中不要写数字,new int[]中的[]在静态初始化时也不要写数字,否则会报错

② Java中的数组的创建也可以写成 int array[]; 的形式,但并不推荐。

Java中的变量未被初始化是不能够被直接使用的,数组变量也一样,例如:

    int[] array;

    System.out.println(array);//err

    //注意:动态初始化会每个数组元素赋值,所以动态初始化后使用数组没有问题,例如:

    int[] array = new int[5];

    System.out.println(array);//right

三、遍历数组

3.1 用循环的方式遍历数组

3.2 用 for each 的方式遍历数组

说明:for-each 是 for 循环的另外一种使用方式,for后面()中的格式是,冒号左边创建一个可临时接收数组元素的变量,注意要于数组元素的类型保持一致;冒号右边写数组名。

3.3 用 Arrays.toString 的方式遍历数组

3.4 一些其他的补充

补充一,关于数组越界访问:

数组越界访问编译器会报数组越界异常,如下图

 补充二,关于程序中有多个错误,编译器是如何报错的:

下面的这段代码中,编译器只会报第一个错误,是因为后面的错误程序还未执行到那里

四、如何理解数组名 

1. 编译如下代码,会在屏幕上打印出,[I@4eec7777

    int[] array = new int[]{1,2,3,4,5};

    System.out.println(array);

2. 如何理解 [I@4eec7777 ?下图中所说的地址是经过处理过的,也就是说不是真正的地址,但它具有唯一性,所以也可以广义的认为是地址;这个地址是数组首元素的地址,即数组名是数组首元素的地址,array不是一个基本数据类型的变量,而是一个引用数据类型的变量 (用来存地址的变量称为引用变量),简称array为引用,这个引用指向着一个对象(该数组开辟的内存空间);new关键字可以简单理解为用来创建对象的(专业术语:实例化对象)。

五、数组的存储

5.1 JVM内存分布

1. 问:内存为什么要划分几个区域?

    答:内存是一段连续的存储空间,主要用来存储程序运行时数据的。比如:

           ① 程序运行时代码需要加载到内存

           ② 程序运行产生的中间数据要存放在内存

           ③ 程序中的常量也要保存

           ④ 有些数据可能需要长时间存储,而有些数据当方法运行结束后就要被销毁

           如果对内存中存储的数据不加区分的随意存储,那对内存管理起来将会非常麻烦,因此JVM也对所使用的内存按照功能的不同进行了划分,以便于更好的管理数据:

虚拟机栈(JVM Stack):与方法调用相关的一些信息,每个方法在执行时,都会先创建一个栈帧,栈帧中包含有:局部变量表、操作数栈、动态链接、返回地址以及其他的一些信息,保存的都是与方法执行时相关的一 些信息。比如:局部变量。当方法运行结束后,栈帧就被销毁了,即栈帧中保存的数据也被销毁了。

② 本地方法栈(Native Method Stack):本地方法栈与虚拟机栈的作用类似,只不过保存的内容是Native方法的局部变量(提供给底层的C/C++代码使用)。 在有些版本的 JVM 实现中(例如HotSpot), 本地方法栈和虚拟机栈是一起的。

③ 堆(Heap):JVM所管理的最大内存区域。使用 new 创建的对象都是在堆上保存 (例如前面的 new int[]{1, 2, 3} ),堆是随着程序开始运行时而创建,随着程序的退出而销毁,堆中的数据只要还有在使用,就不会被销毁。

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

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

---------> 现在我们只简单关心堆 和 虚拟机栈这两块空间,后序JVM中还会更详细介绍。

5.2 内置类型变量与引用类型变量的区别

1. 区别在于部存储的数据意义不同:

     基本数据类型创建的变量,称为基本变量,该变量空间中直接存放的是其所对应的值;

     引用数据类型创建的变量,一般称为引用变量,其空间中存储的是它指向的对象所在内存空间的地址。

2. public static void func() {

            int a = 10;

            int b = 20;

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

    }

3. 在上述代码中,a、b、arr,都是func方法内部的变量,因此其内存空间都在func方法对应的栈帧中分配。 a、b是内置类型的变量,因此其空间中保存的就是给该变量初始化的值。 array是数组类型的引用变量,其内部保存的内容可以理解成是数组在堆空间中数组首地址。

4. 从下图可以看到,引用变量并不直接存储对象本身存储的是对象在堆中开辟空间的起始地址通过该地址,引用变量便可以去操作对象。有点类似C语言中的指针,但是Java中引用要比指针的操作更简单。

5.3 再谈引用变量

下面将简要分析上面代码中的变量在JVM内存中的分布情况,以及在屏幕上打印的100 200 300 400 500的原因:

1. 在func方法栈帧中为array1这个引用开辟空间,并为array1这个引用所指向的对象在堆区开辟3个连续的整型空间,array1数组中的每个元素的值都被初始化为0,把该空间的起始地址0x112存入array1这个引用中。

2. 通过下标的方式将array1数组中下标为0、1、2的元素中的值修改为10、20、30。

3. 在func方法栈帧中为array2这个引用开辟空间,并为array2这个引用所指向的对象在堆区开辟5个连续的整型空间,array2数组中的每个元素的值都被初始化为0,把该空间的起始地址0x345存入array2这个引用中。

4. 通过下标的方式将array2数组中下标为0、1的元素中的值修改为100、200。

5. array1=array2,即让array1这个引用不再指向堆区地址为0x112的数组空间,而让它与array2这个引用指向同一个对象,这一步结束后,堆区地址为0x112的数组空间将被回收。

6. 通过下标的方式将array1这个引用所指向的对象,即堆区地址为0x345的数组空间中下标为2、3、4的元素中的值修改为300、400、500,此时数组array2中的数组元素依次为100、200、300、400、500。

7. 最后通过循环的方式遍历数组,打印数组array2中的每个元素。

通过上面这个例子也可以知道,一个引用不可以指向多个对象,而一个对象可以被多个引用所指

5.4 认识 null

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

2. 注意: Java 中并没有约定 null 和 0 号地址的内存有任何关联。

六、数组作为函数的参数

七、数组作为函数的返回值

八、数组练习

8.1 数组转字符串

8.2 数组拷贝

我们点进 copy0f 的实现可以发现它其实是 System.arraycopy 的一个壳子,

点进  System.arraycopy 会发现看不到它代码的实现,native 其实已经告诉了我们为什么(如果一个方法被native修饰,证明这个方法的底层是C/C++代码实现的,这个方法的运行是运行在本地方法栈中的)

我们可以尝试使用一下  System.arraycopy 如下图。

总结一下,目前我们就有了3种拷贝数组的方式了

① for循环 ②Aarrays.copy0f ③ System.arraycopy

8.3 求数组中元素的平均值

8.4 查找数组中指定元素的下标(顺序查找)

8.5 查找数组中指定元素的下标(二分查找)

自己实现款:

 利用Arrays自带的方法实现:

8.6 数组排序(冒泡排序)

8.7 数组逆序

8.8 判断数组内容是否一致

8.9  给数组中连续几个元素填充为同一个值

8.10 总结Arrays当前已经学过的方法(9个)

toSting、copyOf、copyOfRange、(System.arraycopy)、

length、sort、binarySearch、equals、fill

九、二维数组

1. Java中的二维数组在初始化时不能省略二维数组中每个元素(一维数组)的{},否则会报错。

2. 在遍历二维数组时,通常会采用下面代码图中遍历二维数组的方式二,即通过对二维数组中每个元素是一维数组的理解进行遍历。

3. Java中二维数组采用动态初始化时的列号可以省略, 此时二维数组中的每个一维数组的地址为null,因此不能对省略列号的数组直接进行使用;列号可以省略,提供给了程序员定义不规则二维数组的方式。

    本篇文章已完结,谢谢支持哟 ^^ !!!

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【3dmax笔记】028:倒角的使用方法

一、倒角描述 在3dmax中创建倒角效果可以通过多种方法实现,以下是几种常见的方法: 使用倒角修改器。首先创建一个图形(如矩形和圆),然后对齐它们,将它们转化为可编辑样条线,并附加在一起,选择要倒角的边缘,然后使用倒角修改器来调整高度、轮廓等参数。使用倒角剖面修…

【稳定检索|投稿优惠】2024年医学、药学与生物工程国际会议(ICMPB 2024)

2024年医学、药学与生物工程国际会议(ICMPB 2024) 2024 International Conference on Medicine, Pharmacy, and Biotechnology 【会议简介】 2024年医学、药学与生物工程国际会议将于长沙召开。此次会议将汇聚全球医学、药学与生物工程领域的顶尖学者…

MIT 6.5840(6.824) Lab2:Key/Value Server 设计实现

1 实验要求 在本次 Lab 中,你将在单机上构建一个键/值服务器,以确保即使网络出现故障,每个操作也只能执行一次,并且操作是可线性化的。 客户端可以向键/值服务器发送三个不同的 RPC: Put(key, value) 、 Append(key,…

【Linux】进程间通信(一)---- 匿名管道

【Linux】进程间通信(一)---- 匿名管道 一.序1什么是进程间通信2.进程间通信的标准3.为什么需要进程通信 二.匿名管道1.原理2.使用3.四种情况4.五个特点 一.序 1什么是进程间通信 进程间通信 通信我们大致知道是啥,就是互相传递信息 那进程…

【软考】设计模式之组合模式

目录 1. 说明2. 应用场景3. 结构图4. 构成5. 优点6. 缺点7. java示例 1. 说明 1.将对象组合成树型结构以表示“部分-整体”的层次结构。2.Composite使得用户对单个对象和组合对象的使用具有一致性。3.组合模式(Composite Pattern)是一种结构型设计模式 …

德昂信息-Wyn助力构建HR人员信息分析看板

”葡萄城的Wyn商业智能软件产品为德昂信息提供了强大的支持,借助Wyn商业智能软件,可以通过可视化方式展示整个公司的人员信息及其分析看板。“ ——德昂信息技术(北京)有限公司 公司简介 德昂信息技术(北京)有限公司(以下简称德昂信息&…

STM32_HAL_TIM_通用计时器_实现计时

项目思路 1使用定时器计数每秒一次 2使用一个变量记录定时器响应多少次 3使用UART将记录的次数发出 1STM32Cude设置 1配置时钟源 2打开UART 3打开TIM2 3.1界面介绍 3.2选项介绍 Slave Mode(从模式):当设备被设置为从模式时&#xff0c…

计算机组成原理(超详解!!) 第八节 总线系统

1.总线的概念和结构形态 1.总线(BUS)的基本概念 是构成计算机系统的互联机构,是多个系统功能部件(运算器、控制器、存储器、输入/输出设备)之间进行数据传送的公共通路。 由传输信息的电路和管理信息传输的协议组成…

数据结构【顺序表】

文章目录 1.顺序表的概念线性表物理结构逻辑结构 2.顺序表的分类2.1静态顺序表2.2动态顺序表 3.顺序表接口的实现头文件(SQList.h)如下源文件初始化顺序表销毁顺序表插入扩容尾插头插 封装扩容函数删除尾删头删 查找元素在指定位置前插入数据情况一(指定的位置不是首元素)情况二…

【机器学习】逻辑回归:智能垃圾邮件分类实例

逻辑回归:智能垃圾邮件分类的利器 一、引言二、逻辑回归概述三、垃圾邮件分类实例数据准备特征选择与建模 四、总结与展望 一、引言 随着互联网的迅猛发展,电子邮件已成为人们日常生活和工作中不可或缺的一部分。然而,与此同时,垃…

[蓝桥杯]真题讲解:合并数列(双指针+贪心)

[蓝桥杯]真题讲解&#xff1a;班级活动&#xff08;贪心&#xff09; 一、视频讲解二、正解代码1、C2、python33、Java 一、视频讲解 [蓝桥杯]真题讲解&#xff1a;合并数列&#xff08;双指针贪心&#xff09; 二、正解代码 1、C #include<bits/stdc.h> #define in…

k8s 网络组件详细 介绍

目录 一 k8s 有哪些网络组件 二 k8s 网络概念 1&#xff0c; k8s 三种网络 2&#xff0c;K8S 中 Pod 网络通信 2.1 Pod 内容器与容器之间的通信 2.2 同一个 Node 内 Pod 之间的通信 2.3 不同 Node 上 Pod 之间的通信 三 Flannel 网络组件 1&#xff0c;Flannel …

K8s源码分析(一)-K8s调度框架及调度器初始化介绍

本文首发在个人博客上&#xff0c;欢迎来踩&#xff01; 文章目录 调度框架介绍K8s scheduler 介绍K8s scheduler的初始化Cobra介绍K8s scheduler中初始化的源代码解析 调度框架介绍 这是官方对于v1.27调度框架的介绍文档&#xff1a;https://v1-27.docs.kubernetes.io/docs/…

使用vue3+ts+vite从零开始搭建bolg(五):layout(持续更新中)

五、layout搭建 5.1静态搭建 在src下创建如图文件夹 这里用logo举例&#xff0c;在scripts里export <script lang"ts">export default {name: Logo,}</script> 然后在layout里引入 //引入左侧菜单顶部用户信息 import Logo from ./logo/index.vue 接…

Spring AOP(概念,使用)

目录 Spring AOPAOP是什么什么是Spring AOPAOP实际开发流程1. 引入依赖2. 编写AOP程序 Spring AOP详解Spring AOP中的核心概念Spring AOP的通知类型六种类型PointCutOrder(切面优先级) Spring AOP AOP是什么 Aspect Oriented Programminig(面向切面编程)切面指的是某一类特定…

“Linux”目录结构and配置网络

了解完命令格式和vi、vim编辑器后&#xff0c;我们来认识一下目录的结构&#xff1a; 一、目录 &#xff08;1&#xff09;目录的特点 windows特点&#xff1a; Windows中有C、D、E盘&#xff0c;每个都是一个根系统 Linux特点&#xff1a; linux中只有一个根&#xff08;单…

C++auto关键字、范围for循环

一、auto关键字 1.1auto简介 在早期C/C中auto的含义是&#xff1a;使用auto修饰的变量&#xff0c;是具有自动存储器的局部变量。 C11中&#xff0c;标准委员会赋予了auto全新的含义即&#xff1a;auto不再是一个存储类型指示符&#xff0c;而是作为一个新的类型指示符来指示编…

记录用python转换headers

转换前 转换后效果 代码如下。注意需要在控制台切换到content.txt所在位置&#xff0c;不然运行代码会报file not found错误 # 假设txt文件内容如下 txt open(content.txt).read()# 使用splitlines()方法将txt内容分割为行&#xff0c;然后使用json.loads()方法将每一行转换为…

每日两题 / 437. 路径总和 III 105. 从前序与中序遍历序列构造二叉树(LeetCode热题100)

437. 路径总和 III - 力扣&#xff08;LeetCode&#xff09; 前序遍历时&#xff0c;维护当前路径&#xff08;根节点开始&#xff09;的路径和&#xff0c;同时记录路径上每个节点的路径和 假设当前路径和为cur&#xff0c;那么ans 路径和(cur - target)的出现次数 /*** D…

【吊打面试官系列】Java高并发篇 - 多线程的价值?

大家好&#xff0c;我是锋哥。今天分享关于 【多线程的价值&#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; 多线程的价值&#xff1f; 1、发挥多核 CPU 的优势 多线程&#xff0c;可以真正发挥出多核 CPU 的优势来&#xff0c;达到充分利用 CPU 的目的&#…