Javase.String 类

news2024/12/22 15:15:18

String 类

  • 【本节目标】
  • 1. String类的重要性
  • 2. 常用方法
    • 2.1 字符串构造
    • 2.2 String对象的比较
    • 2.3 字符串查找
    • 2.4 转化
    • 2.5 字符串替换
    • 2.7 字符串截取
    • 2.8 其他操作方法
    • 2.9 字符串的不可变性
    • 2.10 字符串修改
  • 3. StringBuilder和StringBuffer
    • 3.2 面试题:
  • 4. String类oj
    • 4.1第一个只出现一次的字符
    • 4.2最后一个单词的长度
    • 4.3检测字符串是否为回文

【本节目标】

  1. 认识String类
  2. 了解String类的基本用法
  3. 熟练掌握String类的常见操作
  4. 认识StringBuffer和StringBuilder

1. String类的重要性

在C语言中已经涉及到字符串了,但是在C语言中要表示字符串只能使用字符数组或者字符指针,可以使用标准库提供的字符串系列函数完成大部分操作,但是这种将数据和操作数据方法分离开的方式不符合面相对象的思想,而字符串应用又非常广泛,因此Java语言专门提供了String类。

2. 常用方法

2.1 字符串构造

String类提供的构造方式非常多,常用的就以下三种:

public static void main(String[] args) {
    //使用常量串构造
    String s1 = "hello world";
    System.out.println(s1);

    //直接new String 对象
    String s2 = new String("hello world");
    System.out.println(s2);

    //使用字符数组进行构造
    char[] array = {'h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd'};
    String s3 = new String(array);
    System.out.println(s3);
}

其他方法需要用到时,大家参考Java在线文档:String文档

【注意】

  1. String是引用类型,内部并不存储字符串本身,在String类的实现源码中,String类实例变量如下:
    在这里插入图片描述
public static void main(String[] args) {
    //s1和s2引用的不是同一对象
    String s1 = new String("hello");
    String s2 = new String("world");
    //s1和s3引用的是同一对象
    String s3 = s1;

    System.out.println(s1.length());//获取字符串长度 -> 输出5
    System.out.println(s1.isEmpty());//如果字符串长度为0,返回true,否则返回false
}

在这里插入图片描述

  1. 在Java中“”引起来的也是String类型对象。
public static void main(String[] args) {
    //打印"hello"字符串(String对象)的长度
    System.out.println("hello".length());
}

2.2 String对象的比较

字符串的比较是常见操作之一,比如:字符串排序。Java中总共提供了4中方式:

  1. ==比较是否引用同一个对象
    注意:对于内置类型,比较的是变量中的值;对于引用类型比较的是引用中的地址。
public class Test {
    public static void main(String[] args) {
        int a = 10;
        int b = 20;
        int c = 10;

        //对于基本数据类型,==比较的是两个变量中存储的值是否相同
        System.out.println(a == b);//false
        System.out.println(a == c);//true

        //对于引用类型变量,==比较两个引用变量引用的时候为同一对象
        String s1 = new String("hello");
        String s2 = new String("hello");
        String s3 = new String("world");
        String s4 = s1;

        System.out.println(s1 == s2);//false
        System.out.println(s2 == s3);//false
        System.out.println(s1 == s4);//true
    }
}
  1. boolean equals(Object anObject) 方法:按照字典序比较

字典序:字符大小的顺序

String类重写了父类Object中equals方法,Object中equals默认按照==比较,String重写equals方法后,按照如下规则进行比较,比如: s1.equals(s2)

//库方法
public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = value.length;
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}
//Test.java
public static void main(String[] args) {
    String s1 = new String("hello");
    String s2 = new String("hello");
    String s3 = new String("Hello");

    //s1、s2、s3引用的是三个不同的对象,因此==比较的结果全部是false
    System.out.println(s1 == s2);//false
    System.out.println(s1 == s3);//false

    /**
     * equals比较:String对象中逐个字符
     * 虽然s1和s2引用的不是同一个对象,但是两个对象中放置的内容相同,因此输出的是true
     * s1和s3引用的不是同一个对象,而且两个对象中的内容也不同,因此输出false
     */
    System.out.println(s1.equals(s2));//true
    System.out.println(s1.equals(s3));//false
}
  1. int compareTo(String s) 方法: 按照字典序进行比较

与equals不同的是,equals返回的是boolean类型,而compareTo返回的是int类型,具体比较方式:

  • 先按照字典序大小比较,如果出现不等的字符,直接返回两个字符大小的差值
  • 如果前k个字符相等(k为两个字符长度最小值),返回值两个字符串长度差值
public class Test {
    public static void main(String[] args) {
        String s1 = new String("abc");
        String s2 = new String("ac");
        String s3 = new String("abc");
        String s4 = new String("abcs");

        System.out.println(s1.compareTo(s2));//不同输出字符值为-1
        System.out.println(s1.compareTo(s3));//相同输出0
        System.out.println(s1.compareTo(s4));//前k个字符完全相同,输出长度差值-1
    }
}
  1. int compareToIgnoreCase(String str) 方法:与compareTo方式相同,但是忽略大小写比较
