java面试题26 java语言的下面几种数组复制方法中,哪个效率最高?

java面试题26 java语言的下面几种数组复制方法中,哪个效率最高?

 

A for 循环逐一复制
B System.arraycopy
C Array.copyOf
D 使用clone方法

效率:System.arraycopy > clone > Arrays.copyOf > for循环


1、System.arraycopy的用法:

   public static void arraycopy(Object src,int srcPos,Object dest,int destPos,int length)

参数:src - 源数组。srcPos - 源数组中的起始位置。dest - 目标数组。destPos - 目标数据中的起始位置。length - 要复制的数组元素的数量

应用实例:

  public class Main{public static void main(String[] args) {int[] a1={1,2,3,4,5,6};int[] a2={11,12,13,14,15,16};System.arraycopy(a1, 2, a2, 3, 2);System.out.print("copy后结果:");for(int i=0;i<a2.length;i++){System.out.print(a2[i]+" ");    }}}

运行结果:

2、clone 的用法:

java.lang.Object类的clone()方法为protected类型,不可直接调用,需要先对要克隆的类进行下列操作:

首先被克隆的类实现Cloneable接口;然后在该类中覆盖clone()方法,并且在该clone()方法中调用super.clone();这样,super.clone()便可以调用java.lang.Object类的clone()方法。

应用实例:

  //被克隆的类要实现Cloneable接口class Cat implements Cloneable{private String name;private int age;public Cat(String name,int age){this.name=name;this.age=age;}//重写clone()方法protected Object clone()throws CloneNotSupportedException{  return super.clone() ;  }}public class Clone {public static void main(String[] args) throws CloneNotSupportedException {Cat cat1=new Cat("xiaohua",3);System.out.println(cat1);//调用clone方法Cat cat2=(Cat)cat1.clone();System.out.println(cat2);}}

3、复制引用和复制对象的区别

复制引用:是指将某个对象的地址复制,所以复制后的对象副本的地址和源对象相同,这样,当改变副本的某个值后,源对象值也被改变;

复制对象:是将源对象整个复制,对象副本和源对象的地址并不相同,当改变副本的某个值后,源对象值不会改变;

    Cat cat1=new Cat("xiaohua",3);//源对象System.out.println("源对象地址"+cat1);//调用clone方法,复制对象Cat cat2=(Cat)cat1.clone();Cat cat3=(Cat)cat1;//复制引用System.out.println("复制对象地址:"+cat2);System.out.println("复制引用地址:"+cat3);

输出结果:


可以看出,复制引用的对象和源对象地址相同,复制对象和源对象地址不同

4、Arrays.copyOf 的用法:

Arrays.copyOf有十种重载方法,复制指定的数组,返回原数组的副本。具体可以查看jdk api

背后原理探究

首先要申明的是这4种方法中的前3种是没有本质区别的,对象都是浅复制(复制地址),而普通类型都是深复制(复制值),简单来说
浅复制:复制地址,相当于复制了房子(内存数据)的钥匙,一改全改
深复制:复制值,通俗意义上的拷贝,相当于建了一个一模一样的房子(内存的数据)
 接下来说说3种方式:
  System.arraycopy()

    源码如下:
    

    @HotSpotIntrinsicCandidatepublic static native void arraycopy(Object src,  int  srcPos,Object dest,int destPos,int length);

可以看到这是一个native(本地)方法,也就是说是用C++写的,所以其效率比非native方法更高

但更要注意的是上面的注解@HotSpotIntrinsicCandidate,我们来看官方对这个注解的解释

 * The {@code @HotSpotIntrinsicCandidate} annotation is specific to the* HotSpot Virtual Machine. It indicates that an annotated method* may be (but is not guaranteed to be) intrinsified by the HotSpot VM. A method* is intrinsified if the HotSpot VM replaces the annotated method with hand-written* assembly and/or hand-written compiler IR -- a compiler intrinsic -- to improve* performance. The {@code @HotSpotIntrinsicCandidate} annotation is internal to the* Java libraries and is therefore not supposed to have any relevance for application* code.

