014Java基础之常用类:包装类、String、StringBuffer、StringBuilder

1、包装类

1.1、包装类的分类

(1)针对八种基本数据类型相应的引用类型—包装类
(2)有了类的特点,就可以调用类中的方法。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

1.2、包装类和基本数据的转换

JDK5之前采用手动装箱和手动拆箱方式,JDK5之后底层帮我们实现了自动装箱和拆箱。装箱:基本数据类型到包装类;拆箱:包装类到基本数据类型

package com.francis.wrapper;

/**
 * @author Francis
 * @create 2021-07-17 10:16
 */
public class Integer01 {
    public static void main(String[] args) {
//演示 int <--> Integer 的装箱和拆箱
//jdk5 前是手动装箱和拆箱
//手动装箱 int->Integer
        int n1 = 100;
        Integer integer = new Integer(n1);
        Integer integer1 = Integer.valueOf(n1);
//手动拆箱
//Integer -> int
        int i = integer.intValue();
//jdk5 后,就可以自动装箱和自动拆箱
        int n2 = 200;
//自动装箱 int->Integer
        Integer integer2 = n2; //底层使用的是 Integer.valueOf(n2)
//自动拆箱 Integer->int
        int n3 = integer2; //底层仍然使用的是 intValue()方法
    }
}

这里需要注意一点:自动装箱过程中,调用的是Integer的valueOf方法,这个方法会根据数值的范围(-128~127)来决定是创建一个对象,还是从缓存中取出一个对象
金典面试题
在这里插入图片描述

但这里没有向下转型,不应该是输出一个地址吗?疑惑!

1.3、包装类型和 String 类型的相互转换
package com.francis.wrapper;

/**
 * @author Francis
 * @create 2021-07-17 11:29
 */
public class WrapperVSString {
    public static void main(String[] args) {
//包装类(Integer)->String
        Integer i = 100;//自动装箱
//方式 1
        String str1 = i + "";
        System.out.println(str1);
//方式 2
        String str2 = i.toString();
        System.out.println(str2);
//方式 3
        String str3 = String.valueOf(i);
        System.out.println(str3);
//String -> 包装类(Integer)
        String str4 = "12345";
        Integer i2 = Integer.parseInt(str4);//使用到自动装箱
        System.out.println(i2);
        Integer i3 = new Integer(str4);//构造器
        System.out.println(i3);
        System.out.println("ok~~");
    }
}
1.4、Integer 类和 Character 类的常用方法
package com.francis.wrapper;

/**
 * @author Francis
 * @create 2021-07-17 11:33
 */
public class WrapperMethod {
    public static void main(String[] args) {
        System.out.println(Integer.MIN_VALUE); //返回最小值
        System.out.println(Integer.MAX_VALUE);//返回最大值
        System.out.println(Character.isDigit('a'));//判断是不是数字
        System.out.println(Character.isLetter('a'));//判断是不是字母
        System.out.println(Character.isUpperCase('a'));//判断是不是大写
        System.out.println(Character.isLowerCase('a'));//判断是不是小写
        System.out.println(Character.isWhitespace('a'));//判断是不是空格
        System.out.println(Character.toUpperCase('a'));//转成大写
        System.out.println(Character.toLowerCase('A'));//转成小写
    }
}
1.5、金典面试题
package com.francis.wrapper;

/**
 * @author Francis
 * @create 2021-07-17 11:41
 */
public class WrapperExercise01 {
    public static void main(String[] args) {
        Integer i = new Integer(1);
        Integer j = new Integer(1);
        System.out.println(i == j); //False
//==========================================================================
//所以,这里主要是看范围 -128 ~ 127 就是直接返回
/*
//1. 如果 i 在 IntegerCache.low(-128)~IntegerCache.high(127),就直接从数组返回
//2. 如果不在 -128~127,就直接 new Integer(i)
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
*/
        Integer m = 1; //底层 Integer.valueOf(1); -> 阅读源码
        Integer n = 1;//底层 Integer.valueOf(1);
        System.out.println(m == n); //T
//所以,这里主要是看范围 -128 ~ 127 就是直接返回
//,否则,就 new Integer(xx);
        Integer x = 128;//底层 Integer.valueOf(1);
        Integer y = 128;//底层 Integer.valueOf(1);
        System.out.println(x == y);//False
    }

}