public class Test {
    public static void main(String[] args) {
        String s1 = new String("abc");
        String s2 = new String("ac");
        String s3 = new String("ABc");//忽略大小写 -> abc
        String s4 = new String("abcs");

        System.out.println(s1.compareToIgnoreCase(s2));//不同输出字符差值为-1
        System.out.println(s1.compareToIgnoreCase(s3));//相同输出0
        System.out.println(s1.compareToIgnoreCase(s4));//前k个字符完全相同,输出长度差值-1
    }
}

2.3 字符串查找

字符串查找也是字符串中非常常见的操作,String类提供的常用查找的方法:

方法功能
char charAt(int index)返回index位置上字符,如果index为负数或者越界,抛出IndexOutOfBoundsException异常
int indexOf(int ch)返回ch第一次出现的位置,没有返回-1
int indexOf(int ch, int fromIndex)从fromIndex位置开始找ch第一次出现的位置,没有返回-1
int indexOf(String str)返回str第一次出现的位置,没有返回-1
int indexOf(String str, int fromIndex)从fromIndex位置开始找str第一次出现的位置,没有返回-1
int lastIndexOf(int ch)从后往前找,返回ch第一次出现的位置,没有返回-1
int lastIndexOf(int ch, int fromIndex)从fromIndex位置开始找,从后往前找ch第一次出现的位置,没有返回-1
int lastIndexOf(String str)从后往前找,返回str第一次出现的位置,没有返回-1
int lastIndexOf(String str, intfromIndex)从fromIndex位置开始找,从后往前找str第一次出现的位置,没有返回-1
public class Test {
    public static void main(String[] args) {
        String s = "ababcabcd";

        //char charAt(int index):
        System.out.println(s.charAt(3));//b

        //int indexOf(int ch):
        System.out.println(s.indexOf('c'));//4
        //int indexOf(int ch, int fromIndex):
        System.out.println(s.indexOf('c', 3));//4
        //int indexOf(String str):
        System.out.println(s.indexOf("abc"));//2
        //int indexOf(String str, int fromIndex):
        System.out.println(s.indexOf("abc", 3));//5

        //int lastIndexOf(int ch):
        System.out.println(s.lastIndexOf('c'));//7
        //int lastIndexOf(int ch, int fromIndex):
        System.out.println(s.lastIndexOf('c', 3));//-1
        //int lastIndexOf(String str):
        System.out.println(s.lastIndexOf("abc"));//5
        //int lastIndexOf(String str, int fromIndex):
        System.out.println(s.lastIndexOf("abc", 3));//2

    }
}

注意:上述方法都是实例方法。

2.4 转化

  1. 数值和字符串转化
//Test.java
class Student {
    public String name;
    public int age;

    public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
//Test.java
public class Test {
    public static void main(String[] args) {
        //1.数字转字符串
        String s1 = String.valueOf(1234);
        String s2 = String.valueOf(12.34);
        String s3 = String.valueOf(true);
        String s4 = String.valueOf(new Student("张三", 19));

        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s3);
        System.out.println(s4);

        System.out.println("===========================");

        //2. 字符串转数字
        //注意:Integer、Double等是Java中的包装类,后面会详细介绍
        int data1 = Integer.parseInt("1234");
        double data2 = Double.parseDouble("12.34");

        System.out.println(data1);
        System.out.println(data2);
        System.out.println(data1 + data2);
    }
}
  1. 大小写转换
public class Test {
    public static void main(String[] args) {
        String s1 = "hello";
        String s2 = "HELLO";

        //1. 小写转大写
        System.out.println(s1.toUpperCase());
        //2. 大写转小写
        System.out.println(s2.toLowerCase());
    }
}

/*
执行结果:
HELLO
hello
 */
  1. 字符串转数组
public class Test {
    public static void main(String[] args) {
        String s = "hello";
        //1. 字符串转数组
        char[] ch = s.toCharArray();//{'h', 'e', 'l', 'l', 'o'};
        for (int i = 0; i < ch.length; i++) {
            System.out.print(ch[i] + " ");
        }
        System.out.println();
        //2. 数组转字符串
        String s2 = new String(ch);
        System.out.println(s2);
    }
}

/*
执行结果:
h e l l o
hello
 */
  1. 格式化
public class Test {
    public static void main(String[] args) {
        String s = String.format("%d-%d-%d", 2024, 6, 20);
        System.out.println(s);
    }
}

/*
执行结果:
2024-6-20
 */

2.5 字符串替换

使用一个指定的新的字符串替换掉已有的字符串数据,可用的方法如下:

方法功能
String replaceAll(String regex, String replacement)替换所有的指定内容
String replaceFirst(String regex, String replacement)替换收个内容
  • 代码示例: 字符串的替换处理
//Test.java
public static void main(String[] args) {
    String str = "hello world hello mom";
    String[] ret = str.split(" ");//按照空格拆分
    for(String s : ret) {
        System.out.println(s);
    }
}

