JavaSE学习day8 API的使用和String的内存原理

1.API

1.1API概述

  • 什么是API

    API (Application Programming Interface) :应用程序编程接口

  • java中的API

    指的就是 JDK 中提供的各种功能的 Java类,这些类将底层的实现封装了起来,我们不需要关心这些类是如何实现的,只需要学习这些类如何使用即可,我们可以通过帮助文档来学习这些API如何使用。

    其实之前我们已经学过两个API了,只是不知道叫什么而已,比如Scanner和Random这两个类就是java中比较常用的API。

1.2如何使用API帮助文档

  • 打开帮助文档

  •  最左边那个,没有的同学自行下载。

  • 找到索引选项卡中的输入框

  • 在输入框中输入Random

  • 看类在哪个包下,可知这个类在java.util这个包下。

  • 看类的描述

  • 看构造方法

  • 看成员方法,最左边的那一列表示方法的返回值

知识点回顾:调用方法的时候,如果方法有明确的返回值,我们用变量接受,可以手动完成,也可以用快捷方式完成,快捷键为:Ctrl+Alt+V,这个快捷键前面讲过了,这里是温习。

2.String类

2.1String类概述

String 类代表字符串,Java 程序中的所有字符串文字(例如“abc”)都被实现为此类的实例。也就是说,Java 程序中所有的双引号字符串,都是 String 类的对象String 类在 java.lang 包下,所以使用的时候不需要导包!

2.2String类的特点(看似简单实则比较抽象,第一遍基本不会领悟到精髓)

  • 字符串不可变,它们的值在创建后不能被更改

  • 虽然 String 的值是不可变的,但是它们可以被共享

  • 字符串效果上相当于字符数组( char[] ),但是底层原理是字节数组( byte[] )

2.3String类的构造方法

  • 常用的构造方法

    方法名说明
    public String()创建一个空白字符串对象,不含有任何内容
    public String(char[] chs)根据字符数组的内容,来创建字符串对象
    public String(byte[] bys)根据字节数组的内容,来创建字符串对象
    String s = “abc”;直接赋值的方式创建字符串对象,内容就是abc
  • 示例代码

    public class StringDemo01 {
        public static void main(String[] args) {
            //public String():创建一个空白字符串对象,不含有任何内容
            String s1 = new String();
            System.out.println("s1:" + s1);
    ​
            //public String(char[] chs):根据字符数组的内容,来创建字符串对象
            char[] chs = {'a', 'b', 'c'};
            String s2 = new String(chs);
            System.out.println("s2:" + s2);
    ​
            //public String(byte[] bys):根据字节数组的内容,来创建字符串对象
            byte[] bys = {97, 98, 99};
            String s3 = new String(bys);
            System.out.println("s3:" + s3);//要先转换为对应的ASCII码对应的字符在赋值给该String对象
    ​
            //String s = “abc”; 直接赋值的方式创建字符串对象,内容就是abc
            String s4 = "abc";
            System.out.println("s4:" + s4);
        }
    }

运行结果:

尤其要注意第三行的结果,它是先把字节数组里的十进制数字先转换为对应的ASCII码对应的字符,然后再赋值给String变量其次是注意第一行,我们之前在自己设计定义类的时候,打印这个类创建的对象名的时候打印的是一个地址,而在String类里则不是,它打印的是这个对象名所指的内容String它是一个非常特殊的引用类型。如下:

这是为什么呢?这是因为String类定义的变量可以理解为用于指向字符串对象,然后操作该字符串,它并不是一个地址这个是和自己设计的类最大的区别之一

2.4创建字符串对象两种方式的区别

  • 通过构造方法创建

    通过 new 创建的字符串对象,每一次 new 都会申请一个内存空间,虽然内容相同,但是地址值不同,前面熟悉内存分配就知道怎么理解了。

  • 直接赋值方式创建

    " "方式给出的字符串,只要字符序列相同(顺序和大小写),无论在程序代码中出现几次,JVM 都只会建立一个 String 对象,并在字符串池中维护

上面亮点是什么意思可能有人不是很明白,我举一个例子:

在内存中,虽然a和b创建的字符序列相同,但是在内存中是属于不同的空间,而c和d赋值的字符串序列相同,而且又是通过直接赋值的,因此在内存中只会创建“456”这一个对象。而不会创建两个对象因此c和d这俩变量他们的地址相同。文章最后会将它的底层原理。

常用的字符串方法:

 不用刻意去记这些方法,要用的时候我们直接用API文档查阅即可。

2.5字符串的比较

2.5.1==号的作用

  • 比较基本数据类型:比较的是具体的值

  • 比较引用数据类型:比较的是对象地址值,注意是比较地址!!