牛逼!!!!!

package com.francis.wrapper;

/**
 * @author Francis
 * @create 2021-07-17 12:07
 */
public class IntegerExercise01 {
    public static void main(String[] args) {
//示例一
        Integer i1 = new Integer(127);
        Integer i2 = new Integer(127);
        System.out.println(i1 == i2);//F
//示例二
        Integer i3 = new Integer(128);
        Integer i4 = new Integer(128);
        System.out.println(i3 == i4);//F
        //=======new的肯定是不同的对象=================
//示例三
        Integer i5 = 127;//底层 Integer.valueOf(127)
        Integer i6 = 127;//-128~127
        System.out.println(i5 == i6); //T
//示例四
        Integer i7 = 128;
        Integer i8 = 128;
        System.out.println(i7 == i8);//F
//示例五
        Integer i9 = 127; //Integer.valueOf(127)
        Integer i10 = new Integer(127);
        System.out.println(i9 == i10);//F
//示例六
        Integer i11=127;
        int i12=127;
//只有有基本数据类型,判断的是值是否相同
        System.out.println(i11==i12); //T
//示例七
        Integer i13=128;
        int i14=128;
        System.out.println(i13==i14);//T
    }
}

2、String类

2.1、概述

(1)String对象用于保存字符串,也就是一组字符序列,底层是char数组
(2)字符串常量对象是用双引号括起来的字符序列,例如:“hello world”、“12.34”、"你好"等
(3)字符串的字符使用unicode编码,一个字符(不管是中文还是英文)占两个字节
(4)String类有很多重载的构造器
在这里插入图片描述

package com.francis.string;

/**
 * @author Francis
 * @create 2021-07-17 17:43
 */
public class String01 {
    public static void main(String[] args) {
//1.String 对象用于保存字符串,也就是一组字符序列
//2. "jack" 字符串常量, 双引号括起的字符序列
//3. 字符串的字符使用 Unicode 字符编码,一个字符(不区分字母还是汉字)占两个字节
//4. String 类有很多构造器,构造器的重载
// 常用的有 String s1 = new String(); //
//String s2 = new String(String original);
//String s3 = new String(char[] a);
//String s4 = new String(char[] a,int startIndex,int count)
//String s5 = new String(byte[] b)
//5. String 类实现了接口 Serializable【String 可以串行化:可以在网络传输】
// 接口 Comparable [String 对象可以比较大小]
//6. String 是 final 类,不能被其他的类继承
//7. String 有属性 private final char value[]; 用于存放字符串内容
//8. 一定要注意:value 是一个 final 类型, 不可以修改(需要功力):即 value 不能指向
// 新的地址,但是单个字符内容是可以变化
        String name = "jack";
        name = "tom";
        final char[] value = {'a','b','c'};
        char[] v2 = {'t','o','m'};
        value[0] = 'H';
//value = v2; 不可以修改 value 地址
    }
}

在这里插入图片描述

2.2、创建 String 对象的两种方式
2.2.1、直接复制,String s = “hello”

该方式首先查看常量池是否有"hello"数据空间,如果有,s直接指向该地址,如果没有,则新建一个数据空间,然后s再指向该地址。s最终指向的是常量池的空间地址

2.2.2、调用构造器 String s = new String(“hello”)

该方式先在堆中创建String对象空间,里面维护了value属性,指向常量池的"hello"空间,如果常量池没有"hello",新建一个"hello",如果有,直接通过value指向,s最终指向的是堆中的空间地址
在这里插入图片描述

