【JAVA入门】Day03 - 数组

news2024/11/25 0:44:56

【JAVA入门】Day03 - 数组


文章目录

  • 【JAVA入门】Day03 - 数组
  • 一、数组的概念
  • 二、数组的定义
    • 2.1 数组的静态初始化
    • 2.2 数组的地址值
    • 2.3 数组元素的访问
    • 2.4 数组遍历
    • 2.5 数组的动态初始化
    • 2.6 数组的常见操作
    • 2.7 数组的内存分配
      • 2.7.1 Java内存分配
      • 2.7.2 数组的内存图


一、数组的概念

        数组就是一种容器,可以用来存储同种数据类型的多个值。
        数组在存储数据时,存在隐式转换现象。例如:int 类型数组可以存储(byte short int)这几种类型,因为 int 相比另外二者更高级,在存储时会自动转换为 int 型。同样地,如果把 int 型存入 double 数组,一样会转换为 double 类型存储,double 类型可以存储(byte short int long float double)。

二、数组的定义

        数组有两种定义方法:

  • int [ ] array
  • int array[ ]

int是数据类型,[ ]表示是数组,array是数组名。

2.1 数组的静态初始化

在内存中,为数组提前开辟好空间,并将数据存入容器中。

  • 完整格式:
    数据类型[ ] 数组名 = new 数据类型[ ] { 元素1, 元素2, 元素3…};
  • 例:
int[] array = new int[]{ 11, 22, 33 };
  • 简化格式:
    数据类型[ ] 数组名 = { 元素1, 元素2, 元素3…};
  • 例:
int[] array = { 11, 22, 33 };

2.2 数组的地址值