2.5.2equals方法的作用

方法介绍

public boolean equals(String s)     比较两个字符串内容是否相同、区分大小写
示例代码

public class StringDemo02 {
    public static void main(String[] args) {
        //构造方法的方式得到对象
        char[] chs = {'a', 'b', 'c'};
        String s1 = new String(chs);
        String s2 = new String(chs);
​
        //直接赋值的方式得到对象
        String s3 = "abc";
        String s4 = "abc";
​
        //比较字符串对象地址是否相同
        System.out.println(s1 == s2);
        System.out.println(s1 == s3);
        System.out.println(s3 == s4);
        System.out.println("--------");
​
        //比较字符串内容是否相同
        System.out.println(s1.equals(s2));
        System.out.println(s1.equals(s3));
        System.out.println(s3.equals(s4));
    }
}

运行结果:

我想把前面内存搞懂的话,这里这个结果是完全没啥问题的。通过查询API文档,我们额外还知道一个字符串的比较方法:equalsIgnoreCase(),将此字符串与指定对象进行比较,忽略大小写比较字符串。只关心字符内将是否一致!

2.6用户登录案例

2.6.1案例需求

已知用户名和密码,请用程序实现模拟用户登录。总共给三次机会,登录之后,给出相应的提示

2.6.2代码实现

/*
    思路:
        1:已知用户名和密码,定义两个字符串表示即可
        2:键盘录入要登录的用户名和密码,用 Scanner 实现
        3:拿键盘录入的用户名、密码和已知的用户名、密码进行比较,给出相应的提示。字符串的内容比较,用equals() 方法实现
        4:用循环实现多次机会,这里的次数明确,采用for循环实现,并在登录成功的时候,使用break结束循环
 */
public class StringTest01 {
    public static void main(String[] args) {
        //已知用户名和密码,定义两个字符串表示即可
        String username = "itheima";
        String password = "czbk";
​
        //用循环实现多次机会,这里的次数明确,采用for循环实现,并在登录成功的时候,使用break结束循环
        for(int i=0; i<3; i++) {
​
            //键盘录入要登录的用户名和密码,用 Scanner 实现
            Scanner sc = new Scanner(System.in);
​
            System.out.println("请输入用户名:");
            String name = sc.nextLine();
​
            System.out.println("请输入密码:");
            String pwd = sc.nextLine();
​
            //拿键盘录入的用户名、密码和已知的用户名、密码进行比较,给出相应的提示。字符串的内容比较,用equals() 方法实现
            if (name.equals(username) && pwd.equals(password)) {
                System.out.println("登录成功");
                break;
            } else {
                if(2-i == 0) {
                    System.out.println("你的账户被锁定,请与管理员联系");
                } else {
                    //2,1,0
                    //i,0,1,2
                    System.out.println("登录失败,你还有" + (2 - i) + "次机会");
                }
            }
        }
    }
}

这个是基础题,大家慢慢练习。

2.7遍历字符串案例

2.7.1案例需求

键盘录入一个字符串,使用程序实现在控制台遍历该字符串

2.7.2代码实现

/*
    思路:
        1:键盘录入一个字符串,用 Scanner 实现
        2:遍历字符串,首先要能够获取到字符串中的每一个字符
            public char charAt(int index):返回指定索引处的char值,字符串的索引也是从0开始的
        3:遍历字符串,其次要能够获取到字符串的长度
            public int length():返回此字符串的长度
            数组的长度:数组名.length
            字符串的长度:字符串对象.length()
        4:遍历字符串的通用格式
 */
public class StringTest02 {
    public static void main(String[] args) {
        //键盘录入一个字符串,用 Scanner 实现
        Scanner sc = new Scanner(System.in);
​
        System.out.println("请输入一个字符串:");
        String line = sc.nextLine();
​
        for(int i=0; i<line.length(); i++) {
            System.out.println(line.charAt(i));
        }
    }
}

注意一下的是String类型输入是nextLine()而不是nextString()。

2.8统计字符次数案例

2.8.1案例需求

键盘录入一个字符串,统计该字符串中大写字母字符,小写字母字符,数字字符出现的次数(不考虑其他字符)

2.8.2代码实现

/*
  思路:
        1:键盘录入一个字符串,用 Scanner 实现
        2:要统计三种类型的字符个数,需定义三个统计变量,初始值都为0
        3:遍历字符串,得到每一个字符
        4:判断该字符属于哪种类型,然后对应类型的统计变量+1
            假如ch是一个字符,我要判断它属于大写字母,小写字母,还是数字,直接判断该字符是否在对应的范围即可
            大写字母:ch>='A' && ch<='Z'
            小写字母: ch>='a' && ch<='z'
            数字: ch>='0' && ch<='9'
        5:输出三种类型的字符个数
 */