2.3、字符串的特性

(1)String是一个final的,代表不可变的字符序列
(2)字符串是不可变的,一个字符串对象一旦被匹配,其内容是不可变的

2.3.1、金典面试题

(1)
在这里插入图片描述
在这里插入图片描述
(2)
在这里插入图片描述
(3)
在这里插入图片描述

在这里插入图片描述
(4)在这里插入图片描述

2.4、String 类的常见方法

在这里插入图片描述

package com.francis.string;

/**
 * @author Francis
 * @create 2021-07-18 12:50
 */
public class String02 {
    public static void main(String[] args) {
//1. equals 前面已经讲过了. 比较内容是否相同,区分大小写
        String str1 = "hello";
        String str2 = "Hello";
        System.out.println(str1.equals(str2));//
// 2.equalsIgnoreCase 忽略大小写的判断内容是否相等
        String username = "johN";
        if ("john".equalsIgnoreCase(username)) {
            System.out.println("Success!");
        } else {
            System.out.println("Failure!");
        }
// 3.length 获取字符的个数,字符串的长度
        System.out.println("hello".length());
// 4.indexOf 获取子字符(串)在字符串对象中第一次出现的索引,索引从 0 开始,如果找不到,返回-1
        String s1 = "wer@terwe@g";
        int index = s1.indexOf('@');
        System.out.println(index);// 3
        System.out.println("weIndex=" + s1.indexOf("we"));//0
// 5.lastIndexOf 获取子字符(串)在字符串中最后一次出现的索引,索引从 0 开始,如果找不到,返回-1
        s1 = "wer@terwe@g@";
        index = s1.lastIndexOf('@');
        System.out.println(index);//11
        System.out.println("ter 的位置=" + s1.lastIndexOf("ter"));//4
// 6.substring 截取指定范围的子串
        String name = "hello,张三";
//下面 name.substring(6) 从索引 6 开始截取后面所有的内容
        System.out.println(name.substring(6));//截取后面的字符
        System.out.println(name.substring(2,5));//llo
    }
}

在这里插入图片描述

package com.francis.string;

/**
 * @author Francis
 * @create 2021-07-18 12:55
 */
