Java语法学习八之认识String类

String类的重要性

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

在开发和校招笔试中,字符串也是常客,比如:

字符串相加

而且在面试中也频繁被问到,比如:String、StringBuff和StringBulider之间的区别等。

常用方法

字符串构造

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

public class Test {
    public static void main(String[] args) {
       //直接赋值的方法 定义字符串-使用常量串构造
        String str="adcdef";

        //直接newString对象
        String str2=new String("hello");

        //用字符数组 来构造字符串
        char[] array={'a','b','c','d'};
        String str3=new String(array);

        System.out.println(str);
        System.out.println(str2);
        System.out.println(str3);
    }
}

注意:

1. String是引用类型,内部并不存储字符串本身,在String类的实现源码中,String类实例变量如下:

public class Test {
    public static void main(String[] args) {
        String str="abcdef";
        System.out.println(str.length());

        String str2=null;
        System.out.println(str2);
    }
}

public class Test {
    public static void main(String[] args) {
        String str="abcdef";
        System.out.println(str.length());

        String str2=null;
        System.out.println(str2);

        //空指针异常
        System.out.println(str2.length());
    }
}

public class Test {
    public static void main(String[] args) {
        String str="abcdef";
        System.out.println(str.length());//获取字符串长度
        System.out.println(str.isEmpty());//false

        String str2=null;
        System.out.println(str2);

        //空指针异常
        //System.out.println(str2.length());

        String str3="";
        System.out.println(str3.length());//0
        System.out.println(str3.isEmpty());//true 如果字符串长度为0,返回true,否则返回false
    }
}

2. 在Java中“”引起来的也是String类型对象。

// 打印"hello"字符串(String对象)的长度
System.out.println("hello".length());

String对象的比较

字符串的比较是常见操作之一,Java中总共提供了4中方式:
1. ==比较是否引用同一个对象

注意:对于内置类型,==比较的是变量中的;对于引用类型==比较的是引用中的地址

public class Test {
    public static void main(String[] args) {
        String str1="abcdef";
        String str2="abcdef";
        System.out.println(str1==str2);//true

        String str3=new String("hello");
        String str4=new String("hello");
        System.out.println(str3==str4);//false

    }
}

2. boolean equals(Object anObject) 方法:按照字典序比较

public class Test {
    public static void main(String[] args) {
        String str1="abcdef";
        String str2="abcdef";
        //System.out.println(str1==str2);//true

        System.out.println(str1.equals(str2));

        String str3=new String("hello");
        String str4=new String("hello");
        //System.out.println(str3==str4);//false

        System.out.println(str3.equals(str4));

    }
}

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

public boolean equals(Object anObject) {
// 1. 先检测this 和 anObject 是否为同一个对象比较,如果是返回true
        if (this == anObject) {
            return true;
        }

        // 2. 检测anObject是否为String类型的对象,如果是继续比较,否则返回false
        if (anObject instanceof String) {
// 将anObject向下转型为String类型对象
            String anotherString = (String)anObject;
            int n = value.length;

// 3. this和anObject两个字符串的长度是否相同,是继续比较,否则返回false
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;

// 4. 按照字典序,从前往后逐个字符进行比较
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                } 
                return true;
            }
        } 
        return false;
}
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
}

3. int compareTo(String s) 方法: 按照字典序进行比较
与equals不同的是,equals返回的是boolean类型,而compareTo返回的是int类型。具体比较方式:
1. 先按照字典次序大小比较,如果出现不等的字符,直接返回这两个字符的大小差值
2. 如果前k个字符相等(k为两个字符长度最小值),返回值为两个字符串长度差值

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("abcdef");
System.out.println(s1.compareTo(s2)); // 不同输出字符差值-1
System.out.println(s1.compareTo(s3)); // 相同输出 0
System.out.println(s1.compareTo(s4)); // 前k个字符完全相同,输出长度差值 -3
}

4. int compareToIgnoreCase(String str) 方法:与compareTo方式相同,但是忽略大小写比较

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("abcdef");
System.out.println(s1.compareToIgnoreCase(s2)); // 不同输出字符差值-1
System.out.println(s1.compareToIgnoreCase(s3)); // 相同输出 0
System.out.println(s1.compareToIgnoreCase(s4)); // 前k个字符完全相同,输出长度差值 -3
}