public class StringTest03 {
    public static void main(String[] args) {
        //键盘录入一个字符串,用 Scanner 实现
        Scanner sc = new Scanner(System.in);
​
        System.out.println("请输入一个字符串:");
        String line = sc.nextLine();
​
        //要统计三种类型的字符个数,需定义三个统计变量,初始值都为0
        int bigCount = 0;
        int smallCount = 0;
        int numberCount = 0;
​
        //遍历字符串,得到每一个字符
        for(int i=0; i<line.length(); i++) {
            char ch = line.charAt(i);
​
            //判断该字符属于哪种类型,然后对应类型的统计变量+1
            if(ch>='A' && ch<='Z') {
                bigCount++;
            } else if(ch>='a' && ch<='z') {
                smallCount++;
            } else if(ch>='0' && ch<='9') {
                numberCount++;
            }
        }
​
        //输出三种类型的字符个数
        System.out.println("大写字母:" + bigCount + "个");
        System.out.println("小写字母:" + smallCount + "个");
        System.out.println("数字:" + numberCount + "个");
    }
}

本题易错点在那个判断数字的if语句,有的同学可能会写成else if(ch>=0 && ch<=9),这是不对的,数字0和字符0是不一样的。字符0在ASCII的码值为48.理解下面那个代码后就没啥问题了。

2.9字符串拼接案例

2.9.1案例需求

定义一个方法,把 int 数组中的数据按照指定的格式拼接成一个字符串返回,调用该方法,

并在控制台输出结果。例如,数组为 int[] arr = {1,2,3}; ,执行方法后的输出结果为:[1, 2, 3]

2.9.2代码实现

/*
    思路:
        1:定义一个 int 类型的数组,用静态初始化完成数组元素的初始化
        2:定义一个方法,用于把 int 数组中的数据按照指定格式拼接成一个字符串返回。
          返回值类型 String,参数列表 int[] arr
        3:在方法中遍历数组,按照要求进行拼接
        4:调用方法,用一个变量接收结果
        5:输出结果
 */
public class StringTest04 {
    public static void main(String[] args) {
        //定义一个 int 类型的数组,用静态初始化完成数组元素的初始化
        int[] arr = {1, 2, 3};
​
        //调用方法,用一个变量接收结果
        String s = arrayToString(arr);
​
        //输出结果
        System.out.println("s:" + s);
    }
​
    //定义一个方法,用于把 int 数组中的数据按照指定格式拼接成一个字符串返回
    /*
        两个明确:
            返回值类型:String
            参数:int[] arr
     */
    public static String arrayToString(int[] arr) {
        //在方法中遍历数组,按照要求进行拼接
        String s = "";
​
        s += "[";
​
        for(int i=0; i<arr.length; i++) {
            if(i==arr.length-1) {
                s += arr[i];
            } else {
                s += arr[i];
                s += ", ";
            }
        }
​
        s += "]";
​
        return s;
    }
}

注意:在测试类中写方法时方法要加上static关键字修饰,这里前面说过,可能有些忘了,这个知识点后面会系统讲解,反正在测试类中写方法时要加上static这个关键字。而在javabean类(除了测试类就叫javabean类)中,写方法一般是不加static关键字修饰的。不理解没关系,先这样记着

2.10字符串反转案例

2.10.1案例需求

定义一个方法,实现字符串反转。键盘录入一个字符串,调用该方法后,在控制台输出结果

例如,键盘录入 abc,输出结果 cba

2.10.2代码实现

/*
    思路:
        1:键盘录入一个字符串,用 Scanner 实现
        2:定义一个方法,实现字符串反转。返回值类型 String,参数 String s
        3:在方法中把字符串倒着遍历,然后把每一个得到的字符拼接成一个字符串并返回
        4:调用方法,用一个变量接收结果
        5:输出结果
 */
public class StringTest05 {
    public static void main(String[] args) {
        //键盘录入一个字符串,用 Scanner 实现
        Scanner sc = new Scanner(System.in);
​
        System.out.println("请输入一个字符串:");
        String line = sc.nextLine();
​
        //调用方法,用一个变量接收结果
        String s = reverse(line);
​
        //输出结果
        System.out.println("s:" + s);
    }
​
    //定义一个方法,实现字符串反转
    /*
        两个明确:
            返回值类型:String
            参数:String s
     */
    public static String reverse(String s) {
        //在方法中把字符串倒着遍历,然后把每一个得到的字符拼接成一个字符串并返回
        String ss = "";
​
        for(int i=s.length()-1; i>=0; i--) {
            ss += s.charAt(i);
        }
​
        return ss;
    }
}