/*
1.执行结果:
hello
world
hello
mom
*/
  • 代码示例: 字符串的部分拆分
public static void main(String[] args) {
    String str = "hello world hello mom";
    String[] ret = str.split(" ", 2);
    for(String s : ret) {
        System.out.println(s);
    }
}

/*2.执行结果:
hello
world hello mom
*/

拆分是特别常用的操作. 一定要重点掌握. 另外有些特殊字符作为分割符可能无法正确切分, 需要加上转义

  • 代码示例: 拆分IP地址
public static void main(String[] args) {
    String str = "192.168.1.1" ;
    String[] ret = str.split("\\.") ;
    for(String s : ret) {
        System.out.println(s);
    }
}

/*
3.执行结果:
192
168
1
1
*/
注意事项:
1. 字符"|"、"*"、"+"、"."都得加上转义字符,前面加上"\\"。
2. 而如果是"\",那么就得写成"\\\\"。
3. 如果一个字符串中有多个分隔符,可以用"|"作为连字符。
  • 代码示例: 多次拆分
public static void main(String[] args) {
    String str = "name=zhangsan&age=18" ;
    String[] ret = str.split("&") ;
    for (int i = 0; i < ret.length; i++) {
        String[] temp = ret[i].split("=") ;
        System.out.println(temp[0]+" = "+temp[1]);
    }
}

/*
4.执行结果:
name = zhangsan
age = 18
*/

这种代码在以后的开发之中会经常出现

2.7 字符串截取

从一个完整的字符串之中截取出部分内容。可用方法如下:

方法功能
String substring(int beginIndex)从指定索引截取到结尾
String substring(int beginIndex, int endIndex)截取部分内容

代码示例: 观察字符串截取

public class Test {
    public static void main(String[] args) {
        String str = "helloworld";
         System.out.println(str);
        //String substring(int beginIndex):
        System.out.println(str.substring(3));
        //String substring(int beginIndex, int endIndex):
        System.out.println(str.substring(0, 5));//[0,5)
    }
}

/*
执行结果:
helloworld
loworld
hello
 */

注意事项:

  1. 索引从0开始
  2. 注意前闭 后开区间的写法,substring(0, 5) 表示包含 0 号下标的字符, 不包含 5 号下标

2.8 其他操作方法

方法功能
String trim()去掉字符串中的左右空格,保留中间空格
String toUpperCase()字符串转大写
String toLowerCase()字符串转小写
  • 代码示例: 观察trim()方法的使用
public static void main(String[] args) {
    String str = " hello world  ";
    System.out.println("[" + str + "]");
    System.out.println("[" + str.trim() + "]");
}

/*
1.执行结果:
[ hello world  ]
[hello world]
 */

trim 会去掉字符串开头和结尾的空白字符(空格, 换行, 制表符等)。

  • 代码示例: 大小写转换
public static void main(String[] args) {
    String str = " hello%$$%@#$%WORLD 哈哈哈 " ;
    System.out.println(str.toUpperCase());
    System.out.println(str.toLowerCase());
}

/*
2.执行结果:
 HELLO%$$%@#$%WORLD 哈哈哈
 hello%$$%@#$%world 哈哈哈
 */

这两个函数只转换字母。

2.9 字符串的不可变性

String是一种不可变对象. 字符串中的内容是不可改变。字符串不可被修改,是因为:

  1. String类在设计时就是不可改变的,String类实现描述中已经说明了
    以下来自JDK1.8中String类的部分实现:
    在这里插入图片描述
    在这里插入图片描述
    String类中的字符实际保存在内部维护的value字符数组中,该图还可以看出:

    • String类被final修饰,表明该类不能被继承
    • value被修饰被final修饰,表明value自身的值不能改变,即不能引用其它字符数组,但是其引用空间中的内容可以修改。
  2. 所有涉及到可能修改字符串内容的操作都是创建一个新对象,改变的是新对象
    比如 replace 方法:

在这里插入图片描述
【纠正】 网上有些人说:字符串不可变是因为其内部保存字符的数组被final修饰了,因此不能改变。

这种说法是错误的,不是因为String类自身,或者其内部value被final修饰而不能被修改。

final修饰类表明该类不想被继承,final修饰引用类型表明该引用变量不能引用其他对象,但是其引用对象中的内容是可以修改的。

public class Test {
    public static void main1(String[] args) {
        String str = "abc";
        System.out.println(str.replace("a", "qq"));
    }

    public static void main(String[] args) {
        final int[] array = {1, 2, 3, 4, 5};
        array[0] = 100;
        System.out.println(Arrays.toString(array));

        //编译报错:java: 无法为最终变量array分配值
        //array = new int[]{6, 7, 8};
    }
}

为什么 String 要设计成不可变的?(不可变对象的好处是什么?)

  1. 方便实现字符串对象池. 如果 String 可变, 那么对象池就需要考虑写时拷贝的问题了
  2. 不可变对象是线程安全的
  3. 不可变对象更方便缓存 hash code, 作为 key 时可以更高效的保存到 HashMap 中