public class String03 {
    public static void main(String[] args) {
// 1.toUpperCase 转换成大写
        String s = "heLLo";
        System.out.println(s.toUpperCase());//HELLO
// 2.toLowerCase
        System.out.println(s.toLowerCase());//hello
// 3.concat 拼接字符串
        String s1 = "宝玉";
        s1 = s1.concat("林黛玉").concat("薛宝钗").concat("together");
        System.out.println(s1);//宝玉林黛玉薛宝钗 together
// 4.replace 替换字符串中的字符
        s1 = "宝玉 and 林黛玉 林黛玉 林黛玉";
//在 s1 中,将 所有的 林黛玉 替换成薛宝钗
//  s1.replace() 方法执行后,返回的结果才是替换过的. // 注意对 s1 没有任何影响
        String s11 = s1.replace("宝玉", "jack");
        System.out.println(s1);//宝玉 and 林黛玉 林黛玉 林黛玉
        System.out.println(s11);//jack and 林黛玉 林黛玉 林黛玉
// 5.split 分割字符串, 对于某些分割字符,我们需要 转义比如 | \\等
        String poem = "锄禾日当午,汗滴禾下土,谁知盘中餐,粒粒皆辛苦";
// 1. 以 , 为标准对 poem 进行分割 , 返回一个数组
// 2. 在对字符串进行分割时,如果有特殊字符,需要加入 转义符 \
        String[] split = poem.split(",");
        poem = "E:\\aaa\\bbb";
        split = poem.split("\\\\");
        System.out.println("==分割后内容===");
        for (int i = 0; i < split.length; i++) {
            System.out.println(split[i]);
        }
// 6.toCharArray 转换成字符数组
                s = "happy";
        char[] chs = s.toCharArray();
        for (int i = 0; i < chs.length; i++) {
            System.out.println(chs[i]);
        }
// 7.compareTo 比较两个字符串的大小,如果前者大,
// 则返回正数,后者大,则返回负数,如果相等,返回 0
// (1) 如果长度相同,并且每个字符也相同,就返回 0
// (2) 如果长度相同或者不相同,但是在进行比较时,可以区分大小
// 就返回 if (c1 != c2) {
// return c1 - c2;
// }
// (3) 如果前面的部分都相同,就返回 str1.len - str2.len
        String a = "jcck";// len = 3
        String b = "jack";// len = 4
        System.out.println(a.compareTo(b)); // 返回值是 'c' - 'a' = 2 的值
// 8.format 格式字符串
        /* 占位符有:
         * %s 字符串 %c 字符 %d 整型 %.2f 浮点型
         *
         */
        String name = "john";
        int age = 10;
        double score = 56.857;
        char gender = '男';
//将所有的信息都拼接在一个字符串.
 String info = "我的姓名是" + name + "年龄是" + age + ",成绩是" + score + "性别是" + gender + "。希望大家喜欢我!";
        System.out.println(info);
//1. %s , %d , %.2f %c 称为占位符
//2. 这些占位符由后面变量来替换
//3. %s 表示后面由 字符串来替换
//4. %d 是整数来替换
//5. %.2f 表示使用小数来替换,替换后,只会保留小数点两位, 并且进行四舍五入的处理
//6. %c 使用 char 类型来替换
        String formatStr = "我的姓名是%s 年龄是%d,成绩是%.2f 性别是%c.希望大家喜欢我!";
        String info2 = String.format(formatStr, name, age, score, gender);
        System.out.println("info2=" + info2);
    }
}

3、StringBuffer 类

3.1、概述

(1)StringBuffer代表可变的字符序列,可以对字符串的内容进行增删
(2)StringBuffer的很多方法与String相同,但StringBuffer是可变长度的
(3)StringBuffer是一个容器
在这里插入图片描述

package com.francis.stringbuffer;

/**
 * @author Francis
 * @create 2021-07-18 19:59
 */
public class StringBuffer01 {
    public static void main(String[] args) {
//1. StringBuffer 的直接父类 是 AbstractStringBuilder
//2. StringBuffer 实现了 Serializable, 即 StringBuffer 的对象可以串行化
//3. 在父类中 AbstractStringBuilder 有属性 char[] value,不是 final,并且该数组是存放在堆中,而不是常量池中
// 该 value 数组存放 字符串内容,因此存放在堆中的
//4. StringBuffer 是一个 final 类,不能被继承
//5. 因为 StringBuffer 字符内容是存在 char[] value, 所有在变化(增加/删除)
// 不用每次都更换地址(即不是每次创建新对象), 所以效率高于 String
        StringBuffer stringBuffer = new StringBuffer("hello");
    }
}
3.2、String VS StringBuffer

(1)String保存的是字符串常量,里面的值不能修改,即value属性的地址是不变的,每次String更新实际上是产生了一个新的对象,然后对象的引用重新指向新产生的对象,这个新对象的value引用重新指向新的常量池地址,效率低下(原因是private final char[] value)
(2)StringBuffer保存的是字符串变量,里面的值可以修改,即StringBuffer对象的属性的value属性的引用可以修改,并且value属性的内容也可以修改
在这里插入图片描述

3.3StringBuffer的构造器

在这里插入图片描述

3.4、String与StringBuffer互换
package com.francis.stringbuffer;

/**
 * @author Francis
 * @create 2021-07-18 21:09
 */
