《从Java面试题看源码》-Java11中的toString与Java8的区别

简介: 《从Java面试题看源码》-Java11中的toString与Java8的区别

​​在这里插入图片描述
在前面【《从Java面试题来看源码》-LinkedBlockingQueue 源码分析】的文章中,我们看到有一个toString方法是这样的:

public String toString() {
    //
    return Helpers.collectionToString(this);
}

为什么要这样呢?

分析

`使用Helpers类,来输出字符串,与Java8不同。
Helpers类用于并发包输出字符串,该类只在输出数组的时候获取锁,而不是在toString中获取锁`

Java11中用到了Helpers.collectionToString(this)的方式输出字符串,并且与Java8是不同的。

先看看Java11中Helpers类的写法:

 /**
     * Collection.toString() 的一种实现,适用于有锁的类。
     * 代替了以前在整个toString()过程中加锁,或者在每次调用Iterator.next()的时候加锁
     * 该方法只在调用toArray()期间加锁,以减少其他线程对访问集合时产生的影响
     * 并且遵循在加锁期间,不调用任何外部代码
     */
static String collectionToString(Collection<?> c) {
    //这里toArray会加锁
    final Object[] a = c.toArray();
    final int size = a.length;
    if (size == 0)
        return "[]";
    int charLength = 0;

    // Replace every array element with its string representation
    for (int i = 0; i < size; i++) {
        Object e = a[i];
        // Extreme compatibility with AbstractCollection.toString()
        String s = (e == c) ? "(this Collection)" : objectToString(e);
        a[i] = s;
        charLength += s.length();
    }

    return toString(a, size, charLength);
}

/**
 * 与 Arrays.toString() 类似,但调用者保证 size > 0,索引为 0 <= i < size 的每
 * 个元素都是非空 String,charLength 是输入 String 的长度之和。
 */
static String toString(Object[] a, int size, int charLength) {
    // assert a != null;
    // assert size > 0;

    // Copy each string into a perfectly sized char[]
    // Length of [ , , , ] == 2 * size
    final char[] chars = new char[charLength + 2 * size];
    chars[0] = '[';
    int j = 1;
    for (int i = 0; i < size; i++) {
        if (i > 0) {
            chars[j++] = ',';
            chars[j++] = ' ';
        }
        String s = (String) a[i];
        int len = s.length();
        s.getChars(0, len, chars, j);
        j += len;
    }
    chars[j] = ']';
    // assert j == chars.length - 1;
    return new String(chars);
}

整个过程中就是将当前状态队列的元素进行拼接输出,而不会影响到其他线程操作队列,只是在通过toArray()获取队列元素的时候进行加锁。

看看Java8是怎么写的:

public String toString() {
    fullyLock();
    try {
        Node<E> p = head.next;
        if (p == null)
            return "[]";

        StringBuilder sb = new StringBuilder();
        sb.append('[');
        for (;;) {
            E e = p.item;
            sb.append(e == this ? "(this Collection)" : e);
            p = p.next;
            if (p == null)
                return sb.append(']').toString();
            sb.append(',').append(' ');
        }
    } finally {
        fullyUnlock();
    }
}

Java8中,toString输出字符串在前面加了一个锁fullyLock(),fullyLock()使用ReentrantLock对put和take、poll分别加锁。

Java8会在整个toString的拼接过程中,对队列进行加锁,会影响性能。

private final ReentrantLock takeLock = new ReentrantLock();
private final ReentrantLock putLock = new ReentrantLock();
void fullyLock() {
    putLock.lock();
    takeLock.lock();
}

总结

简单点说

Java8中的toString,就如同一个人干活,一群人歇下来看着他干完完,比较粗暴。
在这里插入图片描述
Java11中是:

toString():我要输出了,队列你先把当前值给我

队列:放下原来的事,把toString()要的数据准备好,给了toString()后,继续做原来的事

toString:我可以输出了
在这里插入图片描述
我想这应该是很好理解的。

相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
22天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
55 14
|
1月前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
|
1月前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
21天前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
1月前
|
存储 缓存 Oracle
Java I/O流面试之道
NIO的出现在于提高IO的速度,它相比传统的输入/输出流速度更快。NIO通过管道Channel和缓冲器Buffer来处理数据,可以把管道当成一个矿藏,缓冲器就是矿藏里的卡车。程序通过管道里的缓冲器进行数据交互,而不直接处理数据。程序要么从缓冲器获取数据,要么输入数据到缓冲器。
Java I/O流面试之道
|
27天前
|
Java 编译器 程序员
Java面试高频题:用最优解法算出2乘以8!
本文探讨了面试中一个看似简单的数学问题——如何高效计算2×8。从直接使用乘法、位运算优化、编译器优化、加法实现到大整数场景下的处理,全面解析了不同方法的原理和适用场景,帮助读者深入理解计算效率优化的重要性。
30 6
|
1月前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
58 4
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
117 4
|
1月前
|
存储 缓存 网络协议
计算机网络常见面试题(二):浏览器中输入URL返回页面过程、HTTP协议特点,GET、POST的区别,Cookie与Session
计算机网络常见面试题(二):浏览器中输入URL返回页面过程、HTTP协议特点、状态码、报文格式,GET、POST的区别,DNS的解析过程、数字证书、Cookie与Session,对称加密和非对称加密