java进阶 第三讲-深入了解String

本文深入探讨Java中String类的使用技巧与原理,包括数组拷贝、字符串字面量与new关键字的区别,以及字符串在Java内存中的存储机制。通过具体示例,帮助读者理解如何高效地使用String。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

java进阶 第三讲-深入了解String

1 数组的补充

public class ArrayTest02 {
    public static void main(String[] args) {
        int[] arr = new int[10];
        for (int i = 0; i < arr.length; i++) {
            arr[i] = i+1;
        }
        // 数组的拷贝
        int[] arrCpy = new int[15];
        // 1.自己写代码拷贝
        // 2.使用API,javaAPI给我们提供了很多写好的方法,直接使用就可以了
        System.arraycopy(arr, 0, arrCpy, 0, arr.length);
        /*
        *  1. arrayCopy是一个静态的方法,System类中的,System类位于java.lang包下,不需要导包
        *  2. 该方法一共五个参数
        *       第一个参数:Object src:这里是Object,我们传入的是一个数组,数组也是Object的子类
        *               这就是多态,父类型引用指向子类型对象:Object src = arr;
        *               这个参数是源数组,也是拷贝的对象(蓝本)
        *       第二个参数:int srcPos:这个参数表示从源数组中的哪个位置开始拷贝
        *       第三个参数:Object dest:也是多态的应用,这个参数代表要将源数组拷贝到的目标数组
        *                也就是说要拷贝到的那个数组。
        *       第四个参数:int destPos:这个表示从目标数组的哪里开始放入数据
        *       第五个参数:int length:拷贝多长的数据,拷贝几个数组元素
        * */
        for (int i = 0; i < arrCpy.length; i++) {
            System.out.print(arrCpy[i] + " ");
        }
        System.out.println();
        int i = Arrays.binarySearch(arr, 5); // 返回的是数组下标的索引
        System.out.println(i);
        int[] ints = Arrays.copyOf(arr, 20);
        for (int j = 0; j < ints.length; j++) {
            System.out.print(ints[j] + " ");
        }
    }
}
  • Arrays是一个array的工具类,工具类中的方法多是静态的,直接“类名.”的方式访问。
  • 学会是用API和API帮助文档。不懂的时候,可以去API中看看源码。看源码是培养一个程序员能力的必经之路。

2 String的神奇世界

  • 字面量是字符串的,在java中都被认为是常量。存放在字符串常量池中。
  • 关于编译和运行
java的编译:
	编译有两个大的功能:
	1.按照java的语法规则,编译器对我们开发出来的源程序进行比对。如果符合语法规则,编译通过,实际是我们编的程序通过了,这个步骤就是编。
	2.编译器将检查通过的源程序,翻译成字节码。在这个过程中对源程序进行翻译,翻译成JVM能够直接运行的字节码文件。就我们现在已经学到的知识,我们可以猜想,在编译器翻译我们源程序的过程中,应该做的一些事情:(如果是让你翻译,你怎么做?)
	比如:发现常量,常量是不可变的,是要放入方法区内存的,JVM怎么知道谁放入谁不放入呢?所以在翻译源程序的这个阶段,它会生成一张表,这个表叫做常量表(就是一个数组行不行?),然后把常量放进去。
...........................................................................
	编译就是翻译的过程:比如我写了一篇中文的演讲稿,我叫xjf翻译,你首先是不是要看看我写的东西是不是语法正确,句子通顺?这是第一步,编的过程。
	译的过程:对于已经修改好的演讲稿,符合语法规则的演讲稿,进行英文翻译。翻译成外国人能看得懂的、听得懂的英语文章。在翻译的过程中,xjf发现,有一些很难读的单词,比如说good morning,他就在翻译中给了一个标记:标记成中文的译音:古德莫宁。然后,他把很多这样的注释拿出来做成一个表格,方便以后我演讲的时候用。这就是常量表。
	
	.java文件---> .class文件(这里就增加了很多东西)
	.class文件中不再是java写的代码了,是字节码指令写就的东西。变成使用另外一种语言表述的形式了。
	