字符串查找

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

public class Test {
    public static void main(String[] args) {
        String str1="abcd";

        char ch=str1.charAt(1);
        System.out.println(ch);

        int index=str1.indexOf('d');
        System.out.println(index);

        int index1=str1.indexOf('a',2);
        System.out.println(index1);

        int index2=str1.indexOf("bc");
        System.out.println(index2);

        String str2="ababcabcd";
        //从后往前找'a'
        int index3=str2.lastIndexOf('a');
        System.out.println(index3);

        //从指定位置从后往前找'a'
        int index4=str2.lastIndexOf('a',4);
        System.out.println(index4);
    }
}

转化

1. 数值和字符串转化

public class Test {
    public static void main(String[] args) {
        String s1 = String.valueOf(1234);
        String s2 = String.valueOf(12.34);
        String s3 = String.valueOf(true);
        String s4 = String.valueOf(new Student("Hanmeimei", 18));
        System.out.println(s1);//1234
        System.out.println(s2);//12.34
        System.out.println(s3);//true
        System.out.println(s4);
        System.out.println("=================================");
        // 字符串转数字
        // 注意:Integer、Double等是Java中的包装类型,这个后面会讲到
        int data1 = Integer.parseInt("1234");
        double data2 = Double.parseDouble("12.34");
        System.out.println(data1);//1234
        System.out.println(data2);//12.34
    }
}

2. 大小写转换

public class Test {
    public static void main(String[] args) {
        //大写转换
        String str="ABCD";
        String str3=str.toLowerCase();
        System.out.println(str3);
        System.out.println(str);//并非在原字符本身上做修改

        //小写转换
        String str1="abcd";
        System.out.println(str1.toUpperCase());

    }
}

3. 字符串转数组

public class Test {
    public static void main(String[] args) {
        String str="abcd";
        //把字符串转为数组
        char[] array=str.toCharArray();
        System.out.println(Arrays.toString(array));

        //数组转字符串
        String s=new String(array);
        System.out.println(s);
    }
}

4.格式化

public static void main(String[] args) {
String s = String.format("%d-%d-%d", 2019, 9,14);
System.out.println(s);
}

字符串替换
 

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

public class Test {
    public static void main(String[] args) {
        String str="ababc";
        //字符的整体替换
        String ret=str.replace('a','l');
        System.out.println(ret);

        //字符串的整体替换
        String ret2=str.replace("ab","kkk");
        System.out.println(ret2);

        //替换字符串的第一个
        String ret3=str.replaceFirst("ab","kkkk");
        System.out.println(ret3);

        //字符串的整体替换
        String ret4=str.replaceAll("a","ppp");
        System.out.println(ret4);
    }
}

 

注意事项: 由于字符串是不可变对象, 替换不修改当前字符串, 而是产生一个新的字符串.
 

字符串拆分
 

可以将一个完整的字符串按照指定的分隔符划分为若干个子字符串。

public class Test {
    public static void main(String[] args) {
        String str="abc&def&hij";
        String[] strings=str.split("&");
        for(String s:strings){
            System.out.println(s);
        }
    }
}

public class Test {
    public static void main(String[] args) {
        String str="abc&def&hij";
        //最多分为几组
        String[] strings=str.split("&",2);
        for(String s:strings){
            System.out.println(s);
        }
    }
}

public class Test {
    public static void main(String[] args) {
        String str="abc.def.hij";
        //最多分为几组
        String[] strings=str.split(".",2);
        for(String s:strings){
            System.out.println(s);
        }
    }
}

public class Test {
    public static void main(String[] args) {
        String str="abc&def=hij";
        String[] strings=str.split("&|=");
        for(String s:strings){
            System.out.println(s);
        }
    }
}

 

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

注意事项:

1. 字符"|","*","+","."都得加上转义字符,前面加上 "\\" .
2. 而如果是 "\" ,那么就得写成 "\\\\" .
3. 如果一个字符串中有多个分隔符,可以用"|"作为连字符.
 

多次拆分

public class Test {
    public static void main(String[] args) {
        String str="name=zhangsan&name=list";
        String[] strings=str.split("&");
        for(String s:strings){
            System.out.println(s);
            String[] ss=s.split("=");
            for(String x:ss){
                System.out.println(x);
            }
        }
    }
}

