Java通过底层原码了解数组拷贝

目录
  • 循环对数组进行拷贝
  • Arrays.copyOf方法
  • Arrays.copyofRange方法
  • System.arraycopy方法
  • clone方法
  • 总结

循环对数组进行拷贝

利用循环对数组进行拷贝很简单,就是利用循环将要拷贝的数组的元素逐个赋值给新的数组。具体代码实现如下:

public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] copy = new int[arr.length];
        for (int i = 0; i < arr.length; i++) {
            copy[i] = arr[i];
        }
    }

这是我们一般会想到的方法,下面将一些JAVA为我们提供的一些比较方便的方法。

Arrays.copyOf方法

因为我是拷贝的整型数组,所以我用的就是拷贝整型数组的Arrays.copyOf方法。也可以拷贝其它类型的数组,并不是只能拷贝整型数组。

下面就以拷贝整型数组来讲解Arrays.copyOf方法,我们先来看一下Arrays.copyOf方法底层的原码:

public static int[] copyOf(int[] original, int newLength) {
        int[] copy = new int[newLength];
        System.arraycopy(original, 0, copy, 0,
                         Math.min(original.length, newLength));
        return copy;
    }

我们可以看到Array.copyOf方法的返回值是是一个整型数组,而形参是有两个参数,一个是整型数组,另外一个是一个整型的变量。

形参中整型数组就是我们要拷贝数组的数组名

形参中的整型变量就是我们要拷贝数组的长度

因为是有返回值的,所以我们在使用Arrays.copyOf方法的时候,要有数组来接收返回值。代码实现如下:

public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] copy =Arrays.copyOf(arr,arr.length);
    }

注意:拷贝的长度是可以超过要拷贝数组的长度的,如果超过了拷贝数组的长度的话,那么多出来的元素值就为0;比如:我要拷贝arr数组,但是arr数组的大小为5,如果我在拷贝的时候,将arr.length*2了,那么copy这个数组大小就是变成10,并且6~10个元素值为0.这也算是扩容。

Arrays.copyofRange方法

如果我们要拷贝部分数组的一部分,就可以使用Arrays.copyRange这个方法。同上一样,也是用拷贝整型数组进行讲解,我们先看一下它底层的原码:

public static int[] copyOfRange(int[] original, int from, int to) {
        int newLength = to - from;
        if (newLength < 0)
            throw new IllegalArgumentException(from + " > " + to);
        int[] copy = new int[newLength];
        System.arraycopy(original, from, copy, 0,
                         Math.min(original.length - from, newLength));
        return copy;
    }

我们通过Arrays.copyRange的原码可以看到返回值是一个整型的数组,而形参是一个整型数组,两个整型的变量。

形参中的整型数组就是要拷贝的数组的数组名

形参中的两个整型变量from和to就是你要拷贝数组元素的范围,注意这个范围是[form,to)是左闭右开的,不包括to那个位置的元素。代码实现如下:

public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] copy = Arrays.copyOfRange(arr,1,3);
    }

注意:数组下标是从0开始的,1~3号的元素是指2,3,4.但是右边是开区间,所以不包括4,因此copy这个数组知识拷贝了2和3这两个元素。

System.arraycopy方法

然后我们来讲一下System.arraycopy方法,其实细心的同学已经发现了,在前面两个方法的原码中都有System.arraycopy方法,他们的底层也是靠System.arraycopy方法来实现的。我们来先看一下它底层的原码

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

System.arraycopy是看不到实现的原码的,原因在于native,它的底层是C/C++来实现的。但是native方法的优点就是执行速度非常快。代码实现如下:

public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] copy = new int[arr.length];
        System.arraycopy(arr,0,copy,0,arr.length);
    }

此时拷贝的是arr整个数组,如果想要部分拷贝,可以通过改变形参来完成部分拷贝

clone方法

最后一个是数组名.clone这个方法,这个方法了解即可,代码实现如下:

