做者:林冠宏 / 指尖下的幽灵java
掘金:https://juejin.im/user/587f0dfe128fe100570ce2d8git
博客:http://www.cnblogs.com/linguanh/github
GitHub : https://github.com/af913337456/面试
联系方式 / Contact:913337456@qq.com算法
了解这些术语:数组
对象引用
与 对象
的区别精通java
,写 熟练
首先明确一点,System.arraycopy 操做的是数组,效果是深复制。 是否是以为怎么和你印象的中不同?
重点来了,对于对象数组,例如: User[],这种数组,有一个注意点
,这个点就是:对于数组内的对象是浅拷贝。
ide
一句话:函数
System.arraycopy 对于数组是深拷贝,对于数组内的对象是浅拷贝。由于操做的传入参数是数组,那么回归本意,效果是深复制。
上面的含义必定要区分清楚! 由于如今网上不少观点是混淆,乱JB写的 。this
来看个例子,下面的代码能够本身去运行验证。已经充分验证了上面个人观点。code
public class Test { public static void main(String[] args) { User[] users=new User[]{ new User("111"),new User("222"),new User("333") };//初始化对象数组 User[] target=new User[users.length];//新建一个目标对象数组 System.arraycopy(users, 0, target, 0, users.length); // 复制 System.out.println("数组地址是否同样:"+(users == target?"浅复制":"深复制")); System.out.println("数组内对象地址是否同样:"+(users[0] == target[0]?"浅复制":"深复制")); target[0].setEmail("444"); System.out.println("修改后输出 users[0] ,是否和 target[0]同样是444,users[0]:"+users[0].getEmail()); users[0] = null; // -----------------① System.out.println("遍历 users"); for (User user : users){ System.out.println(user); } System.out.println("遍历 target"); for (User user : target){ System.out.println(user); } users = null; if(target == null){ System.out.println("users = null 后是否 target 也变成了 null,是则证实是浅复制"); }else{ System.out.println("users = null 后是否 target 不受任何影响,是则再次证实数组是深复制"); } } } class User{ private String email; public User(String email) { this.email = email; } public String getEmail() { return email;} public void setEmail(String email) { this.email = email; } @Override public String toString() { return "User [email=" + email+ "]"; } }
输出的结果以下:
数组地址是否同样:深复制 数组内对象地址是否同样:浅复制 修改后输出 users[0],是否和 target[0]同样是444,users[0]:444 遍历 users null User [email=222] User [email=333] 遍历 target User [email=444] User [email=222] User [email=333] users = null 后是否 target 不受任何影响,是则再次证实数组是深复制
上面个人例子还留有一个经典的面试点,既是标号 ① 对应的行:
users[0] = null; // -----------------①
上面咱们谈到了,数组内的对象是浅复制,那么在上面这行我把 users[0] 下标为0的对象弄为 null 后。后面再遍历输出:
System.out.println("遍历 users"); for (User user : users){ System.out.println(user); } System.out.println("遍历 target"); for (User user : target){ System.out.println(user); }
明显地,咱们能够容易知道,user[0] 此时输出的确定是 null。那么 target[0] 呢?浅拷贝的话,target[0] 必然在内存地址和值上面全等于 users[0]
。
可是从 System.out.println("遍历 target");
的结果来看。却发现 target 输出的是:
遍历 target User [email=444] // 第一个不是 null User [email=222] User [email=333]
这是为何呢?其实这是最为基础的: 对象引用与对象的区别
,一名合格,仅仅是合格的 Java 语言使用者,这个得知道。下面咱们来谈谈它。
有一个类:
public class Demo { //默认构造方法 public Demo{ } }
咱们用它建立个对象
Demo fuck = new Demo();
这一条语句,其实包括了四个动做:
堆空间
里建立一个Demo对象。栈空间中
。也就是用来指向Demo对象的对象引用。fuck
Demo fuck;//一个对象引用 fuck = new Demo();//一个对象引用指向一个对象
一个对象能够被多个对象引用同时引用。它们操做的始终是同一个。
Demo fuck,fuck2;//建立多个对象引用 fuck = new Demo(); fuck2 = fuck;
好了,回答以前的坑, users[0] = new User("111")
,users[0] = null
,target[0] = users[0]
,users[0] = null 只是把栈中的 users[0] 对象引用弄为了 null,对象 new User("111") 与它无关,天然没影响,target[0] 是另一个对象引用,也是指向了 new User("111")。
根据 大 Jvm 的 内存回收算法之根搜索,引用链存在、强引用、when 当前应用内存不够了,强制抛出 OOM。那么当 target[0] = null
,new User("111")
对应的这块内存就会进入被回收的队列中,“死去”。
最后这段是否是有点看不懂 ?那证实你要继续努力了。