字符串截取

public class Test {
    public static void main(String[] args) {
        String str1="abcdef";
        //从指定下标开始进行截取
        String str2=str1.substring(1);
        System.out.println(str2);

        //从指定下标开始进行截取 截取到指定位置
        String str3=str1.substring(1,3);//[1,3)左闭右开
        System.out.println(str3);
    }
}

其他操作方法

public class Test {
    public static void main(String[] args) {
        String str1="      abc d eff   ";
        System.out.println(str1);
        String str2=str1.trim();
        System.out.println(str2);
    }
}

字符串的不可变性
 

String是一种不可变对象. 字符串中的内容是不可改变。字符串不可被修改,是因为:
1. String类在设计时就是不可改变的,String类实现描述中已经说明了
以下来自JDK1.8中String类的部分实现:

JDK17

String类中的字符实际保存在内部维护的value字符数组中,该图还可以看出:
             1. String类被final修饰,表明该类不能被继承
             2. value被修饰被final修饰,表明value自身的值不能改变,即不能引用其它字符数组,但是其引用空间中的内容可以修改。

2. 所有涉及到可能修改字符串内容的操作都是创建一个新对象,改变的是新对象
比如 replace 方法:

【纠正】网上有些人说:字符串不可变是因为其内部保存字符的数组被final修饰了,因此不能改变。这种说法是错误的,不是因为String类自身,或者其内部value被final修饰而不能被修改。

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

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

那如果想要修改字符串中内容,该如何操作呢?

字符串修改

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

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

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

public class Test {
    public static void main(String[] args) {
        String str="hello";
        StringBuilder stringBuilder=new StringBuilder();
        stringBuilder.append(str);
        stringBuilder.append("world");
        str=stringBuilder.toString();
        System.out.println(str);

        String str2="hello";
        for (int i = 0; i <1000 ; i++) {
            str2+=i;//内存耗费极大
        }
        System.out.println(str2);

        StringBuilder stringBuilder2=new StringBuilder();
        stringBuilder2.append(str2);
        for (int i = 0; i < 1000; i++) {
            stringBuilder2.append(i);
        }
        str2=stringBuilder2.toString();
        System.out.println();//效率大大提升
    }
}

StringBuilder和StringBuffer
 

StringBuilder的介绍
 

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

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

注意:String和StringBuilder类不能直接转换。如果要想互相转换,可以采用如下原则:
String变为StringBuilder: 利用StringBuilder的构造方法或append()方法。
StringBuilder变为String: 调用toString()方法。

面试题
 

1. String、StringBuffer、StringBuilder的区别

String的内容不可修改,StringBuffer与StringBuilder的内容可以修改.
StringBuffer与StringBuilder大部分功能是相似的
StringBuffer采用同步处理,属于线程安全操作(多线程);而StringBuilder未采用同步处理,属于线程不安全操作(单线程)

2. 以下总共创建了多少个String对象【前提不考虑常量池之前是否存在】

String str = new String("ab"); // 会创建多少个对象 2
String str = new String("a") + new String("b"); // 会创建多少个对象 6

String类oj
 

第一个只出现一次的字符

class Solution {
    public int firstUniqChar(String s) {
        int[] count = new int[26];
        for (int i = 0; i < s.length(); ++i) {
            char ch = s.charAt(i);
            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;// 没找到
    }
}

最后一个单词的长度

import java.util.Scanner;

// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
    public static void main(String[] args) {
        Scanner scan = new Scanner(System.in);
        String str = scan.nextLine();
        //1.直接调用分割方法->split方法
        // String[] ss=str.split(" ");
        // int len=ss[ss.length-1].length();
        // System.out.println(len);

        //2.lastIndexof->subString截取到7->求这个字符串的长度
        int index = str.lastIndexOf(" ");
        String ret = str.substring(index + 1);
        System.out.println(ret.length());

    }
}

检测字符串是否为回文