java一处编译、处处运行。指的是编译好的.class文件,你可以拿到任何有jvm的地方运行。翻译和发行。翻译书籍,你喜欢翻译多少就翻译多少,发行不是这样的。不能说你翻译了就要发行。
java的运行:
	第一步:将字节码文件加载到内存中,也就是说在JVM中有一块内存空间用来放置代码的,这个代码不是我们写的java代码,而是字节码。这也是类加载的过程。在这个过程中,一些特殊的要处理的数据就直接被处理了,比如常量,在类加载的过程中,会从.class文件中的常量表中提取信息,将他们存入到常量池中。常量池是一块内存空间。这些内存空间统称为方法区内存。
	第二步:加载完了以后,CPU要运算,先跟寄存器通信,寄存器从内存中拿到字节码,放入到PC中,只保存一条,执行完以后又读一条,然后就是栈内存空间、堆内存空间的协调工作。
  • 关于String str = “abc”;的理解
main(){
   String str = "abc"; 
}
首先,等号右边先执行,在编译阶段,就会将"abc"当成是一个常量,放入一张常量表中。
    编译的时候,编译器只会检查"="两边是否是同一类型(或者是多态),如果是,编的过程就结束了,译的过程中发现,有"abc"这样的字面量,它会把它看成是常量,放入到常量表中。常量表是一个文件。或者说是文件中的一段描述。
    
    运行的时候,首先进行类加载,将源程序的字节码文件中的内容加载到方法区内存中,这时候JVM就会去检查有没有特殊的要处理的内容,比如static修饰的变量,static代码块,常量等等,如果有,就会在类加载的时候先执行,执行完以后存入方法区内存中。如果是常量,就会把常量存入常量池中。常量池是方法区内存中的一块内存空间,这块内存空间用来存放常量。
    
    在类加载完以后,程序开始运行,执行到
    String str = "abc";的时候,这时候做的事情是,将常量池中的"abc"的内存地址赋值给引用类型String str。这个String str是在栈内存空间中分配空间,用来存储"abc"的内存地址。从此,str所代表的内存空间中存放的是"abc"在方法区内存中的常量池中的内存地址。
        
        

在这里插入图片描述

  • 关于String str = new String(“def”);的理解。

    根据我们已经学到的知识进行思考,
    	String str = new String("def");
    	String str = "def";
    哪种使用方式更好?为什么?
    	第二种方式更好!
    	第一种方式有什么问题?会被回收,第一种方式还会在堆中创建一个对象。堆中要多一个对象,那么就要带来一系列的开销,比如在堆中创建对象需要时间、空间的代价。而且,这种创建的方式是在程序运行过程中,类加载完毕以后才做的。new关键字,就是要在堆中创建对象。
    
    • 关于new String()的理解
    String str = "abc";
    ==等价于
    char[] char = {'a','b','c'};
    String str = new String(char);
    
    String str = new String("abc");
    (**********)API中已经说明了,不推荐使用该构造器进行字符串的构造。
    in other words, the newly created string is a copy of the argument string. Unless an explicit copy of {@code original} is needed, use of this constructor is unnecessary since Strings are immutable.
    
    ...................................................
    public class StringTest {
        public static void main(String[] args) {
            String str = new String("he");
            String str2 = "he";
    
            System.out.println(str == str2); // false
            System.out.println(str.hashCode() == str2.hashCode()); // true
        }
    }
    
    详细解释关于new String("he")的执行过程
     第一步:调用new操作符,现在堆中创建String类型的对象,
     		有两个属性,第一个是char[] value;
     		第二个是int hash;创建对象就是在堆中给对象开辟空间,
     		给属性也开辟属性类型相应大小的空间。
     第二步:给char[]开辟空间的时候,其实先找到了常量池中的"he"对象。
     		然后将这个对象的char[] value数组的地址给到了堆中的value
     		源码中也是这样的。this.value = original.value;
     第四步:将字符串常量池中存放"he"的地址使用hash算法进行计算,
     		存入int hash属性中。
     第五步:在栈中开辟空间用来存放堆内存中String对象的引用(地址),
     		栈中的引用指向堆中的对象。
     		所以:str == str2 是false
    关于 str.hashCode() == str2.hashCode()
    		this.hash = original.hash;
    		调用hashCode方法后,hash值产生,其实只有一个值。
    		
    		使用String str = new String("he");常量池中先有了"he",使用new构造字符串从本质上来讲创建了两个对象,一个对象是在堆中的String类型的对象,一个是在字符串常量池中的"he"。
     
     
     		
    

在这里插入图片描述

3 补充 idea配置javap在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

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

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值