Java——10个关于Java数据类型的面试题

简介: Java——10个关于Java数据类型的面试题

文章目录:


1.Java的基本数据类型(及包装类型)都有哪些各占几个字节?

2.String是基本数据类型吗?

3.String类可以被继承吗?

4.String s = "Hello";s = s + " world!";这两行代码执行后,原始的 String 对象中的内容变了没有?

5.String 类常用方法

6.StringStringBufferStringBuilder的区别?

7.short s1 = 1; s1 = s1 + 1; 有错吗?short s1 = 1; s1 += 1 有错吗?

8.intInteger有什么区别?

9.下面 Integer 类型的数值比较输出的结果为?

10.数据类型之间如何进行转换?

1.Java的基本数据类型(及包装类型)都有哪些各占几个字节?


·       基本数据类型:byte short    int     longfloat doubleboolean    char

·       对应包装类型:ByteShortIntegerLongFloatDoubleBooleanCharacter

·       各自所占字节数:1        2             4            8         4           8              1                2


2.String是基本数据类型吗?


虽然我目前使用的是jdk14,但以主流的jdk1.8来说,Spring类内部实际存储的结构为char数组,源码如下:(所以说String不是Java中的基本数据类型)

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];
    /** Cache the hash code for the string */
    private int hash; // Default to 0
    /** use serialVersionUID from JDK 1.0.2 for interoperability */
    private static final long serialVersionUID = -6849794470754667710L;
    //...其他内容省略
}    


3.String可以被继承吗?

答案:不能!!!  String类在声明时使用final关键字修饰,被final关键字修饰的类无法被继承。


为什么Java语言的开发者,把String类定义为final的呢?

因为字符串是不可变的,所以是多线程安全的,同一个字符串实例可以被多个线程共享。因为不可变对象不能更改,它们可以在多个线程之间自由共享。

因为字符串是不可变的,所以在它创建的时候HashCode就被缓存了,不需要重新计算。这就使得字符串很适合作为Map中的键,字符串的处理速度要快过其它的键对象。这就是HashMap中的键往往都使用字符串。

·       从内存角度来看:字符串常量池的要求:创建字符串时,如果该字符串已经存在于池中,则将返回现有字符串的引用,而不是创建新对象。多个String变量引用指向同一个内地地址。如果字符串是可变的,用一个引用更改字符串将导致其他引用的值错误。这是很危险的。

·       缓存Hashcode:字符串的Hashcodejava中经常配合基于散列的集合一起正常运行,这样的散列集合包括HashSetHashMap以及HashTable。不可变的特性保证了hashcode永远是相同的。不用每次使用hashcode就需要计算hashcode。这样更有效率。因为当向集合中插入对象时,是通过hashcode判别在集合中是否已经存在该对象了(不是通过equals方法逐个比较,效率低)。

·       方便其它类使用:其他类的设计基于string不可变,如set存储string,改变该stringset包含了重复值。

·       安全性:String被广泛用作许多java类的参数,例如网络连接、打开文件等。如果对string的某一处改变一不小心就影响了该变量所有引用的表现,则连接或文件将被更改,这可能导致严重的安全威胁。

总结由于效率和安全性的原因,字符串被设计为不可变。

● final关键字除了修饰类之外,还有哪些用法呢?

final修饰的变量,一旦赋值,不可重新赋值;

final修饰的方法无法被覆盖;

final修饰的实例变量,必须手动赋值,不能采用系统默认值;

final修饰的实例变量,一般和static联用,用来声明常量;

注意:final不能和abstract关键字联合使用。

总之,final表示最终的、不可变的。


4.String s = "Hello";s = s + " world!";这两行代码执行后,原始的String 对象中的内容变了没有?


答案:没有!!!

因为 String被设计成不可变类,所以它的所有对象都是不可变对象。(说白了,就是 final 关键字在起作用)

在这段代码中,s原先指向一个 String 对象,内容是"Hello",然后我们对 s 进行了“+”操作,那么 s 所指向的那个对象是否发生了改变呢?

答案是没有。这时s不指向原来那个对象了,而指向了另一个 String 对象,内容为"Hello world!",原来那个对象还存在于内存之中,只是 s 这个引用变量不再指向它了。