注意红字粗体所说,为了提升性能,在JVM里对该注解的进行了手写,这里要提一个叫JNI(Java Native Interface)的东西,普通的native方法通俗的讲就是编译后还要通过JNI再次编译成.cpp文件才能执行.而有 @HotSpotIntrinsicCandidate这个注解的方法在JVM里就是用.cpp文件写好的,所以就跳过了JNI阶段,所以速度就能提升,这也是System.arraycopy()速度冠绝群雄的原因.
Object.clone()

    源码如下:
 

        @HotSpotIntrinsicCandidateprotected native Object clone() throws CloneNotSupportedException;

可以看到它也是native方法,也有@HotSpotIntrinsicCandidate注解,那为啥速度比上面的大哥慢呢?这就要看到官方解释的一句

It indicates that an annotated method may be (but is not guaranteed to be) intrinsified by the HotSpot VM

注意用词:may be (but is not guaranteed to be),是的,clone()方法就是悲催的but,它并没有被手工写在JVM里面,所以它不得不走JNI的路子,所以它就成了2哥。
 Arrays.copyof()

     源码如下
    

        @HotSpotIntrinsicCandidatepublic static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {@SuppressWarnings("unchecked")T[] copy = ((Object)newType == (Object)Object[].class)? (T[]) new Object[newLength]: (T[]) Array.newInstance(newType.getComponentType(), newLength);System.arraycopy(original, 0, copy, 0,Math.min(original.length, newLength));return copy;}

3弟也有注解 @HotSpotIntrinsicCandidate,但它甚至不是native方法,所以这个注解也就是混子,也更印证了2哥的待遇,

而且可以很明显的看到里面本质是调用了大哥  System.arraycopy()来实现的,所以效率垫底也是妥妥的。
for()

