Java基础知识
ffer:适用于多线程下在字符缓冲区进行大量操作的情况。
5.2示例
以拼接10000次字符串为例,我们看下三者各自需要的时间:
String str = "";
long startTime = System.currentTimeMillis();
for (int i = 0; i < 10000; i++) { str = str + i; } long endTime = System.currentTimeMillis(); long time = endTime - startTime; System.out.println("String消耗时间:" + time); StringBuilder builder = new StringBuilder(""); startTime = System.currentTimeMillis(); for (int j = 0; j < 10000; j++) { builder.append(j); } endTime = System.currentTimeMillis(); time = endTime - startTime; System.out.println("StringBuilder消耗时间:" + time); StringBuffer buffer = new StringBuffer(""); startTime = System.currentTimeMillis(); for (int k = 0; k < 10000; k++) { buffer.append(k); } endTime = System.currentTimeMillis(); time = endTime - startTime; System.out.println("StringBuffer消耗时间:" + time);
运行结果:
String消耗时间:258
StringBuilder消耗时间:0
StringBuffer消耗时间:1
也验证了上面所说的StringBuilder > StringBuffer > String。
6.装箱和拆箱
6.1什么是装箱?什么是拆箱?
装箱:自动将基本数据类型转换为包装器类型。
拆箱:自动将包装器类型转换为基本数据类型。
Integer i = 10; // 装箱
int j = i; // 拆箱
6.2 装箱和拆箱是如何实现的?
装箱过程是通过调用包装器的valueOf方法实现的,而拆箱过程是通过调用包装器实例的 xxxValue方法实现的。(xxx代表对应的基本数据类型)。
怎么证明这个结论呢,我们新建个类Main,在主方法中添加如下代码:
package com.zwwhnly.springbootdemo;
public class Main { public static void main(String[] args) { Integer i = 100; int j = i; } }
然后打开cmd窗口,切换到Main类所在路径,执行命令:javac Main.java,会发现该目录会生成一个Main.class文件,用IDEA打开,会发现编译后的代码如下:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package com.zwwhnly.springbootdemo; public class Main { public Main() { } public static void main(String[] var0) { Integer var1 = Integer.valueOf(100); int var2 = var1.intValue(); } }
6.3示例
示例1:
Double i1 = 100.0;
Double i2 = 100.0;
Double i3 = 200.0;
Double i4 = 200.0;
System.out.println(i1==i2);
System.out.println(i3==i4);
输出结果:
false
false
为什么都返回false呢,我们看下Double.valueOf()方法,就知晓了:
private final double value;
public Double(double value) { this.value = value; } public static Double valueOf(double d) { return new Double(d); }
示例2:
Boolean i1 = false;
Boolean i2 = false;
Boolean i3 = true;
Boolean i4 = true;
System.out.println(i1==i2);
System.out.println(i3==i4);
输出结果:
true
true
为什么都返回true呢,我们看下Boolean.valueOf()方法,就知晓了:
public static final Boolean TRUE = new Boolean(true); public static final Boolean FALSE = new Boolean(false); public static Boolean valueOf(boolean b) { return (b ? TRUE : FALSE); }
7.Java中的值传递和引用传递
7.1基本概念
值传递:传递对象的一个副本,即使副本被改变,也不会影响源对象,因为值传递的时候,实际上是将实参的值复制一份给形参。
引用传递:传递的并不是实际的对象,而是对象的引用,外部对引用对象的改变也会反映到源对象上,因为引用传递的时候,实际上是将实参的地址值复制一份给形参。
说明:对象传递(数组、类、接口)是引用传递,原始类型数据(整形、浮点型、字符型、布尔型)传递是值传递。
7.2示例
示例1(值传递):
package com.zwwhnly.springbootdemo;
public class ArrayListDemo { public static void main(String[] args) { int num1 = 10; int num2 = 20; swap(num1, num2); System.out.println("num1 = " + num1); System.out.println("num2 = " + num2); } public static void swap(int a, int b) { int temp = a; a = b; b = temp; System.out.println("a = " + a); System.out.println("b = " + b); } }
运行结果:
a = 20
b = 10
num1 = 10
num2 = 20
虽然在swap()方法中a,b的值做了交换,但是主方法中num1,num2的值并未改变。
示例2(引用类型传递):
package com.zwwhnly.springbootdemo;
public class ArrayListDemo { public static void main(String[] args) { int[] arr = {1, 2, 3, 4, 5}; change(arr); System.out.println(arr[0]); } public static void change(int[] array) { System.out.println(array[0]); array[0] = 0; } }
运行结果:
1
0
在change()方法中将数组的第一个元素改为0,主方法中数组的第一个元素也跟着变为0。
示例3(StringBuffer类型):
package com.zwwhnly.springbootdemo;
public class ArrayListDemo { public static void main(String[] args) { StringBuffer stringBuffer = new StringBuffer("博客园:周伟伟的博客"); System.out.println(stringBuffer); changeStringBuffer(stringBuffer); System.out.println(stringBuffer); } public static void changeStringBuffer(StringBuffer stringBuffer) { stringBuffer = new StringBuffer("掘金:周伟伟的博客"); stringBuffer.append(",欢迎大家关注"); } }
运行结果:
博客园:周伟伟的博客
博客园:周伟伟的博客
也许你会认为第2次应该输出“掘金:周伟伟的博客,欢迎大家关注”,怎么输出的还是原来的值呢,那是因为在changeStringBuffer中,又new了一个StringBuffer对象,此时stringBuffer对象指向的内存地址已经改变,所以主方法中的stringBuffer变量未受到影响。
如果修改changeStringBuffer()方法的代码为:
public static void changeStringBuffer(StringBuffer stringBuffer) { stringBuffer.append(",欢迎大家关注"); }
则运行结果变为了:
博客园:周伟伟的博客
博客园:周伟伟的博客,欢迎大家关注
示例4(String类型):
package com.zwwhnly.springbootdemo;
public class ArrayListDemo { public static void main(String[] args) { String str = new String("博客园:周伟伟的博客"); System.out.println(str); changeString(str); System.out.println(str); } public static void changeString(String string) { //string = "掘金:周伟伟的博客"; string = new String("掘金:周伟伟的博客"); } }
运行结果:
博客园:周伟伟的博客
博客园:周伟伟的博客
在changeString()方法中不管用
string = "掘金:周伟伟的博客";
还是string = new String("掘金:周伟伟的博客");
,主方法中的str变量都不会受影响,也验证了String创建之后是不可变更的。
示例5(自定义类型):
package com.zwwhnly.springbootdemo;
public class Person { private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public Person(String name) { this.name = name; } }
package com.zwwhnly.springbootdemo;
public class ArrayListDemo { public static void main(String[] args) { Person person = new Person("zhangsan"); System.out.println(person.getName()); changePerson(person); System.out.println(person.getName()); } public static void changePerson(Person p) { Person person = new Person("lisi"); p = person; } }
运行结果:
zhangsan
zhangsan
修改changePerson()方法代码为:
public static void changePerson(Person p) { p.setName("lisi"); }
则运行结果为:
zhangsan
lisi
8.参考链接
Java中的String,StringBuilder,StringBuffer三者的区别