通过上面的说明,我们很容易导出另一个结论,如果经常对字符串进行各种各样的修改,或者说,不可预见的修改,那么使用 String 来代表字符串的话会引起很大的内存开销。因为 String 对象建立之后不能再改变,所以对于每一个不同的字符串,都需要一个 String 对象来表示。这时,应该考虑使用StringBuffer/StringBuilder类,它允许修改,而不是每个不同的字符串都要生成一个新的对象。并且,这两种类的对象转换十分容易。同时,我们还可以知道,如果要使用内容相同的字符串,不必每次都 new 一个 String


5.String 类常用方法


方法

解释说明

char charAt(int index)

返回指定索引处的 char 值。

boolean contains(CharSequence s)

当且仅当此字符串包含指定的 char 值序列时,返回 true

boolean endsWith(String suffix)

测试此字符串是否以指定的后缀结束。

boolean equals(Object anObject)

将此字符串与指定的对象比较。

boolean equalsIgnoreCase(String anotherString)

将此 String 与另一个 String 比较,不考虑大小写。

byte[] getBytes()

使用平台的默认字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。

int indexOf(String str)

返回指定子字符串在此字符串中第一次出现处的索引。

int lastIndexOf(String str)

返回指定子字符串在此字符串中最右边出现处的索引。

int length()

返回此字符串的长度。

String replaceAll(String regex, String replacement)

使用给定的 replacement 替换此字符串所有匹配给定的正则表达式的子字符串。

String[] split(String regex)

根据给定正则表达式的匹配拆分此字符串。

boolean startsWith(String prefix)

测试此字符串是否以指定的前缀开始。

String substring(int beginIndex)

返回一个新的字符串,它是此字符串的一个子字符串。

String substring(int beginIndex, int endIndex)

返回一个新字符串,它是此字符串的一个子字符串。

char[] toCharArray()

将此字符串转换为一个新的字符数组。

String toLowerCase()

使用默认语言环境的规则将此 String 中的所有字符都转换为小写。

String toUpperCase()

使用默认语言环境的规则将此 String 中的所有字符都转换为大写。

String trim()

返回字符串的副本,忽略前导空白和尾部空白。


6.String、StringBuffer、StringBuilder的区别?


可变不可变

String:字符串常量,在修改时不会改变自身;若修改,等于重新生成新的字符串对象。

StringBuffer:在修改时会改变对象自身,每次操作都是对 StringBuffer 对象本身进行修改,不是生成新的对象;使用场景:对字符串经常改变情况下,主要方法:append()insert()等。

线程是否安全

String:对象定义后不可变,线程安全。

StringBuffer:是线程安全的(对调用方法加入同步锁),执行效率较慢,适用于多线程下操作字符串缓冲区大量数据。

StringBuilder:是线程不安全的,适用于单线程下操作字符串缓冲区大量数据。

共同点

StringBuilderStringBuffer有公共父类 AbstractStringBuilder(抽象类)

StringBuilderStringBuffer 的方法都会调用AbstractStringBuilder 中的公共方法,如super.append(...)。只是 StringBuffer 会在方法上加 synchronized 关键字,进行同步。最后,如果程序不是多线程的,那么使用StringBuilder 效率高于 StringBuffer

可以在源码中看到StringBuilderStringBuffer


7.short s1 = 1; s1 = s1 + 1; 有错吗?short s1 = 1; s1 += 1 有错吗?


答案:前者错!后者对!

对于 short s1 = 1; s1 = s1 + 1;由于 1 int 类型,因此 s1+1 运算结果也是 int 型,需要强制转换类型才能赋值给 short 型。

short s1 = 1; s1 += 1;可以正确编译,因为 s1+= 1; 相当于 s1 = (short)(s1 + 1); 其中有隐含的强制类型转换。

short s1=1;
s1= (short) (s1+1);
short s2=1;
s2+=1;


8.int和Integer有什么区别?


java是一个完全面向对象编程语言,但是为了编程的方便还是引入了基本数据类型,为了能够将这些基本数据类型当成对象操作,Java 为每一个基本数据类型都引入了对应的包装类型(wrapper class),int 的包装类就是Integer,从 Java 5 开始引入了自动装箱/拆箱机制,使得二者可以相互转换。

java为每个原始类型提供了包装类型:(再说一遍!!!)

基本数据类型: booleancharbyteshortintlongfloatdouble

包装类型:BooleanCharacterByteShortIntegerLongFloatDouble

package com.szh.java01;
/**
 * byte short int long float double char boolean
 */