public static void main(String[] args) {
        int[] arr = {1,2,3,4,5};
        int[] copy = arr.clone();
    }

这样就拷贝完了,这种方法主要是产生了对象的一个副本,地址还是不一样的。

总结

今天主要讲解了数组拷贝的几种方法,虽然可能不会经常用到,但是我们还是掌握一下比较好。说不定那天就用到了,将来的你一定会感谢现在努力学习的自己。

到此这篇关于Java通过底层原码了解数组拷贝的文章就介绍到这了,更多相关Java数组拷贝内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java编程中拷贝数组的方式及相关问题分析

    JAVA数组的复制是引用传递,而并不是其他语言的值传递. 这里介绍java数组复制的4种方式极其问题: 第一种方式利用for循环: int[] a={1,2,4,6}; int length=a.length; int[] b=new int[length]; for (int i = 0; i < length; i++) { b[i]=a[i]; } 第二种方式直接赋值: int[] array1={1,2,4,6}; int[] array2=a; 这里把array1数组的值复制给arra

  • 详解Java数组的四种拷贝方式

    目录 深拷贝与浅拷贝的区别 1.for循环进行拷贝 拷贝数值类型 拷贝引用类型 2.copyof/copyOfRange 拷贝数值类型 拷贝引用类型 3.arraycopy 拷贝数值类型 拷贝引用类型 4.clone 拷贝数值类型 拷贝引用类型 5.总结 深拷贝与浅拷贝的区别 假设现在有原数组A以及拷贝后的数组B,若是改变A中的某一个值,B数组随之相应的发生变化的拷贝方式称为浅拷贝,反之B数组不受影响,则称为深拷贝:简单总结一下两者的概念: 深拷贝:拷贝后,修改原数组,不会影响到新数组: 浅拷贝

  • Java拷贝数组方法Arrays.copyOf()是地址传递的证明实例

    在看别人的关于CopyOnWriteArrayList 这个类的时候,看到有人提出了关于:数组拷贝的方法Arrays.copyOf() 的问题,如下: 只是复制了引用地址,数组内的对象还是和旧数组一样的? 看了他的问题,我也是有点迷惑了,以前都没想过这个.既然如此,我就有如下的测试,以解心中之疑惑. 看代码名字,就是从其他地方获得三个student对象到这个数组里面,打印没复制之前的数组信息.然后修改新新数组的第0个学生的名字.然后再输出copy出来的数组和源数组,看看啥情况. 结果: 虽然co

  • Java通过底层原码了解数组拷贝

    目录 循环对数组进行拷贝 Arrays.copyOf方法 Arrays.copyofRange方法 System.arraycopy方法 clone方法 总结 循环对数组进行拷贝 利用循环对数组进行拷贝很简单,就是利用循环将要拷贝的数组的元素逐个赋值给新的数组.具体代码实现如下: public static void main(String[] args) { int[] arr = {1,2,3,4,5}; int[] copy = new int[arr.length]; for (int

  • java原码补码反码关系解析

    本文为大家解析了java原码补码反码的关系,供大家参考,具体内容如下 原码:不管源数据是十进制还是十六进制,统统将数字转成二进制形式 反码:把原码的二进制统统反过来,0变成1,1变成0 补码:负数的反码加1,就是负数的补码 例子1: 十进制整数 -5,-5的原码101,-5的反码1,010,-5的补码1,011,  所以,-5的反码是-2,-5的补码是-3,其中前面的"1,",表示正数负数 例子2: 十六进制  -ff9B   原码  11111111111110011011 反码  

  • Java同步锁Synchronized底层源码和原理剖析(推荐)

    目录 1 synchronized场景回顾 2 反汇编寻找锁实现原理 3 synchronized虚拟机源码 3.1 HotSpot源码Monitor生成 3.2 HotSpot源码之Monitor竞争 3.3 HotSpot源码之Monitor等待 3.4 HotSpot源码之Monitor释放 1 synchronized场景回顾 目标:synchronized回顾(锁分类–>多线程)概念synchronized:是Java中的关键字,是一种同步锁.Java中锁分为以下几种:乐观锁.悲观锁(

  • Java集合框架源码分析之LinkedHashMap详解

    LinkedHashMap简介 LinkedHashMap是HashMap的子类,与HashMap有着同样的存储结构,但它加入了一个双向链表的头结点,将所有put到LinkedHashmap的节点一一串成了一个双向循环链表,因此它保留了节点插入的顺序,可以使节点的输出顺序与输入顺序相同. LinkedHashMap可以用来实现LRU算法(这会在下面的源码中进行分析). LinkedHashMap同样是非线程安全的,只在单线程环境下使用. LinkedHashMap源码剖析 LinkedHashM

  • Java 中模仿源码自定义ArrayList

    Java 中模仿源码自定义ArrayList 最近看了下ArrayList的源码,抽空根据ArrayList的底层结构写了一个功能简单无泛型的自定义ArrayLsit,帮助自己更好理解ArrayList:,其实现的底层数据结构为数Object组,代码如下: /** * 自己实现一个ArrayList * */ public class MyArrayList { private Object[] elementData; private int size; public int size(){

  • Java synchronized底层实现原理以及锁优化

    目录 一.概述 synchronized简介 synchronized作用 synchronized的使用 二.实现原理 三.理解Java对象头 四.JVM对synchronized的锁优化 1.偏向锁 2.轻量级锁 3.重量级锁 4.自旋锁 5.锁消除 6.锁粗化 总结 一.概述 synchronized简介 在多线程并发编程中 synchronized 一直是元老级角色,很多人都会称呼它为重量级锁.但是,随着 Java SE 1.6 对synchronized 进行了各种优化之后,有些情况下

  • java中CopyOnWriteArrayList源码解析

    目录 简介 继承体系 源码解析 属性 构造方法 add(Ee)方法 add(intindex,Eelement)方法 addIfAbsent(Ee)方法 get(intindex) remove(intindex)方法 size()方法 提问 总结 简介 CopyOnWriteArrayList是ArrayList的线程安全版本,内部也是通过数组实现,每次对数组的修改都完全拷贝一份新的数组来修改,修改完了再替换掉老数组,这样保证了只阻塞写操作,不阻塞读操作,实现读写分离. 继承体系 public

  • java 中Buffer源码的分析

    java 中Buffer源码的分析 Buffer Buffer的类图如下: 除了Boolean,其他基本数据类型都有对应的Buffer,但是只有ByteBuffer才能和Channel交互.只有ByteBuffer才能产生Direct的buffer,其他数据类型的Buffer只能产生Heap类型的Buffer.ByteBuffer可以产生其他数据类型的视图Buffer,如果ByteBuffer本身是Direct的,则产生的各视图Buffer也是Direct的. Direct和Heap类型Buff

  • Java CAS底层实现原理实例详解

    这篇文章主要介绍了Java CAS底层实现原理实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 一.CAS(compareAndSwap)的概念 CAS,全称Compare And Swap(比较与交换),解决多线程并行情况下使用锁造成性能损耗的一种机制. CAS(V, A, B),V为内存地址.A为预期原值,B为新值.如果内存地址的值与预期原值相匹配,那么将该位置值更新为新值.否则,说明已经被其他线程更新,处理器不做任何操作:无论哪种情

  • 详解Java中一维、二维数组在内存中的结构

    前言 我们知道在Java中数组属于引用数据类型,它整个数组的数组元素既可以是基本数据类型的(如 byte \ int \ short \ long \ float \ double \ char \ boolean 这些),也可以是引用数据类型的.当它的数组元素是基本数据类型时,这个数组就是一个一维数组:当它的数组元素是引用数据类型时,它就是一个多维数组.比如,在一个数组中它的某个元素值其实是一个一维数组,而其他不同的元素也各自包含了一个一维数组,我们就把这个包含很多个一维数组的数组叫做二维数组

随机推荐