2.10 字符串修改

注意:尽量避免直接对String类型对象进行修改,因为String类是不能修改的,所有的修改都会创建新对象,效率非常低下。

public static void main1(String[] args) {
    String s = "hello";
    s += " world";
    System.out.println(s);//hello world
}

但是这种方式不推荐使用,因为其效率非常低,中间创建了好多临时对象。

在这里插入图片描述

public class Test {
    public static void main(String[] args) {
        long start = System.currentTimeMillis();//毫秒
        String s = "";
        for(int i = 0; i < 10000; ++i){
            s += i;
        }
        long end = System.currentTimeMillis();
        System.out.println(end - start);//248


        start = System.currentTimeMillis();
        StringBuffer sbf = new StringBuffer("");
        for(int i = 0; i < 10000; ++i){
            sbf.append(i);
        }
        end = System.currentTimeMillis();
        System.out.println(end - start);//1


        start = System.currentTimeMillis();
        StringBuilder sbd = new StringBuilder();
        for(int i = 0; i < 10000; ++i){
            sbd.append(i);
        }
        end = System.currentTimeMillis();
        System.out.println(end - start);//0
    }
}

/*
执行结果:
248
1
0
*/

可以看待在对String类进行修改时,效率是非常慢的,因此:尽量避免对String的直接需要,如果要修改建议尽量使用StringBuffer或者StringBuilder。

借助StringBuffer 和 StringBuilder

3. StringBuilder和StringBuffer

由于String的不可更改特性,为了方便字符串的修改,Java中又提供StringBuilder和StringBuffer类。这两个类大部分功能是相同的,这里介绍 StringBuilder常用的一些方法,其它需要用到了大家可参阅StringBuilder官方文档

方法功能
StringBuff append(String str)在尾部追加,相当于String的+=,可以追加:boolean、char、char[]、double、float、int、long、Object、String、StringBuff的变量
char charAt(int index)获取index位置的字符
int length()获取字符串的长度
int capacity()获取底层保存字符串空间总的大小
void ensureCapacity(int mininmumCapacity)扩容
void setCharAt(int index, char ch)将index位置的字符设置为ch
int indexOf(String str)返回str第一次出现的位置
int indexOf(String str, int fromIndex)从fromIndex位置开始查找str第一次出现的位置
int lastIndexOf(String str)返回最后一次出现str的位置
int lastIndexOf(String str, int fromIndex)从fromIndex位置开始找str最后一次出现的位置
StringBuff insert(int offset, String str)在offset位置插入:八种基类类型 & String类型 & Object类型数据
StringBuffer deleteCharAt(int index)删除index位置字符
StringBuffer delete(int start, int end)删除[start, end)区间内的字符
StringBuffer replace(int start, int end, String str)将[start, end)位置的字符替换为str
String substring(int start)从start开始一直到末尾的字符以String的方式返回
String substring(int start,int end)将[start, end)范围内的字符以String的方式返回
StringBuffer reverse()反转字符串
String toString()将所有字符按照String的方式返回
public static void main(String[] args) {
    StringBuilder sb1 = new StringBuilder("hello");
    StringBuilder sb2 = sb1;

    //追加:即尾插 ->字符、字符串、整型数字
    sb1.append(' ');//hello
    sb1.append("world");//hello world
    sb1.append(123);//hello world123
    System.out.println(sb1);//hello world123
    System.out.println(sb1 == sb2);//true
    System.out.println("===========================");


    System.out.println(sb1.charAt(0));//获取0号位上的字符 h
    System.out.println(sb1.length());//获取字符串的有效长度14
    System.out.println(sb1.capacity());//获取底层数组的总大小
    System.out.println("===========================");

    sb1.setCharAt(0, 'H');//设置任意位置的字符 Hello world123
    sb1.insert(0, "Hello world!!!");//Hello world!!!Hello world123
    System.out.println(sb1);
    System.out.println(sb1.indexOf("Hello"));//获取Hello第一次出现的位置
    System.out.println(sb1.lastIndexOf("hello"));//获取hello最后一次出现的位置
    System.out.println("===========================");


    sb1.deleteCharAt(0);//删除首字符
    sb1.delete(0,5);//删除[0, 5)范围内的字符
    String str = sb1.substring(0, 5);//截取[0, 5)区间中的字符以String的方式返回
    System.out.println(str);
    System.out.println("===========================");


    sb1.reverse();//字符串逆转
    str = sb1.toString();//将StringBuffer以String的方式返回
    System.out.println(str);
}

从上述例子可以看出:String和StringBuilder最大的区别在于String的内容无法修改,而StringBuilder的内容可以修改。频繁修改字符串的情况考虑使用StringBuilder

注意:String和StringBuilder不能直接转换。如果想要互相转换,可以采用如下原则:

  • String变成StringBuilder:利用StringBuilder的构造方法或者append()方法