public class Test01 {
    public static void main(String[] args) {
        /**
         * 900是基本数据类型int
         * x是Integer包装类型
         * 基本数据类型  --->  包装类型(自动转换,自动装箱机制)
         */
        Integer x=500;
        System.out.println(x);
        /**
         * x是Integer包装类型
         * y是基本数据类型int
         * 包装类型  --->  基本数据类型(自动转换,自动拆箱机制)
         */
        int y=x;
        System.out.println(y);
        /**
         * 这里程序正常执行不报错,是因为自动拆箱机制
         * + 号两边要求是基本数据类型,z是包装类,所以这里为进行自动拆箱机制进行自动转换
         */
        Integer z=1000;
        System.out.println(z+1);
    }
}

9.下面Integer 类型的数值比较输出的结果为?


public class Test{
    public static void main(String[] args) {
        Integer f1 = 100, f2 = 100, f3 = 150, f4 = 150;
        System.out.println(f1 == f2);
        System.out.println(f3 == f4);
    }
}

答案:第一行输出 true,第二行输出 false

首先需要注意的是 f1f2f3f4 四个变量都是 Integer 对象引用,所以下面的==运算比较的不是值而是引用。这里就要提到自动装箱机制了,那么自动装箱的本质是什么呢?当我们给一个Integer 对象赋一个 int 值的时候,会调用 Integer 类的静态方法 valueOf,如果看看valueOf的源代码就知道发生了什么。

public static Integer valueOf(int i) {
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}

IntegerCache Integer 的内部类,其代码如下所示:

private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer[] cache;
        static Integer[] archivedCache;
        static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    h = Math.max(parseInt(integerCacheHighPropValue), 127);
                    // Maximum array size is Integer.MAX_VALUE
                    h = Math.min(h, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                    // If the property cannot be parsed into an int, ignore it.
                }
            }
            high = h;
            // Load IntegerCache.archivedCache from archive, if possible
            VM.initializeFromArchive(IntegerCache.class);
            int size = (high - low) + 1;
            // Use the archived cache if it exists and is large enough
            if (archivedCache == null || size > archivedCache.length) {
                Integer[] c = new Integer[size];
                int j = low;
                for(int i = 0; i < c.length; i++) {
                    c[i] = new Integer(j++);
                }
                archivedCache = c;
            }
            cache = archivedCache;
            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }
        private IntegerCache() {}
    }

这一大堆源码简单地说就是:如果整型字面量的值在-128 127 之间,那么不会 new 新的 Integer 对象,而是直接引用常量池中的Integer对象,所以上面的面试题中f1==f2的结果是 true,而f3==f4 的结果是false


10.数据类型之间如何进行转换?


字符串如何转基本数据类型?

调用基本数据类型对应的包装类中的方法parseXXX(String)valueOf(String)即可返回相应基本类型。

基本数据类型如何转字符串?

一种方法是将基本数据类型与空字符串("")连接(+)即可获得其所对应的字符串;另一种方法是调用 String类中的 valueOf()方法返回相应字符串。

public class IntegerAndString {
    public static void main(String[] args) {
        // String ---> int
        String s1="100";
        int i1=Integer.parseInt(s1);
        System.out.println("String ---> int " + i1);
        // int ---> String
        String s2=i1 + "";
        System.out.println("int ---> String " + s2 + 1);
        // int ---> Integer (自动装箱)
        Integer x=1000;
        System.out.println("x = " + x);
        // Integer ---> int (自动拆箱)
        int y=x;
        System.out.println("y = " + y);
        // String ---> Integer
        Integer k=Integer.valueOf("123");
        System.out.println("String ---> Integer " + k);
        // Integer ---> String
        String str=String.valueOf(k);
        System.out.println("Integer ---> String " + str);
    }
}