class Solution {
    public boolean isPalindrome(String s) {
        s = s.toLowerCase();

        int left = 0;
        int right = s.length() - 1;
        while (left < right) {
            while (left < right && !isCharacterNum(s.charAt(left))) {
                left++;
            }
            while (left < right && !isCharacterNum(s.charAt(right))) {
                right--;
            }
            // left下标 是合法的字符
            // right下标 是合法的字符
            if (s.charAt(left) == s.charAt(right)) {
                left++;
                right--;
            } else {
                return false;
            }
        }
        return true;
    }

    private boolean isCharacterNum(char ch) {
        if (Character.isDigit(ch) || Character.isLetter(ch)) {
            return true;
        }
        return false;
    }
}


 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

高效的Gitlab Flow最佳实践

文章目录 一、git flow二、github flow三、gitlab flow四、基于gitlab flow的最佳实践1.语义化版本号2.测试发布3.bug修复 参考 业界包含三种flow&#xff1a; Git flowGithub flowGitlab flow 三种工作流程&#xff0c;有一个共同点&#xff1a;都采用"功能驱动式开发&…

计算机二级Python基础操作题

题目来源&#xff1a;计算机二级Python半个月抱佛脚大法&#xff08;内呈上真题版&#xff09; - 知乎 第4&#xff0c;5&#xff0c;6&#xff0c;7&#xff0c;9&#xff0c;10&#xff0c;11套 1. 基础题1 sinput() print("{:\"^30x}".format(eval(s))) b …

xilinx linux AXI GPIO 驱动学习

vivado工程 vivado 配置一个 AXI GPIO&#xff0c; 全输出&#xff0c;宽度为1 设备树解读 生成的对应pl.dtsi设备树文件如下 axi_gpio: gpio40020000 {#gpio-cells <2>;clock-names "s_axi_aclk";clocks <&clkc 15>;compatible "xlnx,…

海外盲盒App开发:探索惊喜与乐趣的跨国新体验

在全球化日益加速的今天&#xff0c;人们对新鲜、有趣、充满惊喜的体验需求不断增长。盲盒作为一种充满神秘与乐趣的玩法&#xff0c;正迅速在全球范围内走红。为了满足广大海外用户的盲盒体验需求&#xff0c;我们致力于开发一款海外盲盒App&#xff0c;为用户带来跨国界的惊喜…

C#,图论与图算法,计算无向连通图中长度为n环的算法与源代码

1 无向连通图中长度为n环 给定一个无向连通图和一个数n,计算图中长度为n的环的总数。长度为n的循环仅表示该循环包含n个顶点和n条边。我们必须统计存在的所有这样的环。 为了解决这个问题,可以有效地使用DFS(深度优先搜索)。使用DFS,我们可以找到特定源(或起点)的长度…

高精度AI火灾烟雾检测算法,助力打造更加安全的楼宇环境

一、方案背景 近日&#xff0c;南京居民楼火灾事故导致15人死亡的新闻闹得沸沸扬扬&#xff0c;这一事件又激起了大家对楼宇火灾隐患的进一步担忧。事后我们除了思考政府、消防及物业部门应对此事的解决办法&#xff0c;我们还应该思考如何利用现有的技术帮助人们减少此类事情的…

手撕算法-无重复字符的最长子串

描述 分析 滑动窗口&#xff0c;记录窗口中的所有出现的字符&#xff0c;然后窗口左边界固定&#xff0c;右边界右滑&#xff0c;如果&#xff0c;窗口中不存在新的字符&#xff0c;则右滑成功&#xff0c;否则左边界右滑&#xff0c;直到窗口中不存在右边界的值。 描述感觉不…

Linux初学(八)磁盘管理

一、磁盘管理 1.1 简介 磁盘的工作原理&#xff1a; 添加磁盘对磁盘进行分区格式化磁盘挂载和使用磁盘 磁盘的类型&#xff1a; 固态机械 磁盘的接口类型&#xff1a; IDESTSTSCSI 磁盘工作的原理&#xff1a; 磁盘&#xff0c;特别是硬盘&#xff0c;和内存不同&#xff0c;…

【网络基础】网络层基本协议介绍

目录 一、IP数据包 1.1 网络层的功能 1.2 IP数据包格式 二、ICMP协议介绍 2.1 作用 2.2 常用命令 2.2.1 Ping命令 2.2.2 tracert命令 2.3 广播域 三、ARP协议介绍 3.1 作用 3.2 原理 一、IP数据包 1.1 网络层的功能 定义了基于IP协议的逻辑地址&#xff0c;就是I…