public static void main(String[] args) {
    String str = "hello";
    StringBuilder sb = new StringBuilder(str);//使用构造方法
    //或者
    StringBuilder sb2 = new StringBuilder();
    System.out.println(sb2.append(str));//使用append方法
}
  • StringBuilder变成String:调用toString()方法。
public static void main(String[] args) {
    StringBuilder sb = new StringBuilder("hello");
    String str = sb.toString();
    System.out.println(str);
}

3.2 面试题:

  1. String、StringBuffer、StringBuilder的区别
  • String的内容不可修改,StringBuffer和StringBuilder的内容可以修改
  • StringBuffer和StringBuilder的大部分功能是相似的
  • StringBuffer采用同步处理,属于线程安全操作,而StringBuilder未采用同步处理,属于线程不安全操作
  1. 以下总共创建了多少个String对象 【前提不考虑常量池之前是否存在】
public class Test {
    public static void main(String[] args) {
        String str = new String("ab");//会创建多少个对象
        String str1 = new String("a") + new String("b");//会创建多少个对象
    }
}

在Java中,使用new关键字创建一个对象时,就会在堆内存中分配一个新的对象实例。现在,来分析一下代码,看看分别创建了多少个对象。

对于第一行代码:

String str = new String("ab");

这里会创建两个对象:

  • 一个是在字符串常量池中创建的字符串字面量"ab"(如果池中尚未存在该字面量)。但请注意,如果之前已经有相同的字符串字面量存在于常量池中,则不会创建新的字面量对象。

  • 另一个是通过 new String(“ab”) 在堆上创建的一个新的String对象,该对象的内容与字符串常量池中的"ab"相同,但是是堆上的一个独立副本。

所以,这行代码至少会创建一个堆上的String对象,可能还会在字符串常量池中创建一个对象(如果之前不存在的话)。

对于第二行代码:

String str1 = new String("a") + new String("b");

这里发生的事情稍微复杂一些:

  • 首先,字符串字面量"a"和"b"会被放入字符串常量池(如果它们之前不存在的话)。

  • 然后,通过 new String(“a”)和new String(“b”) 在堆上分别创建两个新的String对象。

当执行字符串连接操作时(使用+运算符),Java会创建一个新的StringBuilder对象(在某些情况下可能是StringBuffer,但在大多数情况下是StringBuilder),并使用其append方法来连接两个字符串。然后,通过调用StringBuilder的toString方法来创建一个表示连接结果的新String对象。

  • 因此,这行代码至少会创建以下对象:

两个在堆上的String对象(通过new String(“a”)和new String(“b”)创建)。

  1. 一个StringBuilder对象(或类似的可变字符序列对象)。
  2. 一个表示连接结果的String对象(通过StringBuilder的toString方法创建)。
  • 另外,如果字符串常量池中之前没有"a"和"b"这两个字面量,那么还会在池中创建这两个对象。

综上所述,str的初始化至少会创建一个对象(堆上的String),可能还会在字符串常量池中创建一个;而str1的初始化则会创建至少四个对象(两个堆上的String,一个StringBuilder,以及一个表示连接结果的String),同样,如果字符串常量池中之前没有相关字面量,还会在池中创建。不过,请注意,JVM的具体实现和版本可能会影响这些细节。

4. String类oj

4.1第一个只出现一次的字符

第一个只出现一次的字符

描述:

	给定一个字符串 s ,找到 它的第一个不重复的字符,并返回它的索引 。如果不存在,则返回 -1 
	
	示例 1:
	输入: s = "leetcode"
	输出: 0
	
	示例 2:
	输入: s = "loveleetcode"
	输出: 2
	
	示例 3:
	输入: s = "aabb"
	输出: -1

代码展示:

class Solution {
    public static int firstUniqChar(String s) {
        int[] count = new int[26];//这个数组拿来计数:字符出现的次数
        for(int i = 0; i < s.length(); i++) {
            //从字符串中拿出字符:
            char ch = s.charAt(i);//length
            //放在计数的数组中
            count[ch - 'a'] ++;
        }
        //计数完重新遍历一遍字符串,找到它的第一个不重复的字符
        for(int i = 0; i < s.length(); i++) {
            char ch = s.charAt(i);
            if(count[ch - 'a'] == 1) {
                return i;
            }
        }
        return -1;
    }
}

public class Test {
    public static void main(String[] args) {
        int ret = Solution.firstUniqChar("loveleetcode");
        System.out.println(ret);
    }
}

解析:

  • 初始化计数数组:首先,代码创建了一个大小为26的整数数组count,用于记录字符串中每个小写字母出现的次数。因为英文字母有26个,所以数组大小是26。
  • 统计字符出现次数:接着,代码遍历输入字符串s,对每个字符进行计数。字符的计数位置是通过将字符转换为0-25之间的索引来确定的(通过ch - ‘a’)。例如,字符’a’对应索引0,字符’b’对应索引1,以此类推。
  • 寻找第一个只出现一次的字符:完成计数后,代码再次遍历字符串s。这次遍历的目的是找到第一个在count数组中计数为1的字符,即只出现一次的字符。一旦找到这样的字符,就返回它在字符串中的位置(索引)。
  • 没有只出现一次的字符:如果字符串中没有只出现一次的字符,函数将返回-1。

