《从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:我可以输出了
在这里插入图片描述
我想这应该是很好理解的。

相关文章
|
6天前
|
负载均衡 NoSQL 算法
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
这篇文章是关于Java面试中Redis相关问题的笔记,包括Redis事务实现、集群方案、主从复制原理、CAP和BASE理论以及负载均衡算法和类型。
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
|
6天前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
6天前
|
存储 监控 安全
一天十道Java面试题----第三天(对线程安全的理解------>线程池中阻塞队列的作用)
这篇文章是Java面试第三天的笔记,讨论了线程安全、Thread与Runnable的区别、守护线程、ThreadLocal原理及内存泄漏问题、并发并行串行的概念、并发三大特性、线程池的使用原因和解释、线程池处理流程,以及线程池中阻塞队列的作用和设计考虑。
|
4天前
|
存储 缓存 网络协议
复盘女朋友面试4个月的Java基础题
这篇文章是关于Java基础面试题的复盘,涵盖了HashMap原理、对象序列化作用等高频面试问题,并强调了Java基础知识的重要性。
复盘女朋友面试4个月的Java基础题
|
6天前
|
存储 NoSQL Java
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
这篇文章是关于Java面试中的分布式架构问题的笔记,包括分布式架构下的Session共享方案、RPC和RMI的理解、分布式ID生成方案、分布式锁解决方案以及分布式事务解决方案。
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
|
6天前
|
存储 安全 Java
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
这篇文章是关于Java面试的第二天笔记,涵盖了HashMap与HashTable的区别、ConcurrentHashMap的实现原理、IOC容器的实现方法、字节码的概念和作用、Java类加载器的类型、双亲委派模型、Java异常体系、GC如何判断对象可回收、线程的生命周期及状态,以及sleep、wait、join、yield的区别等十道面试题。
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
|
6天前
|
SQL Java 数据库连接
一天五道Java面试题----第六天(1)
这篇文章是关于Java面试中常见的五个问题,包括MyBatis和Hibernate的对比、MyBatis中#{}和${}的区别、MyBatis插件的运行原理及开发流程、索引的基本原理以及MySQL聚簇索引和非聚簇索引的区别。
|
6天前
|
缓存 NoSQL Redis
一天五道Java面试题----第九天(简述MySQL中索引类型对数据库的性能的影响--------->缓存雪崩、缓存穿透、缓存击穿)
这篇文章是关于Java面试中可能会遇到的五个问题,包括MySQL索引类型及其对数据库性能的影响、Redis的RDB和AOF持久化机制、Redis的过期键删除策略、Redis的单线程模型为何高效,以及缓存雪崩、缓存穿透和缓存击穿的概念及其解决方案。
|
6天前
|
前端开发 Java 数据库连接
一天十道Java面试题----第五天(spring的事务传播机制------>mybatis的优缺点)
这篇文章总结了Java面试中的十个问题,包括Spring事务传播机制、Spring事务失效条件、Bean自动装配方式、Spring、Spring MVC和Spring Boot的区别、Spring MVC的工作流程和主要组件、Spring Boot的自动配置原理和Starter概念、嵌入式服务器的使用原因,以及MyBatis的优缺点。
|
6天前
|
存储 关系型数据库 MySQL
一天五道Java面试题----第八天(怎么处理慢查询--------->简述Myisam和innodb的区别)
这篇文章是关于Java面试中关于数据库性能优化和MySQL特性的五个问题,包括处理慢查询、ACID特性保证、MVCC概念、MySQL主从同步原理以及MyISAM和InnoDB存储引擎的区别。