1.探索ArrayList源码之实现的三个接口作用

首先来看它的继承体系

image.png

public class ArrayList<E> extends AbstractList<E>
 implements List<E>, RandomAccess, Cloneable, java.io.Serializable

**它实现的三了接口,相同点都是空接口,作为一个标记出现

  • 1 RandomAccess:
public interface RandomAccess {
}

RandomAccess作为随机访问的标志,代表只要实现了这个接口,就能支持快速随机访问。

下面来看下它的用法:

Collections类中的binarySearch()方法
instanceof其作用是用来判断某对象是否为某个类或接口类型
由此可以看出,根据判断list是否实现RandomAccess接口来决定实行indexedBinarySerach(list,key)或iteratorBinarySerach(list,key)方法。

indexedBinarySerach(list,key)方法源码:

private static <T> int indexedBinarySearch(List<? extends T> l, T key, Comparator<? super T> c) {
        int low = 0;
        int high = l.size()-1;

        while (low <= high) {
            int mid = (low + high) >>> 1;
            // 普通for循环获取元素
            T midVal = l.get(mid);
            int cmp = c.compare(midVal, key);

            if (cmp < 0)
                low = mid + 1;
            else if (cmp > 0)
                high = mid - 1;
            else
                return mid; // key found
        }
        return -(low + 1);  // key not found
    }


/** List 接口 */
public interface List<E> extends Collection<E> {
      ...
      ...
/**
     * Returns the element at the specified position in this list.
     *
     * @param index index of the element to return
     * @return the element at the specified position in this list
     * @throws IndexOutOfBoundsException if the index is out of range
     *         (<tt>index &lt; 0 || index &gt;= size()</tt>)
     */
    E get(int index);
}

iteratorBinarySerach(list,key)方法源码

 private static <T>
    int iteratorBinarySearch(List<? extends Comparable<? super T>> list, T key)
    {
        int low = 0;
        int high = list.size()-1;
        ListIterator<? extends Comparable<? super T>> i = list.listIterator();

        while (low <= high) {
            int mid = (low + high) >>> 1;
            // 迭代器获取元素
            Comparable<? super T> midVal = get(i, mid);
            int cmp = midVal.compareTo(key);

            if (cmp < 0)
                low = mid + 1;
            else if (cmp > 0)
                high = mid - 1;
            else
                return mid; // key found
        }
        return - (low + 1);  // key not found
    }


    /**
     * Gets the ith element from the given list by repositioning the specified
     * list listIterator.
     */
    private static <T> T get(ListIterator<? extends T> i, int index) {
        T obj = null;
        int pos = i.nextIndex();
        if (pos <= index) {
            do {
                obj = i.next();
            } while (pos++ < index);
        } else {
            do {
                obj = i.previous();
            } while (--pos > index);
        }
        return obj;
    }

通过查看源代码,发现实现RandomAccess接口的List集合采用一般的for循环遍历,而未实现这接口则采用迭代器。ArrayList用for循环遍历比iterator迭代器遍历快,LinkedList用iterator迭代器遍历比for循环遍历快。

总结:
判断出接收的List子类是ArrayList还是LinkedList,需要用instanceof来判断List集合子类是否实现RandomAccess接口!从而能够更好选择更优的遍历方式,提高性能!

  • 2 Cloneable
    Cloneable也是一个标记接口,只有实现这个接口后,然后在类中重写Object中的clone方法,后面通过类调用clone方法才能克隆成功,如果不实现这个接口,则会抛出CloneNotSupportedException(克隆不被支持)异常。
    Object中的clone方法:
protected native Object clone throws CloneNotSupportedException;

这里有一个疑问:Object中的clone方法是一个空的方法,那么他是如何判断类是否实现了cloneable接口呢?
原因在于这个方法中有一个native关键字修饰。

下面对 native 方法做简单解释:

native关键字的函数都是操作系统实现的,java只能调用。简单地讲,一个Native 方法就是一个java调用非java代码的接口。
java是跨平台的语言,既然是跨了平台,所付出的代价就是牺牲一些对底层的控制,而java要实现对底层的控制,就要一些其他语言的帮助,这个就是native的作用了
native的意思就是通知操作系统,这个函数你必须给我实现,因为我要使用。所以native关键字的函数都是操作系统实现的
本地方法非常有用,因为它有效地扩充了jvm.事实上,我们所写的java代码已经用到了本地方法,在sun的java的并发(多线程)的机制实现中,许多与操作系统的接触点都用到了本地方法,这使得java程序能够超越java运行时的界限。有了本地方法,java程序可以做任何应用层次的任务。
每一个native方法在jvm中都有一个同名的实现体,native方法在逻辑上的判断都是由实现体实现,另外这种native修饰的方法对返回类型,异常控制等都没有约束。
由此可见,这里判断是否实现cloneable接口,是在调用jvm中的实现体时进行判断的。