IDEA使用手册

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…

优化选址问题 | 基于禁忌搜索算法求解基站选址问题含Matlab源码

目录 问题代码问题 禁忌搜索算法(Tabu Search)是一种局部搜索算法的扩展,它通过引入一个禁忌列表来避免陷入局部最优解,并允许在一定程度上接受较差的解来跳出局部最优。在基站选址问题中,我们可以使用禁忌搜索算法来寻找满足覆盖要求且基站数量最少的选址方案。 以下是…

无人机采集图像的相关知识

1.飞行任务规划 一般使用飞行任务规划软件进行飞行任务的设计&#xff0c;软件可以自动计算相机覆盖和图像重叠情况。比如ArduPilot (ArduPilot - Versatile, Trusted, Open) 和UgCS (http://www.ugcs.com)是两个飞行任务规划软件&#xff0c;可以适用大多数无人机系统。 2.图…

Java 模拟Spring,实现IOC和AOP的核心(二)

接着上一篇&#xff0c;在上一篇完成了有关IOC的注解实现&#xff0c;这一篇用XML的方式实现IOC&#xff0c;并且完成AOP。 简易的IOC框图 注解的方式实现了左边的分支&#xff0c;那么就剩下右边的XML分支&#xff1a; XmlContext&#xff1a; 这个类是也是AbstractApplicat…

海量数据处理项目-技术Leader必备方法论-SMART衡量需求-工作的利器

海量数据处理项目-技术Leader必备方法论-SMART衡量需求-工作的利器

阿里的库存秒杀是如何实现的?

一、阿里的库存秒杀的实现 阿里有很多业务&#xff0c;几十上百个业务线&#xff0c;各自都有一些需要做抢购、秒杀、热点扣将的场景。他们都用哪些方案呢? 我看了很多资料&#xff0c;也找了很多人做交流&#xff0c;最终得到的结论是啥都有&#xff0c;主要总结几个主流的&…

STM32 ESP8266模块的曲折探索

这是本文的配套资料&#xff0c;最终工程请参考 新_ESP8266资料\stm32f103成功移植的项目 【免费】stm32f103c8t6esp8266资料资源-CSDN文库 一、等到了ready 产品参数 我使用的是ai-thinker的esp8266-01s&#xff0c;以下为产品规格书 引脚定义&#xff1a; 依据引脚定义&…

【深度学习目标检测】二十四、基于深度学习的疲劳驾驶检测系统-含数据集、GUI和源码(python,yolov8)

设计一个疲劳驾驶检测系统的重要性主要体现在以下几个方面&#xff1a; 提高道路安全&#xff1a;疲劳驾驶是引发交通事故的重要因素之一。驾驶员在长时间驾驶或缺乏休息的情况下&#xff0c;反应速度和判断能力会显著下降&#xff0c;从而增加事故风险。通过实时检测驾驶员的疲…

金三银四面试题(一):JVM类加载与垃圾回收

面试过程中最经典的一题&#xff1a; 请你讲讲在JVM中类的加载过程以及垃圾回收&#xff1f; 加载过程 当Java虚拟机&#xff08;JVM&#xff09;启动时&#xff0c;它会通过类加载器&#xff08;ClassLoader&#xff09;加载Java类到内存中。类加载是Java程序运行的重要组成…

AI视频风格转换动漫风:Stable Diffusion+TemporalKit

话不多说&#xff0c;直接开干。 基本方法 首先通过 Temporal-Kit 这个插件提取视频中的关键帧图片&#xff0c;然后使用 Stable Diffusion WebUI 重绘关键帧图片&#xff0c;然后再使用 Temporal-Kit 处理转换后的关键帧图片&#xff0c;它会自动补充关键帧之间的图片&#…

Karmada 管理有状态应用 Xline 的早期探索与实践

背景与动机 目前随着云原生技术和云市场的不断成熟&#xff0c;越来越多的 IT 厂商开始投入到跨云多集群的怀抱当中。以下是 flexera 在 2023 年中关于云原生市场对多云多集群管理的接受程度的调查报告&#xff08;http://info.flexera.com&#xff09; 从 flexera 的报告中可…