java 关于primitive数组 内存问题

本文详细解析了Java中数组在堆内存与栈内存中的分配方式,包括数组初始化的区别及内存分配原理,帮助开发者理解Java内存管理机制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.我们知道java里new 出来的都是存放在堆里的。
问题是我们如下情况:
int [] arr={1,2,3,4},
我查看了一下深入java 虚拟机 测试了一下就没有new 直接 int[] arr={1,2,3,4}也是存放在堆里的。
下面是深入java虚拟机里堆数组相关字节码的定义,看清楚了 说明每句最后一句话将新数组的对象引用压入栈。(是对象引用)


而我编写了两个小例子生成字节码:
例子1:

public class TestArray {
	 public static void main(String[] args){  
		 int [] arr1=new int[3];
	 }
}
生成的字节码文件如下:
public class TestArray extends java.lang.Object{
public TestArray();
  Code:
   0:	aload_0
   1:	invokespecial	#8; //Method java/lang/Object."<init>":()V
   4:	return

public static void main(java.lang.String[]);
  Code:
   0:	iconst_3
   1:	newarray int
   3:	astore_1
   4:	return
}

例子2:
public class TestArray {
	 public static void main(String[] args){  
		 int [] arr1={1,2,3};
	 }
}
生成的字节码文件如下:
public class TestArray extends java.lang.Object{
public TestArray();
  Code:
   0:	aload_0
   1:	invokespecial	#8; //Method java/lang/Object."<init>":()V
   4:	return

public static void main(java.lang.String[]);
  Code:
   0:	iconst_3
   1:	newarray int
   3:	dup
   4:	iconst_0
   5:	iconst_1
   6:	iastore //将int value 和索引index 弹出栈,赋值arr[index]=value;
   7:	dup
   8:	iconst_1
   9:	iconst_2
   10:	iastore
   11:	dup
   12:	iconst_2
   13:	iconst_3
   14:	iastore
   15:	astore_1 //我数组对象引用赋值给变量arr1
   16:	return
}
看到了没有都是通过newarray int 生成字节码文件。
所以不管哪种方式 int [] arr1={1,2,3}; 还是int [] arr1=new int[3];数组都是在堆上,引用在栈上。
2.另外java中针对数组和对象使用不同操作码(专门为对象设计了另外的操作码)。如对象采用如下:


3. 数组平时要注意地方:
String[] str = {"1","2","3"}与String[] str = new String[]{"1","2","3"}在内存里有什么区别?
这里的区别仅仅是代码书写上的: 
String[] str = {"1","2","3"}; 这种形式叫数组初始化式(Array Initializer),只能用在声明同时赋值的情况下。 
而 String[] str = new String[]{"1","2","3"} 是一般形式的赋值,=号的右边叫数组字面量(Array Literal),数组字面量可以用在任何需要一个数组的地方(类型兼容的情况下)。如: 
String[] str = {"1","2","3"}; // 正确的 
String[] str = new String[]{"1","2","3"} // 也是正确的 
而 
String[] str; 
str = {"1","2","3"}; // 编译错误 
因为数组初始化式只能用于声明同时赋值的情况下。 
改为: 
String[] str; 
str = new String[] {"1","2","3"}; // 正确了 
又如: 
void f(String[] str) { 

f({"1","2","3"}); // 编译错误 
正确的应该是: 
f(new String[] {"1","2","3"}); 

4.对于内存方面来说,总的一条java 里数组都是存放在堆里的,而C/C++中的数组是可以在栈空间中分配。

### 如何在 Java 中计算两个数组的交集 为了实现这一功能,在 Java 中可以采用多种方法来找出两个整数数组 `nums1` 和 `nums2` 的交集。下面介绍一种利用哈希表的方法,这种方法能够有效地处理当其中一个数组无法完全加载到内存中的情况[^3]。 #### 方法一:使用哈希映射 (HashMap) 此方式适用于其中有一个数组可能过大而不能全部载入内存的情况。具体做法如下: 先遍历较小的那个数组并将它的每一个元素作为键存放到 HashMap 中,对应的值设为该元素出现次数;接着逐一遍历另一个数组的同时检查当前元素是否存在于已构建好的 map 当中——如果存在,则说明这是共同拥有的元素之一,并将其加入结果集中,同时减少相应 key 对应 value 值的数量直至变为零为止。 ```java import java.util.*; public class Solution { public int[] intersect(int[] nums1, int[] nums2) { Map<Integer, Integer> countMap = new HashMap<>(); List<Integer> resultList = new ArrayList<>(); // Count each element in first array. for (int num : nums1){ countMap.put(num, countMap.getOrDefault(num, 0)+1); } // Check second array against counted elements from first one. for (int num : nums2){ if(countMap.containsKey(num)){ resultList.add(num); // Decrease counter or remove entry when exhausted. int currentCount = countMap.get(num)-1; if(currentCount==0){ countMap.remove(num); }else{ countMap.put(num,currentCount); } } } // Convert list back to primitive type array before returning. return resultList.stream().mapToInt(i->i).toArray(); } } ``` 这段代码实现了上述逻辑并返回了一个新的只包含输入两数组共有部分的新数组。注意这里的结果顺序并不固定,因为集合内部默认是没有顺序概念的。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值