• Java基础面试题附答案 101-120(六)


    面试题 101-120(六)

    101、你是如何调用 wait()方法的?使用 if 块还是循环?为什么?

    102、什么是多线程环境下的伪共享(false sharing)?

    103、什么是 Busy spin?我们为什么要使用它?

    104、Java 中怎么获取一份线程 dump 文件?

    105、Swing 是线程安全的?

    106、什么是线程局部变量?

    107、用 wait-notify 写一段代码来解决生产者-消费者问题?

    108、用 Java 写一个线程安全的单例模式(Singleton)?

    109、Java 中 sleep 方法和 wait 方法的区别?

    110、什么是不可变对象(immutable object)?Java 中怎么创建一个不可变对象?

    111、我们能创建一个包含可变对象的不可变对象吗?

    112、Java 中应该使用什么数据类型来代表价格?

    113、怎么将 byte 转换为 String?

    114、Java 中怎样将 bytes 转换为 long 类型?

    115、我们能将 int 强制转换为 byte 类型的变量吗?如果该值大于byte 类型的范围,将会出现什么现象?

    116、存在两个类,B 继承 A ,C 继承 B,我们能将 B 转换为 C 么?如 C = (C) B;

    117、哪个类包含 clone 方法?是 Cloneable 还是 Object?

    118、Java 中 ++ 操作符是线程安全的吗?

    119、a = a + b 与 a += b 的区别

    120、我能在不进行强制转换的情况下将一个 double 值赋值给 long类型的变量吗?


    101、你是如何调用 wait()方法的?使用 if 块还是循环?为什么?

    wait() 方法应该在循环调用,因为当线程获取到 CPU 开始执行的时候,其他条件可能还没有满足,所以在处理前,循环检测条件是否满足会更好。下面是一段标准的使用 wait 和 notify 方法的代码:

    // The standard idiom for using the wait method
    synchronized (obj) {
        while (condition does not hold)
        obj.wait();
        // (Releases lock, and reacquires on wakeup)
        ... // Perform action appropriate to condition
    }
    

    102、什么是多线程环境下的伪共享(false sharing)?

    伪共享是多线程系统(每个处理器有自己的局部缓存)中一个众所周知的性能问题。伪共享发生在不同处理器的上的线程对变量的修改依赖于相同的缓存行。

    103、什么是 Busy spin?我们为什么要使用它?

    Busy spin 是一种在不释放 CPU 的基础上等待事件的技术。它经常用于避免丢失 CPU 缓存中的数据(如果线程先暂停,之后在其他 CPU 上运行就会丢失)。所以,如果你的工作要求低延迟,并且你的线程目前没有任何顺序,这样你就可以通过循环检测队列中的新消息来代替调用 sleep() 或 wait() 方法。它唯一的好处就是你只需等待很短的时间,如几微秒或几纳秒。LMAX 分布式框架是一个高性能线程间通信的库,该库有一个 BusySpinWaitStrategy 类就是基于这个概念实现的,使用 busy spin 循环 EventProcessors 等待屏障。

    104、Java 中怎么获取一份线程 dump 文件?

    在 Linux 下,你可以通过命令 kill -3 PID (Java 进程的进程 ID)来获取 Java应用的 dump 文件。在 Windows 下,你可以按下 Ctrl + Break 来获取。这样 JVM 就会将线程的 dump 文件打印到标准输出或错误文件中,它可能打印在控制台或者日志文件中,具体位置依赖应用的配置。

    105、Swing 是线程安全的?

    不是,Swing 不是线程安全的。你不能通过任何线程来更新 Swing 组件,如JTable、JList 或 JPanel,事实上,它们只能通过 GUI 或 AWT 线程来更新。这就是为什么 Swing供 invokeAndWait() 和 invokeLater() 方法来获取其他线程的 GUI 更新请求。这些方法将更新请求放入 AWT 的线程队列中,可以一直等待,也可以通过异步更新直接返回结果。你也可以在参考答案中查看和学习到更详细的内容。

    106、什么是线程局部变量?

    线程局部变量是局限于线程内部的变量,属于线程自身所有,不在多个线程间共享。Java 提供 ThreadLocal 类来支持线程局部变量,是一种实现线程安全的方式。但是在管理环境下(如 web 服务器)使用线程局部变量的时候要特别小心,在这种情况下,工作线程的生命周期比任何应用变量的生命周期都要长。任何线程局部变量一旦在工作完成后没有释放,Java 应用就存在内存泄露的风险。

    107、用 wait-notify 写一段代码来解决生产者-消费者问题?

    只要记住在同步块中调用 wait() 和 notify()方 法 ,如果阻塞,通过循环来测试等待条件。

    108、用 Java 写一个线程安全的单例模式(Singleton)?

    当我们说线程安全时,意思是即使初始化是在多线程环境中,仍然能保证单个实例。Java 中,使用枚举作为单例类是最简单的方式来创建线程安全单例模式的方式。

    109、Java 中 sleep 方法和 wait 方法的区别?

    虽然两者都是用来暂停当前运行的线程,但是 sleep() 实际上只是短暂停顿,因为它不会释放锁,而 wait() 意味着条件等待,这就是为什么该方法要释放锁,因为只有这样,其他等待的线程才能在满足条件时获取到该锁。

    110、什么是不可变对象(immutable object)?Java 中怎么创建一个不可变对象?

    不可变对象指对象一旦被创建,状态就不能再改变。任何修改都会创建一个新的对象,如 String、Integer 及其它包装类

    111、我们能创建一个包含可变对象的不可变对象吗?

    是的,我们是可以创建一个包含可变对象的不可变对象的,你只需要谨慎一点,不要共享可变对象的引用就可以了,如果需要变化时,就返回原对象的一个拷贝。最常见的例子就是对象中包含一个日期对象的引用。

    112、Java 中应该使用什么数据类型来代表价格?

    如果不是特别关心内存和性能的话,使用 BigDecimal,否则使用预定义精度的double 类型。

    113、怎么将 byte 转换为 String?

    可以使用 String 接收 byte[] 参数的构造器来进行转换,需要注意的点是要使用的正确的编码,否则会使用平台默认编码,这个编码可能跟原来的编码相同,也可能不同。

    114、Java 中怎样将 bytes 转换为 long 类型?

    1、不借助其他任何已经有的类,直接进行转换。

       /**
         * 将字节数组转为long
         * 如果input为null,或offset指定的剩余数组长度不足8字节则抛出异常
         * @param input 
         * @param offset 起始偏移量
         * @param littleEndian 输入数组是否小端模式
         * @return
         */
    public static long longFrom8Bytes(byte[] input, int offset, Boolean littleEndian){
        long value=0;
        // 循环读取每个字节通过移位运算完成long的8个字节拼装
        for (int  count=0;count<8;++count){
            int shift=(littleEndian?count:(7-count))<<3;
            value |=((long)0xff<< shift) & ((long)input[offset+count] << shift);
        }
        return value;
    }
    

    2、借助java.nio.ByteBuffer实现,只要将byte[]转换为ByteBuffer就可以实现所有primitive类型的数据读取。

       /**
         * 利用 {@link java.nio.ByteBuffer}实现byte[]转long
         * @param input
         * @param offset 
         * @param littleEndian 输入数组是否小端模式
         * @return
         */
    public static long bytesTolong(byte[] input, int offset, Boolean littleEndian) {
        // 将byte[] 封装为 ByteBuffer 
        ByteBuffer buffer = ByteBuffer.wrap(input,offset,8);
        if(littleEndian){
            // ByteBuffer.order(ByteOrder) 方法指定字节序,即大小端模式(BIG_ENDIAN/LITTLE_ENDIAN)
            // ByteBuffer 默认为大端(BIG_ENDIAN)模式 
            buffer.order(ByteOrder.LITTLE_ENDIAN);
        }
        return buffer.getlong();
    }
    

    3、借助java.io.DataInputStream实现,只要将byte[]转换为DataInputStream就可以实现所有primitive类型的数据读取。

    115、我们能将 int 强制转换为 byte 类型的变量吗?如果该值大于byte 类型的范围,将会出现什么现象?

    是的,我们可以做强制转换,但是 Java 中 int 是 32 位的,而 byte 是 8 位的,所以,如果强制转化是,int 类型的高 24 位将会被丢弃,byte 类型的范围是从 -128 到 127。

    116、存在两个类,B 继承 A ,C 继承 B,我们能将 B 转换为 C 么?如 C = (C) B;

    这属于强制类型转换,如果被转换的B实例不是C类型,会有异常。

    比如你的ABC分别对应动物,猫,黑猫。
    向上转型就是比如 C c = new C(); B b = c; 你把c转型为B,黑猫是猫吗?是啊,所以这是ok的。
    但是反过来 B b = new B(); C c = (C)b; 这就不ok了,只知道这个b是一只猫,他不一定是黑猫。
    但如果这个b已经确定是一只黑猫了,那就可以转型了 B b = new C(); C c = (C)b; 这里的b本来就是黑猫啊。

    117、哪个类包含 clone 方法?是 Cloneable 还是 Object?

    java.lang.Cloneable 是一个标示性接口,不包含任何方法,clone 方法在 object 类中定义。并且需要知道 clone() 方法是一个本地方法,这意味着它是由c 或 c++ 或 其他本地语言实现的。

    118、Java 中 ++ 操作符是线程安全的吗?

    不是线程安全的操作。它涉及到多个指令,如读取变量值,增加,然后存储回内存,这个过程可能会出现多个线程交差。

    119、a = a + b 与 a += b 的区别

    += 隐式的将 '+' 操作的结果类型强制转换为持有结果的类型。如果两个类型相加,如 byte、short 或者 int,首先会将它们提升到 int 类型,然后在执行加法操作,因为 a+b 操作会将 a、b 提升为 int 类型,所以将 int 类型赋值给 byte 就会编译出错

    byte a = 127;
    byte b = 127;
    b = a + b;  // error : cannot convert from int to byte
    b += a;     // ok
    

    120、我能在不进行强制转换的情况下将一个 double 值赋值给 long类型的变量吗?

    不行,因为 double 类型的范围比 long 类型更广,所以必须要进行强制转换。


    上一篇:Java基础面试题附答案 81-100(五)
    下一篇:

  • 相关阅读:
    js(一)
    css (一)
    html
    Spring MVC(四)
    Spring MVC(三)
    Spring MVC(二)
    Spring MVC(一)
    Druid应用
    C3P0使用
    jdbc(二)
  • 原文地址:https://www.cnblogs.com/newRyan/p/12697131.html
Copyright © 2020-2023  润新知