潮汕美食网站怎么做/营销公司
开篇
为了挖掘 String 的小秘密, 我们先引入一个 Java 命令 javap
关于 javap
javap 是 JDK 自带的反汇编器,可以查看java编译器为我们生成的字节码。通过它,我们可以对照源代码和字节码,从而了解很多编译器内部的工作。
- 常用命令选项: -c 输出类中各方法的未解析的代码,即构成 Java 字节码的指令。
javap 能做些什么 ?
先来看一段简单的代码
public class Demo1 {public static void main(String[] args) {int i = 3;int j = 5;}}
使用 javap 命令得到的结果
- 首先在 DOS 命令行上 输入 javac Demo1.java 编译成 class 文件
- 最后输入 javap -c Demo1
输出结果
public class Demo1 {public Demo1();Code:0: aload_01: invokespecial #1 // Method java/lang/Object."<init>":()V4: returnpublic static void main(java.lang.String[]);Code:0: iconst_3 // 取值1: istore_1 // 赋值2: iconst_53: istore_24: return
}
结合源代码 和 输出结果 iconst_3 和 iconst_5 我们可以推断出是 取出常量值 3 和 5 而 istore_1 和 istore_2 是赋值给第一个变量和第二个变量
(其中const 是 常量 constant 的缩写 store是存储的意思)
经典的 i++ 和 ++i 问题
java源代码
public class Demo2 {public void fun1() {int i = 0;System.out.println(i++);}public void fun2() {int j = 0;System.out.println(++j);}}
你可以尝试用上面的 javap 命令去反编译这段代码, 看看你能不能分析出 两个自增方式的原理, 以及下面思考题
思考
int i = 0;for(int j=0; j<100; j++) {i = i++;}System.out.println("result: " + i);// 打印结果是多少 ?
String 介绍
String是 Java 中常用的一个类, 通过查看源代码可以看出, 该类由 final 修饰。
注意:String 是 不可变类 并不仅仅因为这一点
public final class String implements java.io.Serializable, Comparable<String>, CharSequence
String对象的创建
第一种方式
String s = "hello";//创建了一个对象
先去常量池中查看是否有相同的字符串
1.有的话直接返回引用
2.没有的话创建该字符串, 然后放入到常量池中, 最后返回对象引用
反编译结果
Code:0: ldc #2 // String helloworld2: astore_13: return
第二种方式
String s = new String("hello");//创建了两个对象
不管常量池中有没有该字符串对象, 都先会在堆中创建一个对象,
然后通过 String 对象的 intern() 方法 往常量池中引入 s 引用
反编译结果
Code:0: new #2 // class java/lang/String3: dup4: ldc #3 // String helloworld6: invokespecial #4 // Method java/lang/String."<init>":(Ljava/lang/String;)V9: astore_110: return
String常用方法
参考API文档
String拼接字符串
方式一
通过java中唯一被重载的操作符 " + " 拼接字符串
方式二
StringBuilder 的 append() 方法
方式三
StringBuffer 的 append() 方法
第一种方式的原理
测试代码
public class Demo6 {public static void main(String[] args) {String s = "hello";s += "world";}}
反编译结果
public class Demo6 {public Demo6();Code:0: aload_01: invokespecial #1 // Method java/lang/Object."<init>":()V4: returnpublic static void main(java.lang.String[]);Code:0: ldc #2 // String hello2: astore_13: new #3 // class java/lang/StringBuilder6: dup7: invokespecial #4 // Method java/lang/StringBuilder."<init>":()V10: aload_111: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;14: ldc #6 // String world16: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;19: invokevirtual #7 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;22: astore_123: return
}
通过反编译结果, 我们可以看出 当我们进行拼接字符串时, 创建了一个 StringBuilder 对象, 然后调用了它的 append 方法实现拼接字符串