目录
打赏
0
0
0
0
85
分享
相关文章
# 【Java全栈学习笔记-U1-day02】变量+数据类型+运算符
本篇笔记主要围绕Java全栈学习的第二天内容展开,涵盖了变量、数据类型、运算符以及Scanner类的应用。首先介绍了变量的概念与命名规范,以及如何定义和使用变量;接着详细讲解了Java中的基本数据类型,包括整型、浮点型、字符型、布尔型等,并通过实例演示了数据类型的运用。随后,深入探讨了各类运算符(赋值、算术、关系、逻辑)及其优先级,帮助理解表达式的构成。最后,介绍了如何利用Scanner类实现用户输入功能,并通过多个综合示例(如计算圆面积、购物打折、变量交换及银行利息计算)巩固所学知识。完成相关作业将进一步加深对这些基础概念的理解与实践能力。
23 12
java面试-基础语法与面向对象
本文介绍了 Java 编程中的几个核心概念。首先,详细区分了方法重载与重写的定义、发生阶段及规则;其次,分析了 `==` 与 `equals` 的区别,强调了基本类型和引用类型的比较方式;接着,对比了 `String`、`StringBuilder` 和 `StringBuffer` 的特性,包括线程安全性和性能差异;最后,讲解了 Java 异常机制,包括自定义异常的实现以及常见非检查异常的类型。这些内容对理解 Java 面向对象编程和实际开发问题解决具有重要意义。
41 15
课时9:Java数据类型划分(数据类型划分)
课时9介绍了Java的数据类型划分及其操作。Java数据类型分为基本数据类型和引用数据类型。基本数据类型包括八种:数值型(整形、浮点型)、字符型和布尔型;引用数据类型有数组、类和接口。重点讲解了每种数据类型的默认值、适用场景及选择原则,如整数用int,小数用double,日期时间用Long等。掌握这些内容对编程至关重要。
课时9:Java数据类型划分(数据类型划分)
|
26天前
|
课时12:Java数据类型划分(字符型)
课时12介绍了Java中的字符型数据(char),包括字符的定义、与int类型的转换及应用。通过范例展示了如何观察字符、编码转换、大小写字母互转、保存中文字符,以及Java中字符处理的灵活性。字符型使用单引号声明,可与int互相转换,支持Unicode编码,能表示任何文字,包括中文。虽然实际开发中直接操作字符型的情况不多,但理解其特性有助于更好地编程。
48 17
|
25天前
|
java变量与数据类型:整型、浮点型与字符类型
### Java数据类型全景表简介 本文详细介绍了Java的基本数据类型和引用数据类型,涵盖每种类型的存储空间、默认值、取值范围及使用场景。特别强调了`byte`、`int`、`long`、`float`、`double`等基本类型在不同应用场景中的选择与优化,如文件流处理、金融计算等。引用数据类型部分则解析了`String`、数组、类对象、接口和枚举的内存分配机制。
55 15
Java社招面试中的高频考点:Callable、Future与FutureTask详解
大家好,我是小米。本文主要讲解Java多线程编程中的三个重要概念:Callable、Future和FutureTask。它们在实际开发中帮助我们更灵活、高效地处理多线程任务,尤其适合社招面试场景。通过 Callable 可以定义有返回值且可能抛出异常的任务;Future 用于获取任务结果并提供取消和检查状态的功能;FutureTask 则结合了两者的优势,既可执行任务又可获取结果。掌握这些知识不仅能提升你的编程能力,还能让你在面试中脱颖而出。文中结合实例详细介绍了这三个概念的使用方法及其区别与联系。希望对大家有所帮助!
210 60
|
25天前
|
课时14:Java数据类型划分(初见String类)
课时14介绍Java数据类型,重点初见String类。通过三个范例讲解:观察String型变量、&quot;+&quot;操作符的使用问题及转义字符的应用。String不是基本数据类型而是引用类型,但使用方式类似基本类型。课程涵盖字符串连接、数学运算与字符串混合使用时的注意事项以及常用转义字符的用法。
|
25天前
|
课时13:Java数据类型划分(布尔型)
观察布尔型的操作。布尔是一位数学家的名字,这个布尔发明了两个单词:True、False(一个表示真一个表示假)。一般布尔类型都只表示逻辑的计算结果。
|
26天前
|
课时11:Java数据类型划分(浮点类型)
课时11介绍了Java中的浮点数据类型。主要内容包括:1. 定义小数,默认使用Double类型;2. 定义Float变量,需在数值后加&quot;F&quot;或&quot;f&quot;进行强制转换;3. 观察不同类型计算结果,如Int型除法会丢失精度,需至少包含一个Double或Float类型以确保准确性。总结指出,在复杂计算中推荐使用Double类型以避免精度损失。
|
26天前
|
课时10:Java数据类型划分(整型类型)
本文主要围绕Java中整型数据展开,详细讲解整型变量、常量的概念,整型数据运算规则,包括数据溢出问题及解决方法,数据类型转换(自动转换与强制转换)的原理和注意事项,同时介绍了整型数据默认值的相关知识,以及byte数据类型与int数据类型的关系和使用场景,帮助读者全面掌握Java整型数据的相关内容。

热门文章

最新文章