简而言之,这段代码通过两次遍历字符串:第一次统计字符出现次数,第二次找到第一个只出现一次的字符。

图文:
在这里插入图片描述

4.2最后一个单词的长度

最后一个单词的长度
描述:

计算字符串最后一个单词的长度,单词以空格隔开,字符串长度小于5000。(注:字符串末尾不以空格为结尾)
输入描述:
输入一行,代表要计算的字符串,非空,长度小于5000。

输出描述:
输出一个整数,表示输入字符串最后一个单词的长度。

示例:

输入:hello nowcoder
输出:8
说明:最后一个单词为nowcoder,长度为8 

代码展示:

  • 方法1:
import java.util.Scanner;
// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    //方法1:使用split方法 -> 切割字符串
    public static void main1(String[] args) {
        Scanner scanner = new Scanner(System.in);
        String s = scanner.nextLine();
        //使用split切割字符串,装在新的数组中
        String[] ss = s.split(" ");
        //取出最后一个索引的内容
        int len = ss[ss.length - 1].length();
        System.out.println(len);
    }
}

解析:

  • 创建Scanner对象:首先,代码创建了一个Scanner对象来读取用户的输入。
    读取一行文本:使用scanner.nextLine()方法读取用户输入的一行文本,并将其存储在字符串s中。
  • 分割字符串:使用s.split(" ")方法将字符串s按照空格分割成多个子字符串,并将这些子字符串存储在一个字符串数组ss中。
  • 获取最后一个单词的长度:通过ss[ss.length - 1]获取数组ss中的最后一个元素(即最后一个单词),然后使用.length()方法计算这个单词的长度,并将长度存储在变量len中。
  • 打印长度:最后,代码使用System.out.println(len)打印出最后一个单词的长度。

简而言之,这段代码就是读取一行文本,找出其中的最后一个单词,并输出这个单词的长度。

  • 方法2:
import java.util.Scanner;
// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Test {
    //方法2:使用lastIndexOf() 和 substring()
    public static void main(String[] args) {
        //lastIndexOf() : 从字符串的后面开始找: " ",找到就返回索引,否则就返回-1
        //substring(int beginIndex) : 从beginIndex开始发生截断
        Scanner scanner = new Scanner(System.in);
        String s = scanner.nextLine();
        //找最后一个空格后面的字符串
        int index = s.lastIndexOf(" ");
        //从index索引开始发生截断 -> 得到index+1后面的字符串
        int len = s.substring(index + 1).length();
        System.out.println(len);
    }
}

解析:

  • 读取输入:首先,通过Scanner对象读取用户输入的一行文本。
  • 查找最后一个空格:使用lastIndexOf(" ")方法从输入的文本中查找最后一个空格的位置,并返回其索引。如果没有找到空格,该方法会返回-1,但在这个上下文中,我们假设输入中至少有一个空格。
  • 截取字符串:使用substring(index + 1)方法从最后一个空格之后开始截取字符串。index + 1确保了我们从空格之后的第一个字符开始截取。
  • 计算长度并输出:计算截取后的字符串的长度,并使用System.out.println输出这个长度。

简而言之,这段代码读取一行文本,找到最后一个空格后的字符串部分,并输出其长度。

4.3检测字符串是否为回文

检测字符串是否为回文
描述:

如果在将所有大写字符转换为小写字符、并移除所有非字母数字字符之后,短语正着读和反着读都一样。则可以认为该短语是一个 回文串 。

字母和数字都属于字母数字字符。

给你一个字符串 s,如果它是 回文串 ,返回 true ;否则,返回 false 。

示例:

示例 1:
输入: s = "A man, a plan, a canal: Panama"
输出:true
解释:"amanaplanacanalpanama" 是回文串。

示例 2:
输入:s = "race a car"
输出:false
解释:"raceacar" 不是回文串。

示例 3:
输入:s = " "
输出:true
解释:在移除非字母数字字符之后,s 是一个空字符串 "" 。
由于空字符串正着反着读都一样,所以是回文串。

代码展示:

class Solution {
    public boolean isPalindrome(String s) {
        //所有大写字符转换为小写字符:
        s = s.toLowerCase();
        //开始判断:定义两个索引:一个从左边开始, 一个从右边开始,进行对比
        int left = 0;
        int right = s.length() - 1;
        //开始对比:
        while(left < right) {
            //遇到非字母和非数字就跳过,不进行比较
            while(left < right && !isNumberAndCharacter(s.charAt(left))) {
                left++;
            }
            while(left < right && !isNumberAndCharacter(s.charAt(right))) {
                right--;
            }
            //走到这,证明字符都是字母,可以进行对比
            if(s.charAt(left) == s.charAt(right)) {
                left++;
                right--;
                //对比完,就往下走
            }else {
                //不相同,说明不是回文串
                return false;
            }
        }
        //走到这,证明是回文串
        return true;
    }