public class StringAndStringBuffer {
    public static void main(String[] args) {
//看 String——>StringBuffer
        String str = "hello tom";
//方式 1 使用构造器
//注意: 返回的才是 StringBuffer 对象,对 str 本身没有影响
        StringBuffer stringBuffer = new StringBuffer(str);
//方式 2 使用的是 append 方法
        StringBuffer stringBuffer1 = new StringBuffer();
        stringBuffer1 = stringBuffer1.append(str);
//看看 StringBuffer ->String
        StringBuffer stringBuffer3 = new StringBuffer("hello Beijing");
//方式 1 使用 StringBuffer 提供的 toString 方法
        String s = stringBuffer3.toString();
//方式 2: 使用构造器来搞定
        String s1 = new String(stringBuffer3);
    }

}
3.5、StringBuffer的常用方法
package com.francis.stringbuffer;

/**
 * @author Francis
 * @create 2021-07-18 21:18
 */
public class StringBufferMethod {
    public static void main(String[] args) {
        StringBuffer s = new StringBuffer("hello");
//增
        s.append(',');// "hello,"
        s.append("张三丰");//"hello,张三丰"
        s.append("赵敏").append(100).append(true).append(10.5);//"hello,张三丰赵敏 100true10.5" System.out.println(s);//"hello,张三丰赵敏 100true10.5"
//删
        /*
         * 删除索引为>=start && <end 处的字符
         * 解读: 删除 11~14 的字符 [11, 14)
         */
        s.delete(11, 14);
        System.out.println(s);//"hello,张三丰赵敏 true10.5"
//改
//使用 周芷若 替换 索引 9-11 的字符 [9,11)
        s.replace(9, 11, "周芷若");
        System.out.println(s);//"hello,张三丰周芷若 true10.5"
//查找指定的子串在字符串第一次出现的索引,如果找不到返回-1
        int indexOf = s.indexOf("张三丰");
        System.out.println(indexOf);//6
//插
//在索引为 9 的位置插入 "赵敏",原来索引为 9 的内容自动后移
        s.insert(9, "赵敏");
        System.out.println(s);//"hello,张三丰赵敏周芷若 true10.5"
//长度
        System.out.println(s.length());//22
        System.out.println(s);
    }
}
3.5、练习题

(1)

package com.francis.stringbuffer;

/**
 * @author Francis
 * @create 2021-07-18 21:41
 */
public class StringBufferExercise01 {
    public static void main(String[] args) {
        String str = null;// ok
        StringBuffer sb = new StringBuffer(); //ok
        sb.append(str);//需要看源码 , 底层调用的是 AbstractStringBuilder 的 appendNull
        System.out.println(sb.length());//4
        System.out.println(sb);//null

//下面的构造器,会抛出 NullpointerException
        StringBuffer sb1 = new StringBuffer(str);//看底层源码 super(str.length() + 16);
        System.out.println(sb1);
    }
}