JVM怎样使Native Method跑起来:

我们知道,当一个类第一次被使用到时,这个类的字节码会被加载到内存,并且只会加载一次。
在这个被加载的字节码的入口维持着一个该类所有方法描述符的list,这些方法描述符包含这样一些信息:方法代码存于何处,它有哪些参数,方法的描述符(public之类)等等。
如果一个方法描述符内有native,这个描述符块将有一个指向该方法的实现的指针。
这些实现在一些DLL文件内,但是它们会被操作系统加载到java程序的地址空间。
当一个带有本地方法的类被加载时,其相关的DLL并未被加载,因此指向方法实现的指针并不会被设置。
当本地方法被调用之前,这些DLL才会被加载,这是通过调用java.system.loadLibrary()实现的。

最后需要提示的是,使用本地方法是有开销的,它丧失了java的很多好处。如果别无选择,我们才选择使用本地方法。

深入理解深度克隆与浅度克隆:

  • 浅度克隆:
    定义一个学生类
public class Student{
    private String name;   //姓名
    private int age;       //年龄
    private StringBuffer sex;  //性别
     
     get、set、toString方法
 }

定义一个学校类,类中重写clone方法

public  class School implements Cloneable{  
    private String schoolName;   //学校名称 
    private int stuNums;         //学校人数 
    private Student stu;         //一个学生
    
    get、set、toString方法
    
    @Override
    protected School clone() throws CloneNotSupportedException {
        // TODO Auto-generated method stub
        return (School)super.clone();
    }
}

最后写一个main类来测试一下:

public class CloneTest {

    public static void main(String[] args) throws CloneNotSupportedException {

        Student student = new Student();
        student.setAge(18);
        student.setName("小王");
        student.setSex(new StringBuffer("男"));

        School school = new School();
        school.setSchoolName("实验小学");
        school.setStuNums(100);
        school.setStu(student);

        System.out.println("school的hashCode:"+school.hashCode() + "; 
                               school的student的hashCode:" + school.getStu().hashCode());

        School school2 = school.clone();
        System.out.println("school2的hashCode:"+school2.hashCode() + "; 
                               school2的student的hashCode:" + school.getStu().hashCode());
    }
}

控制台打印:

school的hashCode:204349222; school的student的hashCode:231685785
school2的hashCode:114935352; school2的student的hashCode:231685785

可以看到克隆的school2跟源对象school的hashCode不同,也就是说clone方法并不是把school的引用赋予school2,而是在堆中重新开辟了一块空间,将school复制过去,将新的地址返回给school2。
但是,它们里面的student对象的hashCode仍然是相同的,这两个指向了同一个对象。
也就是说修改school2的基本数据类型与Stirng类型,不会改变school的基本数据类型与Stirng类型。

基本数据类型例如int,在clone的时候会重新开辟一个四个字节的大小的空间,将其赋值。而String则由于String变量的唯一性,如果在school2中改变了String类型的值,则会生成一个新的String对象,对之前的没有影响。 这就是浅度隆。

  • 深度克隆:
    需要让student实现cloneable接口,重写clone方法
 @Override
 protected Student clone() throws CloneNotSupportedException {
     return (Student) super.clone();
 }

然后,在school的clone方法中将school中的stu对象手动clone。

 @Override
 protected School clone() throws CloneNotSupportedException {
     School school = null;
     school = (School)super.clone();
     school.stu = stu.clone();
     return school;
 }

再次main方法执行

school的hashCode:204349222; school的student的hashCode:231685785
school2的hashCode:114935352; school2的student的hashCode:2110121908

但是,school2修改sex的值,发现school也随着改变了

// 修改school2的student2的值
Student student2 = school2.getStu();
student2.setSex(student2.getSex().append(6666));
student2.setName("如花");
student2.setAge(68);
// 修改school2其它值
school2.setSchoolName("希望小学");
school2.setStuNums(1000);
System.out.println("school: " + school);
System.out.println("school2: " + school2);

控制台打印:

school的hashCode:204349222; school的student的hashCode:231685785
school2的hashCode:114935352; school2的student的hashCode:2110121908
school: School{schoolName='实验小学', stuNums=100, stu=Student{name='小王', age=18, sex=男6666}}
school2: School{schoolName='希望小学', stuNums=1000, stu=Student{name='如花', age=68, sex=男6666}}

原因在于sex的类型是Stringbuffer,而StringBuffer类型没有实现cloneable接口,也没有重写clone方法。在clone的时候将StringBuffer对象的地址传递了过去}

这种情况应该怎么解决呢
StringBuffer是没有实现cloneable接口的,所以无法克隆,因此在设置stu2的sex时,创建一个新的StringBuffer对象

// student2.setSex(student2.getSex().append(6666));
// 创建一个新的StringBuffer对象。解决修改sex影响源数据的问题
student2.setSex(new StringBuffer("newString"));

控制台打印:

school2的hashCode:114935352; school2的student的hashCode:2110121908
school: School{schoolName='实验小学', stuNums=100, stu=Student{name='小王', age=18, sex=男}}
school2: School{schoolName='希望小学', stuNums=1000, stu=Student{name='如花', age=68, sex=newString}}
  • 3 Serializable

Serializable接口也是一个标记,没有方法

首先来看下如何把对象输出到文件中,创建一个类实现序列化接口

public class Person implements Serializable {