    //字母和数字都属于字母数字字符。 -> 假如是非字母和非数字就跳过,不进行比较
    private boolean isNumberAndCharacter(char ch) {
        //用isDigit() : 判断字符是否为数字
        //用isLetter() : 判断字符是否为字母
        if(Character.isDigit(ch) || Character.isLetter(ch)) {
            return true;
        }else {
            return false;
        }
    }
}
public class Test {
    public static void main(String[] args) {
        Solution solution = new Solution();
        System.out.println(solution.isPalindrome("A man, a plan, a canal: Panama"));
    }
}

解析:
代码定义了一个Solution类,其中包含两个方法:isPalindrome和isNumberAndCharacter。这个类的主要功能是检查一个字符串是否是一个“回文串”,即正读和反读都相同的字符串

  1. isPalindrome方法:

    • 首先,该方法将输入字符串s转化为小写,以确保大小写不影响回文的判断。

    • 定义了两个指针,left从字符串的开始位置出发,right从字符串的末尾位置出发。

    • 使用一个while循环,当left小于right时执行循环体内的代码。
      在循环体内,代码会跳过left指针位置上的非字母和非数字字符,直到找到第一个字母或数字字符。

      • 同样,代码也会跳过right指针位置上的非字母和非数字字符,直到找到第一个字母或数字字符。
      • 接着,比较left和right指针位置上的字符是否相同。如果相同,则将两个指针向中间移动;如果不同,则直接返回false,表示该字符串不是回文串。
    • 如果left指针超过了right指针,说明已经检查完整个字符串,且字符串是回文的,所以返回true。

  2. isNumberAndCharacter方法:

    • 这个辅助方法用于检查一个字符是否是字母或数字。
    • 使用Character.isDigit(ch)检查字符ch是否为数字。
    • 使用Character.isLetter(ch)检查字符ch是否为字母。
    • 如果字符是数字或字母,则返回true;否则返回false。

简而言之,这段代码通过双指针的方法,从字符串的两端向中间遍历,同时跳过非字母和非数字的字符,来判断一个字符串是否是回文串。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1843910.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

使用虚拟滚动条优化通过el-collapse展示多条数据的性能问题

我们将一个10000条的数据通过el-collapse展示出来&#xff0c;同时在点开每一个item时&#xff0c;要内置一个编辑器&#xff0c;对文本内容进行编辑。其实&#xff0c;如果仅10000条数据的文本的单独展示&#xff0c;可能性能不会太差&#xff0c;但由于每一条都需要带有一个文…

web中间件漏洞-jboss部署war包

web中间件漏洞-jboss部署war包 攻击机服务器准备好的ma.war

FreeBSD在zfs挂接第二块ssd 硬盘

为FreeBSD机器新增加了一块ssd硬盘&#xff1a;骑尘 256G 先格式化分区硬盘 进入bsdconfig 选Disk Management 选择ada1 &#xff0c;也就是新增加的硬盘 选择auto 然后选择Entire Disk 提示信息 The existing partition scheme on this disk (MBR) │ …

Python数据科学 | 是时候跟Conda说再见了

本文来源公众号“Python数据科学”&#xff0c;仅用于学术分享&#xff0c;侵权删&#xff0c;干货满满。 原文链接&#xff1a;是时候跟Conda说再见了 1 简介 conda作为Python数据科学领域的常用软件&#xff0c;是对Python环境及相关依赖进行管理的经典工具&#xff0c;通…

容声神助攻!欧洲杯最刺激一战诞生,神来之笔背后有高人

2-2&#xff0c;当比分最终被定格在这个数字时&#xff0c;克罗地亚的老将们即使职业生涯已经经历了太多太多惨烈的比赛&#xff0c;此刻依然难掩心中的复杂情绪。 欧洲杯开赛至今最刺激的一战&#xff0c;从0-1落后的长时间焦虑&#xff0c;到下半场3分钟扳平反超的狂喜&…

【嵌入式Linux】<总览> 文件IO(更新中)

文章目录 前言 一、常用函数 1. open函数 2. close函数 3. write函数 4. read函数 5. dup函数 6. dup2函数 二、文件读写细节 1. 换行符 2. 文件描述符 3. errno和perror 前言 在Linux系统中&#xff0c;一切皆文件。因此&#xff0c;掌握Linux下文件IO常用的函数…

为什么美业门店要用专业的美业系统?博弈美业SaaS管理系统Java源码分享

美容、医美等美业门店需要使用专业的美业系统&#xff0c;而不是普通的管理系统&#xff0c;美业专用系统的优势在哪&#xff1f; 专业的美业系统与普通系统相比&#xff0c;更加贴合美业门店的经营需求&#xff0c;提供了更全面、便捷、高效的管理功能&#xff0c;有助于提升…

端到端自动驾驶的基础概念