(2

package com.francis.stringbuffer;

/**
 * @author Francis
 * @create 2021-07-18 22:09
 */
public class StringBufferExercise02 {
    public static void main(String[] args) {
/*
输入商品名称和商品价格,要求打印效果示例, 使用前面学习的方法完成:
商品名 商品价格
韩顺平循序渐进学 Java 零基础
第 562页
手机 123,564.59 //比如 价格 3,456,789.88
要求:价格的小数点前面每三位用逗号隔开, 在输出。
思路分析
1. 定义一个 Scanner 对象,接收用户输入的 价格(String)
2. 希望使用到 StringBuffer 的 insert ,需要将 String 转成 StringBuffer
3. 然后使用相关方法进行字符串的处理
代码实现
*/
//new Scanner(System.in)
        String price = "8123564.59";
        StringBuffer sb = new StringBuffer(price);
//先完成一个最简单的实现 123,564.59
//找到小数点的索引,然后在该位置的前 3 位,插入,即可
// int i = sb.lastIndexOf(".");
// sb = sb.insert(i - 3, ",");
//上面的两步需要做一个循环处理,才是正确的
        for (int i = sb.lastIndexOf(".") - 3; i > 0; i -= 3) {
            sb = sb.insert(i, ",");
        }
        System.out.println(sb);//8,123,564.59
        
    }
}

4、StringBuilder

4.1、概述

(1)一个可变的字符序列,此类提供与StringBuffer兼容的API,但不保证同步。该类被设计用作StringBuffer的一个简易替换,用在字符串缓冲区被单个线程使用的时候。如果可能,建议先使用该类,因为在大多数实现中,它比StringBuffer要快
(2)在StringBuilder上的主要操作时append和insert方法,可重载这些方法,以接收任意类型的数据
在这里插入图片描述
(3)与StringBuffer的区别在于StringBuilder的所有方法都是线程不安全的,以append方法为例
在这里插入图片描述

package com.francis.stringbuilder;

/**
 * @author Francis
 * @create 2021-07-18 22:35
 */
public class StringBuilder01 {
    public static void main(String[] args) {
//1. StringBuilder 继承 AbstractStringBuilder 类
//2. 实现了 Serializable ,说明 StringBuilder 对象是可以串行化(对象可以网络传输,可以保存到文件)
//3. StringBuilder 是 final 类, 不能被继承
//4. StringBuilder 对象字符序列仍然是存放在其父类 AbstractStringBuilder 的 char[] value;
// 因此,字符序列是堆中
//5. StringBuilder 的方法,没有做互斥的处理,即没有 synchronized 关键字,因此在单线程的情况下使用
// StringBuilder
        StringBuilder stringBuilder = new StringBuilder();
    }
}

5、String、StringBuffer、StringBuilder的联系与区别

(1)String是不可变字符序列,效率低,但复用性高,是线程安全
(2)StringBuffer、StringBuilder非常类似,均代表可变的字符序列,而且方法也一样
(3)StringBuffer:可变的字符序列,增删的效率较高、几乎所有的方法都采用synchronized关键字修饰,是线程安全的
(4)StringBuilder:可变的字符序列,效率最高,但是方法都没有synchronized关键字修饰,是线程不安全的
(5)String的使用说明:
String s = “a”;//创建了一个字符串
s += “b”;//实际上原来的“a”字符串对象已经丢弃,现又产生了一个新的字符串对象"ab",如果多次执行这种改变字符串内容的操作(实际不是原来的对象了),会导致大量副本字符串对象留在内存中,降低效率,如果这样的操作放到循环中,会极大的影响程序的性能。如果我们对String进行大量的修改,不要使用String

5.1、三者效率测试
public static void main(String[] args) {
long startTime = 0L;
long endTime = 0L;
StringBuffer buffer = new StringBuffer("");
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//StringBuffer 拼接 20000 次
buffer.append(String.valueOf(i));
}
endTime = System.currentTimeMillis();
System.out.println("StringBuffer 的执行时间:" + (endTime - startTime));
StringBuilder builder = new StringBuilder("");
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//StringBuilder 拼接 20000 次
韩顺平循序渐进学 Java 零基础
第 567页
builder.append(String.valueOf(i));
}
endTime = System.currentTimeMillis();
System.out.println("StringBuilder 的执行时间:" + (endTime - startTime));
String text = "";
startTime = System.currentTimeMillis();
for (int i = 0; i < 80000; i++) {//String 拼接 20000
text = text + i;
}
endTime = System.currentTimeMillis();
System.out.println("String 的执行时间:" + (endTime - startTime));
}

在这里插入图片描述

5.2、使用原则

(1)如果字符串存在大量的修改操作,一般使用StringBuffer或StringBuilder
(2)如果字符串存在大量的修改操作,并在单线程情况下,使用StringBuilder
(3)如果字符串存在大量的修改操作,并在多线程情况下,使用StringBuffer
(4)如果字符串很少修改,被多个对象引用,使用String,比如配置信息等。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值