Java数据类型【Java面试题】

x33g5p2x  于2021-12-30 转载在 Java  
字(4.5k)|赞(0)|评价(0)|浏览(338)

1、Java 的基本数据类型都有哪些各占几个字节?

按照口诀记忆:

● 数据类型:byte short int long float double boolean char

● 占用字节数:12484812(byte对应1,short对应2,以此类推)

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

通过JDK源代码可以看到,Stirng是class,是引用类型,底层用 char 数组实现的。

3、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);

其中有隐含的强制类型转换。

4、int和Integer有什么区别?

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

java 为每个原始类型提供了包装类型:

● 基本数据类型: boolean,char,byte,short,int,long,float,double

● 包装类型:Boolean,Character,Byte,Short,Integer,Long,Float,Double

5、下面 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。

首先需要注意的是 f1、f2、f3、f4 四个变量都是 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 {
        // high value may be configured by property
        int h = 127;
        String integerCacheHighPropValue =
            sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
        if (integerCacheHighPropValue != null) {
            try {
                int i = parseInt(integerCacheHighPropValue);
                i = Math.max(I, 127);
                // Maximum array size is Integer.MAX_VALUE
                h = Math.min(I, Integer.MAX_VALUE - (-low) -1);
            } catch( NumberFormatException nfe) {
                // If the property cannot be parsed into an int, ignore it.
            }
        }
        high = h;

        cache = new Integer[(high - low) + 1];
        int j = low;
        for(int k = 0; k < cache.length; k++)
            cache[k] = new Integer(j++);

        // 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。

6、String 类常用方法?

| <br>方法<br> | <br>解释说明<br> |
| <br>char charAt(int index)<br> | <br>返回指定索引处的 char 值。<br> |
| <br>boolean contains(CharSequence s)<br> | <br>当且仅当此字符串包含指定的 char 值序列时,返回 true。<br> |
| <br>boolean endsWith(String suffix)<br> | <br>测试此字符串是否以指定的后缀结束。<br> |
| <br>boolean equals(Object anObject)<br> | <br>将此字符串与指定的对象比较。<br> |
| <br>boolean equalsIgnoreCase(String anotherString)<br> | <br>将此 String 与另一个 String 比较,不考虑大小写。<br> |
| <br>byte[] getBytes()<br> | <br>使用平台的默认字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。<br> |
| <br>int indexOf(String str)<br> | <br>返回指定子字符串在此字符串中第一次出现处的索引。<br> |
| <br>int lastIndexOf(String str)<br> | <br>返回指定子字符串在此字符串中最右边出现处的索引。<br> |
| <br>int length()<br> | <br>返回此字符串的长度。<br> |
| <br>String replaceAll(String regex, String replacement)<br> | <br>使用给定的 replacement 替换此字符串所有匹配给定的正则表达式的子字符串。<br> |
| <br>String[] split(String regex)<br> | <br>根据给定正则表达式的匹配拆分此字符串。<br> |
| <br>boolean startsWith(String prefix)<br> | <br>测试此字符串是否以指定的前缀开始。<br> |
| <br>String substring(int beginIndex)<br> | <br>返回一个新的字符串,它是此字符串的一个子字符串。<br> |
| <br>String substring(int beginIndex, int endIndex)<br> | <br>返回一个新字符串,它是此字符串的一个子字符串。<br> |
| <br>char[] toCharArray()<br> | <br>将此字符串转换为一个新的字符数组。<br> |
| <br>String toLowerCase()<br> | <br>使用默认语言环境的规则将此 String 中的所有字符都转换为小写。<br> |
| <br>String toUpperCase()<br> | <br>使用默认语言环境的规则将此 String 中的所有字符都转换为大写。<br> |
| <br>String trim()<br> | <br>返回字符串的副本,忽略前导空白和尾部空白。<br> |

7、String、StringBuffer、StringBuilder的区别?

● 可变不可变

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

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

● 线程是否安全

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

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

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

● 共同点

StringBuilder与StringBuffer有公共父类 AbstractStringBuilder(抽象类)。

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

8、数据类型之间的转换?

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

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

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

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

相关文章