    private String name;
    private int age;
    ...
}

测试类

package com.Serializable;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;

/**
 * @author: jk
 * @since: 2020-03-08 01:14
 * <p>
 * 描述
 * </p>
 **/
public class SerializableTest {

    public static void main(String[] args) throws IOException {

        // 1 创建对象
        Person person = new Person("小王", 18);

        // 2 创建对象输出流
        ObjectOutputStream objOut = new ObjectOutputStream(new 
                                          FileOutputStream("C:/Users/Administrator/Desktop/photo/person.txt"));

        // 3 调用方法将对象写入流中
        objOut.writeObject(person);

        // 4 关闭输出流
        objOut.close();
    }
}
image.png

Serializable的步骤:

1.首先要创建某些OutputStream对象:
OutputStream outputStream = new FileOutputStream("output.txt")

2.将其封装到ObjectOutputStream对象内:
ObjectOutputStream objectOutputStream = new ObjectOutputStream(outputStream);

3.此后只需调用writeObject()即可完成对象的序列化,并将其发送给OutputStream:objectOutputStream.writeObject(Object);

4.最后不要忘记关闭资源:
objectOutputStream.close(), outputStream .close();

Serializable的一些说明:

对象的序列化处理非常简单,只需对象实现了Serializable 接口即可(该接口仅是一个标记,没有方法)
序列化的对象包括基本数据类型,所有集合类以及其他许多东西,还有Class 对象
对象序列化不仅保存了对象的“全景图”,而且能追踪对象内包含的所有句柄并保存那些对象;
接着又能对每个对象内包含的句柄进行追踪

使用transient、static关键字修饰的的变量,在序列化对象的过程中,该属性不会被序列化。
但是,会发现static修饰的成员变量在序列化后然后紧接着反序列化,打印出来的这个类的成员变量是有值的!!!原因是读取的值是当前jvm中的方法区对应此变量的值。

反序列化:

public class SerializableTest {

    public static void main(String[] args) throws IOException, ClassNotFoundException {

       // serializable();
        System.out.println(deserialization());

    }

private static String deserialization() throws IOException, ClassNotFoundException {
    // 1 创建输入流
    ObjectInputStream objIn = new ObjectInputStream(new FileInputStream("C:/Users/Administrator/Desktop/photo/person.txt"));
    // 2 调用方法读取对象
    Person person = (Person) objIn.readObject();
    // 3 关闭流
    objIn.close();
    return person.toString();
   }
}

控制台打印:

Person{name='小王', age=18}
  • 关于序列化冲突问题:

就是当创建一个类实现了Serializable接口之后,会为这个类添加序列化号
当修改这个类的时候,这个类的序列化号也会发生改变。
因此,在修改类前序列化;修改类之后反序列化,会校验文件的序列化号和该类的序列化号是否一致,不一致则抛出序列化号不一致的异常,也就是序列化冲突

解决方案:

在指定的类中指定序列化号
private static final long serialVersionUID = 123456789L;

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容