欢迎大家关注我的B站&#xff1a; 偷吃薯片的Zheng同学的个人空间-偷吃薯片的Zheng同学个人主页-哔哩哔哩视频 (bilibili.com) 目录 1.端到端自动驾驶的定义 1.1特斯拉FSD 1.2端到端架构演进 1.3大模型 1.4世界模型 1.5纯视觉传感器 2.落地的挑战 1.端到端自动驾驶的定…

RSA学习

[MRCTF2020]Easy_RSA 先来分析一下这个RSA代码的特殊性&#xff0c;这个不是传统的RSA,随机生成N&#xff0c;并保证为N%8的余数是5 zlib 用于数据压缩&#xff0c;但是并似乎没有用到 gen_p(): 生成随机的1024位质数p。计算np*q&#xff0c;并没有直接用于加密。计算F_n…

返回给前端数据的封装

返回格式如下&#xff1a; { "code": 200/400, "msg": "成功"/"失败", "total": n, "data": [ {}&#xff0c;{}]} 1.在common中新增Result 类&#xff0c;代码如下 package com.xxx0523.common; import lombo…

【OpenVINO™】使用 OpenVINO™ C# 异步推理接口部署YOLOv8 ——在Intel IGPU 上速度依旧飞起!!

OpenVINO Runtime支持同步或异步模式下的推理。Async API的主要优点是&#xff0c;当设备忙于推理时&#xff0c;应用程序可以并行执行其他任务&#xff08;例如&#xff0c;填充输入或调度其他请求&#xff09;&#xff0c;而不是等待当前推理首先完成。 当我们使用异步API时&…

【React】使用Token做路由权限控制

在components/AuthRoute/index.js中 import { getToken } from /utils import { Navigate } from react-router-domconst AuthRoute ({ children }) > {const isToken getToken()if (isToken) {return <>{children}</>} else {return <Navigate to"/…

算法设计与分析 实验4 动态规划法求扔鸡蛋问题

目录 一、实验目的 二、问题描述 三、实验要求 四、实验内容 动态规划法 算法描述 算法伪代码描述 算法复杂度分析 数据测试 二分优化的动态规划法 算法描述 二分优化&#xff1a; 算法伪代码 算法复杂度分析 数据测试 单调决策优化的动态规划法 算法描述 算…

【机器学习】与【深度学习】的前沿探索——【GPT-4】的创新应用

gpt4o年费&#xff1a;一年600&#xff0c; 友友们&#xff0c;一起拼单呀&#xff0c;两人就是300&#xff0c;三个人就是200&#xff0c;以此类推&#xff0c; 我已经开通年费gpt4o&#xff0c;开通时长是 从2024年6月20日到2025年7月16日 有没有一起的呀&#xff0c;有需要的…

vue3页面传参

一&#xff0c;用query传参 方法&#xff1a; router.push({path: ‘路由地址’, query: ‘参数’}) 例子&#xff1a;a页面携带参数跳转到b页面并且b页面拿到a页面传递过来的参数 在路由router.ts配置 a页面&#xff1a; <template><div >a页面</div>…

Spatio-temporal Relation Modeling for Few-shot Action Recognition

标题&#xff1a;少样本动作识别的时空关系建模 源文链接&#xff1a;Thatipelli_Spatio-Temporal_Relation_Modeling_for_Few-Shot_Action_Recognition_CVPR_2022_paper.pdf (thecvf.com)https://openaccess.thecvf.com/content/CVPR2022/papers/Thatipelli_Spatio-Temporal_…

多目标跟踪中用到的求解线性分配问题(Linear Assignment Problem,LAP)Python

多目标跟踪中用到的求解线性分配问题&#xff08;Linear Assignment Problem&#xff0c;LAP&#xff09;Python flyfish 如果想看 C版本的&#xff0c;请点这里。 线性分配问题&#xff08;LAP&#xff0c;Linear Assignment Problem&#xff09;是一个经典的优化问题&…

虚拟机配置桥接模式

背景 因为要打一些awd比赛,一些扫描工具什么的,要用到kali,就想着换成一个桥接模式 但是我看网上的一些文章任然没弄好,遇到了一些问题 前置小问题 每次点开虚拟网络编辑器的时候都没有vmnet0,但是点击更改的时候却有vmnet0 第一步: 点击更改设置 第二步: 把wmnet0删掉 …

AD使用快捷键

1、如何实现元器件旋转45放置 在Preferences >> PCB Editor >> General中将Rotation Step&#xff08;旋转的步进值&#xff09;由90改为45&#xff0c;这样以后每次按空格键旋转器件时旋转角度为45。 2、显示网络、隐藏网络 N 3、对齐 2、设置DRC检查选项&#xf…

[17] 使用Opencv_CUDA 进行滤波操作

使用Opencv_CUDA 进行滤波操作 邻域处理操作 > 滤波操作&#xff0c;拒绝或者允许某特定频段通过如果图像某处的灰度级变化缓慢&#xff0c;那么就是低频区域&#xff0c;如果灰度级变化剧烈&#xff0c;就是高频区域邻域滤波即卷积操作形态学处理&#xff1a;膨胀&#xf…