int[] arr = {1,2,3,4,5};
System.out.println(arr);  //[I@6d03e736

        这里直接输出,用数组名作参数,返回的是数组的地址值,表示数组在内存中的位置。[ 表示当前是一个数组,I 表示这是一个 int 型数组,@是一个间隔符号,6d03e736 是数组真正的地址值(十六进制)。

2.3 数组元素的访问

        利用索引来访问数组某个元素。
        格式:数组名[索引];
例:

int[] arr = {1, 2, 3, 4, 5};
int number = arr[0];
System.out.println(number); //1
System.out.println(arr[1]); //2

        利用索引也可以把数据存储到数组中:

arr[0] = 100;
System.out.println(arr[0]); //100

2.4 数组遍历

        数组遍历,就是把数组中所有的内容都取出来。

//1.定义数组
int[] arr = {1, 2, 3, 4, 5};
//2.获取数组里面所有的元素
for(i = 0; i < 5; i++;) {
	System.out.println(arr[i]); //1 2 3 4 5
}
//3.如果不知道数组长度,可以用length
for(i = 0; i < arr.length; i++) {
	System.out.println(arr[i]); //1 2 3 4 5
}

【例1】遍历数组并求和

//1.定义一个数组,并添加数据1,2,3,4,5
int[] arr = {1, 2, 3, 4, 5};
//2.遍历数组得到每一个数据,累加求和
int sum = 0;  //求和变量
for(int i = 0; i < arr.length; i++) {
	sum = sum + arr[i];
}

【例2】统计数组元素个数

//统计数组中能被3整除的数组有多少个
int[] arr = {1,2,3,4,5,6,7,8,9,10};
count = 0;
for(int i = 0; i < arr.length; i++) {
	if(arr[i] % 3 == 0){
		System.out.println(arr[i]);
		count++;
	}
}
System.out.println("数组中能被3整除的数字有" + count + "个");

【例3】变化数组
定义一个数组,遍历之,如果是奇数,当前数字乘2,如果是偶数,当前数字变为二分之一。

int[] arr = {1,2,3,4,5,6,7,8,9,10};
for(int i = 0; i < arr.length; i++) {
	if(arr[i] % 2 == 0) {
		arr[i] = arr[i] / 2;
	}
	else {
		arr[i] = arr[i] * 2;
	}
	System.out.println(arr[i]);
}

2.5 数组的动态初始化

        动态初始化时,只指定数组长度,由系统为数组分配初始值。
格式:数据类型[ ] 数组名 = new 数据类型[数组长度];
例:

int[] arr = new int[3];

        动态初始化后,可以直接给新增的空间位置赋值,即填入元素。

//初始化一个字符串数组
String[] arr = new String[3];
arr[0] = "xiaozhan";
arr[1] = "dingzhen";

        获取元素时,除了已赋值的元素,还可以获取未赋值的空位置,这个位置上存储的值应为默认的初始值。

System.out.println(arr[0]);  //xiaozhan
System.out.println(arr[1]);  //dingzhen
System.out.println(arr[2]);  //null(初始值)

        不同数据类型的数组初始化后有不同的初始值:

//整数类型:0
//小数类型:0.0
//字符类型:'/u0000' 即空格
//布尔类型:false
//引用类型:null (String也是引用类型)

2.6 数组的常见操作

【例1】求最值。
需求:已知数组元素为 {33,5,22,44,55}

int[] arr = {33,5,22,44,55};
int max = arr[0];
for(int i = 0; i < arr.length; i++) {
	if(arr[i] >= max) {
    	max = arr[i];
    }
}
System.out.println(max);
  • 注意,max 的初始值一定要是数组中存在的值,这样才不容易出错。

【例2】遍历数组求和。
需求:生成10个1~100之间的随机数存入数组。然后求出数组的和、平均数,统计多少个元素比平均数小,最后打印数组。

import java.util.Random;

int[] arr = new int[10];
        Random r = new Random();

        for(int i = 0; i < arr.length; i++) {
            //每循环一次,就生成一个新的随机数
            int number = r.nextInt(100) + 1;
            //把生成的随机数添加到数组当中
            arr[i] = number;
        }

        //遍历数组求和
        int sum = 0;
        for(int i = 0; i < arr.length; i++) {
            sum += arr[i];
        }
        System.out.println("数组中所有数的和为:"+ sum);

        //求平均值
        int avg = sum / arr.length;
        System.out.println("数组中所有数的平均值为:"+ avg);

        //统计有多少个数据比平均值小
        int count = 0;
        for(int i = 0; i < arr.length; i++) {
            if(arr[i] < avg) {
                count++;
            }
        }
        System.out.println("有"+count+"个数比平均值小");

        //遍历打印数组
        for(int i = 0; i < arr.length; i++)
            System.out.print(arr[i] + " ");

【例3】交换数组中的数据
需求:定义一个数组,存入1,2,3,4,5.按照要求交换索引对应的元素。
交换前:1,2,3,4,5
交换后,5,2,3,4,1

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

//利用第三方变量进行交换
int temp = arr[0];
arr[0] = arr[4];
arr[4] = temp;

for(int i = 0; i < arr.length; i++) {
	System.out.print(arr[i]+" ");
}

需求变式:将数组倒置。

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

for(int i = 0,j = arr.length - 1 ; i < j; i++,j--) {
	temp = arr[i];
	arr[i] = arr[j];
	arr[j] = temp;
}
for(int i = 0; i < arr.length; i++) {
	System.out.print(arr[i] + " ");
}

【例4】打乱数组顺序。
需求:给定一个数组{1,2,3,4,5},要求打乱这五个数字的顺序。
思路:可以利用 Random 把索引打乱,数组元素自然也会打乱。

int[] arr = {1,2,3,4,5};
//索引范围:0 1 2 3 4
Random r = new Random();
//从0索引开始,打乱数据的顺序
for(int i = 0; i < arr.length; i++) {
	int randomIndex = r.nextInt(arr.length);//随机0~4,括号里要填5,这里arr.length就是5,这样写更泛用
	int temp = arr[i];
	arr[i] = arr[randomIndex];
	arr[randomIndex] = temp;
}
for(int i = 0; i < arr.length; i++) {
	System.out.print(arr[i] + " ");
}

2.7 数组的内存分配

2.7.1 Java内存分配

Java的虚拟机(JVM)在运行时也会占用一部分计算机的内存空间,而 JVM 有更精细的规划:
在这里插入图片描述
注:从 JDK8 开始,“方法区”被移除,改换“元空间”,把原方法区多种功能进行拆分,一部分放到了中,一部分放入了元空间中。

各区域用途

  • 栈                      方法运行时使用的内存,比如 main 方法运行,进入方法栈中执行。
  • 堆                      存储对象或者数组,new 创建的东西,都存储在堆内存。
  • 方法区               存储可以运行的 class 文件。
  • 本地方法栈        JVM 在使用操作系统功能的时候使用。
  • 寄存器               给CPU使用。

2.7.2 数组的内存图

数组在内存中的存储同时用到了栈空间和堆空间,具体存储方式如下:

public static void main(String[] args) {
	int[] arr = new int[2];
	
	sout(arr);
	sout(arr[0]);
	sout(arr[1]);
	
	arr[0] = 11;
	arr[1] = 22;
	sout(arr[0]); 

	sout("-----------------");
	int[] arr2 = {33, 44, 55};
	sout(arr2);
	sout(arr2[0]);
	sout(arr2[1]);
	sout(arr2[2]);
}

在这里插入图片描述
        如图所示,创建一个二元数组,在堆中开辟了一处空间,存放了数组长度、数组元素和索引,然后将这块堆空间的地址值记录下来放入栈中,代表数组 arr 的地址。通过寻访栈内存中的数组地址值,可以找到堆内存中的这块空间,从而找到数组的数据存储所在。
        下图中我们又创建了一个数组 arr2,这次直接将其初始化,同样地也在堆空间里开辟了一块空间存储,然后将这块空间的地址值存储在栈空间。在访问时,先访问栈空间,再根据栈空间的地址值访问堆空间,找到数组内容。
在这里插入图片描述
在这里插入图片描述
        除了以上那种基本的存储方法,还存在两个数组指向同一个空间的内存这样的情况。

public static void main(String[] args) {
	int[] arr1 = {11, 22};
	int[] arr2 = arr1;

	sout(arr1[0]);
	sout(arr2[0]);
	
	arr2[0] = 33;

	sout(arr1[0]);
	sout(arr2[0]);
}

        根据上方的代码,显然 arr1 和 arr2 是同一个数组。在创建 arr1 数组后,仍是开辟了一块堆空间和一块栈空间。
在这里插入图片描述
        而在创建 arr2 数组后,指向了 arr1 ,它们存储的是同一个的地址值。
在这里插入图片描述
        因此它们指向的一定是同一块堆空间,这样就只开辟了一块堆空间。在寻址时,找的自然也是同一块堆空间。因此,我们通过 arr2[0] = 33;修改了堆空间索引为 0 的位置上的值:
在这里插入图片描述
此时 arr1[0] 位置上的值,也会发生改变。因此:

sout(arr1[0]);
sout(arr2[0]);

        输出的值应为一样的,都是33。这是 JAVA 内存存储方式导致的。在使用时,只需要默认认为 arr1 和 arr2 是同一个数组即可,修改 arr1 ,arr2 也会发生变化。

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

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

相关文章

SAPUI5基础知识1 - 概览,库,支持工具,自学教程

1. SAPUI5 概览 1.1 SAPUI5 SAPUI5是一种用于构建企业级Web应用程序的开发框架。它是由SAP开发的&#xff0c;基于HTML5、CSS3和JavaScript技术。 SAPUI5提供了一套丰富的UI控件和工具&#xff0c;使开发人员能够快速构建现代化、可扩展和可定制的应用程序。 它还提供了数据…

cmake进阶:变量的作用域说明一(从函数作用域方面)

一. 简介 如同 C 语言一样&#xff0c;在 cmake 中&#xff0c;变量也有作用域的概念&#xff0c;本文我们就来聊一聊关于 cmake 中变量作用域的问题。 接下来从三个方面进行介绍&#xff1a;函数作用域、目录作用域以及全局作用域。 二. 函数作用域 我把这个作用域叫做函数…

文献速递:深度学习医学影像心脏疾病检测与诊断---利用深度学习进行动态心脏PET的自动帧间患者运动校正

Title 题目 Automatic Inter-frame Patient Motion Correction for Dynamic Cardiac PET Using Deep Learning 利用深度学习进行动态心脏PET的自动帧间患者运动校正 01 文献速递介绍 OSITRON正电子发射断层扫描&#xff08;PET&#xff09;心肌灌注成像已被证明相较于其他…

LeetCode-741. 摘樱桃【数组 动态规划 矩阵】

LeetCode-741. 摘樱桃【数组 动态规划 矩阵】 题目描述&#xff1a;解题思路一&#xff1a;动态规划&#xff0c;定推初遍举。解题思路二&#xff1a;倒序循环解题思路三&#xff1a;0 题目描述&#xff1a; 给你一个 n x n 的网格 grid &#xff0c;代表一块樱桃地&#xff0…

Android硬件加速hardwareAccelerated支持/不支持的绘图接口

Android硬件加速hardwareAccelerated支持/不支持的绘图接口 Android硬件加速也即在Androidmanifest.xml配置开启GPU渲染&#xff1a; <application android:hardwareAccelerated"true" > 配置后&#xff0c;Android将启用GPU渲染&#xff0c;在trace里面看会…

2023 年全国职业院校技能大赛(高职组)“云计算应用”赛项赛卷 B(容器云)

#需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包及镜像&#xff09;或有问题的&#xff0c;可私聊博主&#xff01;&#xff01;&#xff01; #需要资源&#xff08;软件包…

【一起深度学习——kaggle叶子分类】

kaggle 叶子分类 目的&#xff1a;将叶子进行分类。实现步骤&#xff1a;1、数据处理&#xff1a;2、加载数据3、 定义残差块4、定义Resnet模型。5、定义训练以及评估函数&#xff1a;6、开始训练&#xff1a;7、输出结果&#xff1a; 目的&#xff1a;将叶子进行分类。 实现步…

观测与预测差值自动变化系统噪声Q的自适应UKF(AUKF_Q)MATLAB编写

简述 基于三维模型的UKF&#xff0c;设计一段时间的输入状态误差较大&#xff0c;此时通过对比预测的状态值与观测值的残差&#xff0c;在相应的情况下自适应扩大系统方差Q&#xff0c;构成自适应无迹卡尔曼滤波&#xff08;AUKF&#xff09;&#xff0c;与传统的UKF相比&…

【Qt】按钮类控件

文章目录 1 :peach:Push Button:peach:2 :peach:Radio Buttion:peach:3 :peach:Check Box:peach:4 :peach:Tool Button:peach: 1 &#x1f351;Push Button&#x1f351; 使⽤ QPushButton 表⽰⼀个按钮&#xff0c;这也是当前我们最熟悉的⼀个控件了&#xff0c;QPushButton …

Ubuntu添加非root用户到Docker用户组

前言 首先平常公司的Linux生产环境为了防止误操作导致灾难性问题&#xff0c;一般都不会给我们开发开放root管理员的账号权限。所以平常在Ubuntu的普通用户登录的时候&#xff0c;要操作Dcoker一般都需要带上sudo来提升命令执行权限。为了解决这一问题&#xff0c;我们只需要将…

正点原子[第二期]Linux之ARM(MX6U)裸机篇学习笔记-13-按键实验

前言&#xff1a; 本文是根据哔哩哔哩网站上“正点原子[第二期]Linux之ARM&#xff08;MX6U&#xff09;裸机篇”视频的学习笔记&#xff0c;在这里会记录下正点原子 I.MX6ULL 开发板的配套视频教程所作的实验和学习笔记内容。本文大量引用了正点原子教学视频和链接中的内容。…

Axure中继器介绍以及案例分享

中继器是 Axure 中一个比较高阶的应用&#xff0c;它可以让我们在纯静态网页中模拟出类似带有后台数据交互的增删改查的效果。 一、中继器的基本使用方法&#xff1a; 整体流程分为三个步骤 ☆创建中继器 我们先在 Axured画布中拖入一个中继器元件 双击中继器后的效果 打开之…

java高并发实战<1>

我们一个请求--->tomcat--->db 我们只需要把我们的应用部署在tomcat中&#xff0c; 就可以了 这就是你单体的感念&#xff0c;单机结构你只用一个服务器就完成了你项目的部署单点问题一旦这台机器挂了,用户就没有办法用你这个服务,单机能力有限 随着你用户量增长的过程…

04.添加自定义监控项

添加自定义监控项 监控项就是监控每一个指标 1.命令行&#xff0c;手动取值 [rootyunlong66 ~]# iostat |awk $1 ~/sda/ sda 5.89 36.10 122.71 557910 1896585 [rootyunlong66 ~]# iostat |awk $1 ~/sda/{print $2} 5.892.修改zabbix-age…

OpenNJet下载安装及入门实战教程

一、什么是OpenNJet OpenNJet是一款开放原子开源基金会孵化及运营的开源项目。OpenNJet采用C语言实现。是一款高性能、轻量级的WEB应用及代理软件。    OpenNJet 应用引擎是高性能、轻量级的WEB应用与代理软件。作为云原生服务网格的数据平面&#xff0c;NJet具备动态配置加载…

【Git】回滚旧提交版本且不影响最新提交版本

【Git】回滚旧提交版本且不影响最新提交版本 一、场景假设 远程仓库origin中有一个分支main&#xff0c;有4次提交记录&#xff1a;v1、v2、v3、v4。 二、需求 需要回滚旧提交版本&#xff0c;但不影响已有的所有提交版本&#xff08;即不影响最新提交版本&#xff09;&…

k8s保持pod健康

存活探针 Kubemetes 可以通过存活探针 (liveness probe) 检查容器是否还在运行。可以为 pod 中的每个容器单独指定存活探针。如果探测失败&#xff0c;Kubemetes 将定期执行探针并重新启动容器。 Kubemetes 有以下三种探测容器的机制&#xff1a; HTTP GET 探针对容器的 IP 地…

深入探索归并排序算法:分而治之的排序艺术

在计算机科学领域&#xff0c;排序算法是一项基础且重要的技术&#xff0c;归并排序作为一种经典的分治算法&#xff0c;以其稳定性和高效性而闻名。本文将带您深入探索归并排序算法的原理、实现方法以及应用场景&#xff0c;揭示这一排序艺术背后的精髓。 **归并排序算法简介…

【管理篇】管理三步曲:管理规划(一)

目录标题 管理到底都要做哪些事呢如何开始带团队&#xff1f; 职能&#xff1a;如何界定团队是干什么的&#xff1f;目标&#xff1a;如何为团队设定合理的目标规划资源&#xff1a;需要申请哪些资源&#xff08;1&#xff09;你是否了解资源的丰富性&#xff1f;&#xff08;2…

判断dll/lib是32/64位、查看lib是导入库/静态库的方法 、查看dll包含的符合、lib包含的函数

一、判断dll/lib是32/64位 原文链接&#xff1a;https://www.cnblogs.com/bandaoyu/p/16752602.html 1. 简便方法&#xff1a; 直接用记事本或者notepad(或txt文本)打开exe文件&#xff08;dll文件&#xff09;&#xff0c;会有很多乱码&#xff0c;不要头疼&#xff0c;接下…