这个就可以退出群聊吧!for()无论是基本数据类型还是引用数据类型统统是深复制,而且其也不是封装的函数,所以退出群聊妥妥的。
 最后上测试代码,有兴趣的可以看看:

    package 问题;
     
   

 import java.util.Arrays;public class CopySpeed implements Cloneable {/*** 数组复制速度 1.System.arraycopy() 2.clone() 3.Arrays.copyof() 4.for()****/int i = 0;String value = "123";/** 基本数据类型测试** */public long[] CopySpeedTest(int[] t) {/** 初始化所有**/int length = t.length;int[] systemCopy = new int[t.length];int[] clone = new int[t.length];int[] arraysCopyOf = new int[t.length];int[] For = new int[t.length];/** 执行复制操作,并统计时间**/long begin = System.currentTimeMillis();System.arraycopy(t, 0, systemCopy, 0, t.length);long end = System.currentTimeMillis();long arraycopyTimes = end - begin;begin = System.currentTimeMillis();clone = t.clone();end = System.currentTimeMillis();long cloneTimes = end - begin;begin = System.currentTimeMillis();arraysCopyOf = Arrays.copyOf(t, t.length);end = System.currentTimeMillis();long arraysCopyOfTimes = end - begin;/** 为了方便查看,这里设定下面程序只执行一次**/if (i == 0) {/** 查看哈希值*/System.out.println("t:\t" + t.hashCode());System.out.println("systemCopy:\t" + systemCopy.hashCode());System.out.println("clone:\t" + clone.hashCode());System.out.println("arraysCopyOf:\t" + arraysCopyOf.hashCode());i++;}/** 运行时间统计**/long[] times = new long[3];times[0] = arraycopyTimes;times[1] = cloneTimes;times[2] = arraysCopyOfTimes;return times;}/*** 引用数据类型结果** */public long[] CopySpeedTest(CopySpeed[] t) {/** 初始化所有**/int length = t.length;CopySpeed[] systemCopy = new CopySpeed[length];CopySpeed[] clone = new CopySpeed[length];CopySpeed[] arraysCopyOf = new CopySpeed[length];/** 执行复制操作,并统计时间**/long begin = System.currentTimeMillis();System.arraycopy(t, 0, systemCopy, 0, t.length);long end = System.currentTimeMillis();long arraycopyTimes = end - begin;begin = System.currentTimeMillis();clone = t.clone();end = System.currentTimeMillis();long cloneTimes = end - begin;begin = System.currentTimeMillis();arraysCopyOf = Arrays.copyOf(t, t.length);end = System.currentTimeMillis();long arraysCopyOfTimes = end - begin;/** 为了方便查看,这里设定下面程序只执行一次**/if (i == 1) {/** 查看哈希值*/System.out.println("t[0]:\t" + t[0].hashCode());System.out.println("systemCopy[0]:\t" + systemCopy[0].hashCode());System.out.println("clone[0]:\t" + clone[0].hashCode());System.out.println("arraysCopyOf[0]:\t" + arraysCopyOf[0].hashCode());/** 修改新对象的值来判断是浅复制还是深复制**/System.out.println("深浅复制判断,以value属性判断");System.out.println("修改前t[0]:\t" + t[0].value);System.out.println("修改前systemCopy[0]:\t" + systemCopy[0].value);System.out.println("修改前clone[0]:\t" + clone[0].value);System.out.println("修改前arraysCopyOf[0]:\t" + arraysCopyOf[0].value);System.out.println("---------------------------");t[0].value = "t";systemCopy[0].value = "systemCopy";clone[0].value = "clone";arraysCopyOf[0].value = "arraysCopyOf";System.out.println("修改后t[0]:\t" + t[0].value);System.out.println("修改后systemCopy[0]:\t" + systemCopy[0].value);System.out.println("修改后clone[0]:\t" + clone[0].value);System.out.println("修改后arraysCopyOf[0]:\t" + arraysCopyOf[0].value);i++;}/** 运行时间统计*/long[] times = new long[3];times[0] = arraycopyTimes;times[1] = cloneTimes;times[2] = arraysCopyOfTimes;return times;}public static void main(String args[]) {CopySpeed speed = new CopySpeed();System.out.println("基本类型");long[] baseTimes = new long[] { 0, 0, 0 };int[] baseArrays = new int[10000000];for (int i = 0; i < baseArrays.length; i++) {baseArrays[i] = i;}for (int i = 0; i < 20; i++) {// System.out.print(i+"次");long[] temp = speed.CopySpeedTest(baseArrays);baseTimes[0] += temp[0];baseTimes[1] += temp[2];baseTimes[2] += temp[1];// System.out.println();}baseTimes[0] /= 20;baseTimes[1] /= 20;baseTimes[2] /= 20;System.out.println(Arrays.toString(baseTimes));System.out.println("引用类型");long[] ObjectTimes = new long[] { 0, 0, 0 };CopySpeed[] ObjectArrays = new CopySpeed[10000000];for (int i = 0; i < ObjectArrays.length; i++) {ObjectArrays[i] = new CopySpeed();}for (int i = 0; i < 20; i++) {// System.out.print(i+"次");long[] temp = speed.CopySpeedTest(ObjectArrays);ObjectTimes[0] += temp[0];ObjectTimes[1] += temp[1];ObjectTimes[2] += temp[2];// System.out.println();}ObjectTimes[0] /= 20;ObjectTimes[1] /= 20;ObjectTimes[2] /= 20;System.out.println(Arrays.toString(ObjectTimes));}}

结果:

 

   基本类型t:    1552787810systemCopy:    1361960727clone:    739498517arraysCopyOf:    125130493[10, 14, 15]引用类型t[0]:    166239592systemCopy[0]:    166239592clone[0]:    166239592arraysCopyOf[0]:    166239592深浅复制判断,以value属性判断修改前t[0]:    123修改前systemCopy[0]:    123修改前clone[0]:    123修改前arraysCopyOf[0]:    123---------------------------修改后t[0]:    t修改后systemCopy[0]:    t修改后clone[0]:    t修改后arraysCopyOf[0]:    t[22, 31, 55]

结果分析:

    基本类型t:    1552787810systemCopy:    1361960727clone:    739498517arraysCopyOf:    125130493[10, 14, 15]


   
    对于基本数据类型可以看到4个对象的hashcode完全不同,也就验证了前面的深复制的猜想
   
    而10,14,15的运行时间也能证明他们的效率差别,当然因为是基本数据类型,
    而且数据量不算恐怖,所以差距并不大,特别是clone和arraysCopyOf
    可以明显的看到,clone因为JNI的耽误,而Arrays.CopyOf因为System.Copy的加成速度也不赖。
   
    

    引用类型t[0]:    166239592systemCopy[0]:    166239592clone[0]:    166239592arraysCopyOf[0]:    166239592


    对于引用类型,额可以看到4个对象的hashcode值完全相同,说明指向的是同一块内存
     

    深浅复制判断,以value属性判断修改前t[0]:    123修改前systemCopy[0]:    123修改前clone[0]:    123修改前arraysCopyOf[0]:    123---------------------------修改后t[0]:    t修改后systemCopy[0]:    t修改后clone[0]:    t修改后arraysCopyOf[0]:    t[22, 31, 55]


    这里以对象的属性value做测试,可以看到t[0]里面的数据更改后,所有的数据都更改了,
     
    说明3种方法对于对象都是浅复制
答案为B

我是歌谣,若与不合理之处欢迎指出。喜欢敲代码,没事刷刷题。

阅读目录(置顶)(长期更新计算机领域知识)

阅读目录(置顶)(长期更新计算机领域知识)

阅读目录(置顶)(长期科技领域知识)

歌谣带你看java面试题

 

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

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

相关文章

pycharm使用笔记2-远程连接(转)

原文地址:https://blog.csdn.net/jinxiaonian11/article/details/70208920 随着科技的发展&#xff0c;远程办公已经是一种趋势&#xff0c;远程开发能力对于每一个程序员来说都是必不可少的。有时候就算在公司&#xff0c;在进行开发的时候有许多的数据都是储存在服务器上的&a…

第三章 使用属性升级MyBank

1、C#中的访问修饰符&#xff1a; a) public 【公共的】访问不受到任何限制&#xff0c;级别最高。一般用于修饰方法&#xff0c;提供给其他类调用。 b) private 【私有的】只有类的内部可以使用&#xff0c;一般用于修饰字段&#xff0c;保证数据的安全性。 注&#xff1a;如果…

java面试题27 java中下面哪些是Object类的方法()

java面试题27 java中下面哪些是Object类的方法&#xff08;&#xff09; A notify() B notifyAll() C sleep() D wait() 蒙蔽树上蒙蔽果&#xff0c;蒙蔽树下你和我。遇到这种题&#xff0c;我默默的打开了编译工具 Object类中方法&#xff1a; protected Object clone()…

HK-2000 数采仪系统说明之 7.HK7710 DTU 简单配置说明

DTU配置列表: 1 移动服务中心设置(MSC) 2 终端单元设置 3 网络通道设置 4 用户串口设置 移动服务中心设置(MSC) 一般默认&#xff0c;当需要连接的网络是APN网络是需要配置&#xff0c;该项下的APN接入点信息 MSC设置列表: 1 服务代码 2 P…

shiro学习(1):shiro简介

Apache Shiro是Java的一个安全框架。对比另一个安全框架Spring Sercurity&#xff0c;它更简单和灵活。 Shiro可以帮助我们完成&#xff1a;认证、授权、加密、会话管理、Web集成、缓存等。 Apache Shiro特性 Authentication&#xff1a;身份认证/登录&#xff0c;验证用户是…

是要我向垃圾学习吗

之前把程序给改了&#xff0c;造成程序不能通过编译&#xff0c;然后 领导让我不要乱改&#xff0c;程序那么写肯定是有原因的&#xff0c;照着以前的弄。 我心里那叫一个羊驼&#xff0c;我改的是我自己写的那部分结构&#xff0c;我在编写新功能的逻辑的时候&#xff0c;发现…

微软启动了自爆程序,让我们一起帮它倒计时

……“公元2008年10月20日&#xff0c;注定成为人类信息技术史上不平凡的一天&#xff0c;因为在这一天&#xff0c;曾经创造了无数辉煌的计算机软件帝国微软公司&#xff0c;启动了自爆程序&#xff0c;剩下的&#xff0c;就是倒计时了……” ——《地球人类信息技术编年史》 …

python字典遍历的几种方法(转)

源地址&#xff1a;https://www.cnblogs.com/stuqx/p/7291948.html&#xff08;1&#xff09;遍历key值>>> a {a: 1, b: 2, c: 3} >>> for key in a:print(key:a[key])a:1 b:2 c:3 >>> for key in a.keys():print(key:a[key])a:1 b:2 c:3在使用上&a…

shiro学习(2):第一个shiro程序

工具idea 首先创建maven项目 配置文件 <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http…

解决水晶报表部署时出错的问题

我在使用VS2008自带的水晶报表开发一个项目时&#xff0c;遇上这么一个问题&#xff0c;就是在开发环境下运行没有问题&#xff0c;而且在开发的机器上部署发布的项目时也没有问题&#xff0c;但是在服务器上部署发布的项目就出现web.config配置文件出错。 错误报的是&#xff…

用树莓派(等)为 USB Midi 键盘增添连接方式

我在去年买了一个 M-Audio 的 Midi 键盘&#xff0c;用来连接电脑或者 iPad 弹琴。但是由于琴摆放的位置没有办法拉充电线&#xff0c;所以我能弹琴多久很大程度上取决于设备还有多少电。前一阵子从朋友手里白嫖了个橘子派&#xff0c;我就考虑用这个板子给我的 Midi 键盘做个无…

python ftp 上传文档出现 553 Could not create file

报错内容&#xff1a; *cmd* TYPE I*put* TYPE I\r\n*get* 200 Switching to Binary mode.\r\n*resp* 200 Switching to Binary mode.*cmd* PASV*put* PASV\r\n*get* 227 Entering Passive Mode (10,65,252,38,254,125).\r\n*resp* 227 Entering Passive Mode (10,65,252,38,25…

shiro学习(3):用户权限

工具idea 首先创建maven项目 配置文件 <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http…

WPF学习笔记(三)

1.1 事件概括 第一节中我们给窗体添加了一个按钮&#xff0c;不过好像Button点个几下也只有些发光样式的变化&#xff0c;什么你还把系统皮肤去掉了&#xff1f;算了承认下确实够寒碜&#xff0c;那让我们再动动手。 1.1.1 路由事件简述 public HelloWorld() { Button button …

结构体的赋值和初始化与取出结构体变量中的成员

1 /*结构体的赋值和初始化*/2 3 # include <stdio.h>4 5 struct Student 6 {7 int age;8 float score;9 char sex;10 };11 12 int main(void)13 {14 struct Student st { 80, 66.6, F};//定义同时就赋值15 struct Student st2;//下一行不能写一句类似于…

shiro学习(4):shiro认证流程

Shiro登录校验流程实现与分析 什么是Shiro Apache Shiro是一个强大且易用的Java安全框架,执行身份验证、授权、密码和会话管理。使用Shiro的易于理解的API,您可以快速、轻松地获得任何应用程序,从最小的移动应用程序到最大的网络和企业应用程序。 三个核心组件 Subject, Se…

python之Map函数

转载 https://www.cnblogs.com/gongxr/p/7247855.html# map()函数使用举例 # 功能&#xff1a;map()接受一个函数f和一个或多个list&#xff0c;将f依次作用在list的每个元素&#xff0c;得到一个新的列表 # 语法&#xff1a;map(方法名&#xff0c;列表&#xff0c;[列表2]) #…

Windows Mobile开发应该选择哪种开发语言?

Windows Mobile开发应该选择哪种开发语言&#xff1f;这个问题曾经被问了很多很多次&#xff0c;特别是打算开始学习Windows Mobile开发的朋友。Native Code or Managed Code? C/C or C#/VB.NET? 简单的说&#xff0c;Native Code的代码执行效率高&#xff0c;开发效率低&…

在ASP.NET MVC中实现Select多选

我们知道&#xff0c;在ASP.NET MVC中实现多选Select的话&#xff0c;使用Html.ListBoxFor或Html.ListBox方法就可以。在实际应用中&#xff0c;到底该如何设计View Model&#xff0c; 控制器如何接收多选Select的选中项呢&#xff1f; 实现效果如下&#xff1a; 初始状态某些选…

shiro学习(5):ini文件和自定义realm

工具idea 首先创建maven项目 配置文件 <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http…