这是一个很经典的题目,这种编程思维很漂亮,我们要仔细吸收,手动敲出来。

2.11 敏感词过滤

要求:控制台输入一个字符串,若字符串含有敏感词的话,将其替换为“***”,可以建立一个敏感词库。敏感词库包含: JB、SB、垃圾

代码如下:

public class demo05 {
    public static void main(String[] args) {
        Scanner sc=new Scanner(System.in);
        System.out.println("输入你的发言");
        String s=sc.next();
        s = filter(s);
        System.out.println(s);
    }
​
    public static String filter(String s){
        String []a={"JB","SB","垃圾"};//定义敏感词库
        for (int i = 0; i < a.length; i++) {
            s=s.replace(a[i],"***");
        }
        return s;
    }
}

运行结果:

敏感词库可以自己定义,这个代码实现不难,但是在今后开发的论坛项目中,基本都会涉及到敏感词的屏蔽,大家从这里就知道替换的原理是怎么实现的了。

StringBuilder类:

先看一个代码:

发现代码执行得很缓慢,我大概用手机计时了一下,发现在我的电脑上执行力3分钟20秒,时间是相当缓慢的。这是为什么呢?不着急,我们再看一个:

发现执行非常快,几乎是立马执行。这是为什么?这也是我们为什么要使用StringBuilder这个类的原因

StringBuilder可以看作是一个容器,创建之后里面的内容是可变的。而String创建后里面的内容是不可变的。因此用StringBuilder这个类操作能提高效率。因此我们知道,String变量每次的修改其实都是产生并指向了新的字符串对象。 原来的字符串对象都是没有改变的,所以称不可变字符串。

这几个方法是StringBuilder类常用的几个方法。

关于字符串的四个小扩展:(了解)

  1. 字符串存储的内存原理

  2. 这几种赋值方法原理是一样的吗?

  3. 字节码文件加载到方法区,这是之前我们就知道的。

  4. 忘记内存的可以复习一下前面的内容。

  5. 可知 String s = “abc”;直接赋值有如下特点:

    此时字符串abc是存在字符串常量池中的。

    先检查字符串常量池中有没有字符串abc,如果有,不会创建新的,而是直接复用。如果没有abc,才会创建一个新的

    所以,直接赋值的方式,代码简单,而且节约内存文章后面会细讲。

  6. new出来的字符串

    看到new关键字,一定是在堆里面开辟了一个小空间!!

    String s1 = new String(“abc”);

    String s2 = “abc”;

    s1记录的是new出来的在堆里面的地址值

    s2是直接赋值的,所以记录的是字符串常量池中的地址值

  7. ==号比较的到底是什么?

    如果比较的是基本数据类型:比的是具体的数值是否相等。

    如果比较的是引用数据类型:比的是地址值是否相等。

    结论:==只能用于比较基本数据类型。不能比较引用数据类型。

  8. 再看一张图:

  9. 字符串拼接的时候,没有变量参与(重点重点再重点)

    String s = “a” + "b" + "c";

    此时没有变量参与,那么在编译的时候,就已经变成最终的结果了“abc”,这是字符串的常量优化机制!!!

  10. 字符串拼接的时候,有变量参与

    String s1 = “a”;

    String s2 = s1 + “b”;

    在底层,会创建一个StringBuilder对象,再利用append方法,把s1的内容和字符串a都拼接到StringBuilder容器当中,最后再调用toString方法变回一个字符串。在这个过程中,效率比较低,而且内存也有点浪费

  11. 继续看:

  12. 继续:

  13. StringBuilder提高效率原理图

    StringBuilder sb = new StringBuilder(); ​ sb.append("a"); ​ sb.append("b"); ​ sb.append("c"); ​ System.out.println(sb); 我们在代码中,只创建了一个StringBuilder容器对象,把所有的字符串都添加到同一个StringBuilder容器对象当中。而StringBuilder容器对象,里面的内容是可以发生改变的。从头到尾操作的都是同一个,所以效率较高。看下面的面试题:

有人会问打印结果为什么会是false?我们找到API文档找到toString()方法。如下:

转到方法定于定义(快捷键:ctrl+B)

发现toString()方法返回的是一个返回newString。是一个新开辟的内存空间,因此地址是不同的。

接着:

由于没有变量拼接,此时没有变量参与,那么在编译的时候,就已经变成最终的结果了“abc”,这是字符串的常亮优化机制!!!因此地址是一样的。

好了基础阶段就会进入尾声了,通过这段时间的学习,大家对面向对象是不是有了更深的理解呢?

  • 11
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 13
    评论
评论 13
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值