精通java的new关键

合集下载
相关主题
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

精通java的new关键
先来看一个例子,代码如下:
Java代码
1.public class Test {
2.public static void main(String[] args) {
3.String str = "abc";
4.String str1 = "abc";
5.String str2 = new String("abc");
6.System.out.println(str == str1);
7.System.out.println(str1 == "abc");
8.System.out.println(str2 == "abc");
9.System.out.println(str1 == str2);
10.System.out.println(str1.equals(str2));
11.System.out.println(str1 == str2.intern());
12.System.out.println(str2 == str2.intern());
13.System.out.println(str1.hashCode() == str2.
hashCode());
14.}
15.}
如果您能对这8个输出结果直接判断出来,下面的分析就不用看了。

但是我想还是有很多人对这个String对象这个问题只是表面的理解,下面就来分析一下Java语言String类和对象及其运行机制的问题。

做个基础的说明,堆(heap)内存和栈(Stack)内存的问题。

堆和栈的数据结构这里就不解释了。

Java语言使用内存的时候,栈内存主要保存以下内容:基本数据类型和对象的引用,而堆内存存储对象,栈内存的速度要快于堆内存。

总结成一句话就是:引用在栈而对象在堆。

Java中的比较有两种,是==和equals()方法,equals()是Object类的方法,定义在Object类中的equals()方法是如下实现的:
Java代码
1.public boolean equals(Object obj){
2.return (this==obj);
3.}
String类重写了equals()方法,改变了这些类型对象相等的原则,即判断对象是否相等依据的原则为判断二者的内容是否相等。

了解以上内容后我们来说说String,String类的本质是字符数组
char[],其次String类是final的,是不可被继承的,这点可能被大多数人忽略,再次String是特殊的封装类型,使用String时可以直接赋值,也可以用new来创建对象,但是这二者的实现机制是不同的。

还有一个String池的概念,
Java运行时维护一个String池,池中的String对象不可重复,没有创建,有则作罢。

String池不属于堆和栈,而是属于常量池。

下面分析上方代码的真正含义
Java代码
1.String str = "abc";
2.String str1= "abc";
第一句的真正含义是在String池中创建一个对象”abc”,然后引用时str指向池中的对象”abc”。

第二句执行时,因为”abc”已经存在于String 池了,所以不再创建,则str==str1返回true就明白了。

str1==”abc”肯定正确了,在String池中只有一个”abc”,而str和str1都指向池中的”abc”,就是这个道理。

Java代码
1.String str2 = new String("abc");
这个是Java SE的热点问题,众所周知,单独这句话创建了2个String 对象,而基于上面两句,只在栈内存创建str2引用,在堆内存上创建一个String 对象,内容是”abc”,而str2指向堆内存对象的首地址。

下面就是str2==”abc”的问题了,显然不对,”abc”是位于String 池中的对象,而str2指向的是堆内存的String对象,==判断的是地址,肯定不等了。

str1.equals(str2),这个是对的,前面说过,String类的equals重写了Object类的equals()方法,实际就是判断内容是否相同了。

下面说下intern()方法,在JavaDoc文档中,这样描述了intern()方法:返回字符串对象的规范化表示形式。

怎么理解这句话?实际上过程是这样进行的:该方法现在String池中查找是否存在一个对象,存在了就返回String 池中对象的引用。

那么本例中String池存在”abc”,则调用intern()方法时返回的是池中”abc”对象引用,那么和str/str1都是等同的,和str2就不同了,因为str2指向的是堆内存。

hashCode()方法是返回字符串内容的哈希码,既然内容相同,哈希码必然相同,那他们就相等了,这个容易理解。

再看下面的例子:
Java代码
1.public class Test {
2.private static String str = "abc";
3.public static void main(String[] args) {
4.String str1 = "a";
5.String str2 = "bc";
6.String combo = str1 + str2;
7.System.out.println(str == combo);
8.System.out.println(str == combo.intern());
9.}
10.}
这个例子用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,显然str==combo 是错误的,而str==combo.intern()是正确的,在String池中也存在”abc”,那就直接返回了,而str也是指向String池中的”abc”对象的。

此例说明任何重新修改String都是重新分配内存空间,这就使得String对象之间互不干扰。

也就是String中的内容一旦生成不可改变,直至生成新的对象。

同时问题也来了,使用+连接字符串每次都生成新的对象,而且是在堆内存上进行,而堆内存速度比较慢(相对而言),那么再大量连接字符串时直接+是不可取的,当然需要一种效率高的方法。

Java提供的StringBuffer和StringBuilder就是解决这个问题的。

区别是前者是线程安全的而后者是非线程安全的,StringBuilder在JDK1.5之后才有。

不保证安全的StringBuilder有比StringBuffer更高的效率。

自JDK1.5之后,Java虚拟机执行字符串的+操作时,内部实现也是StringBuilder,之前采用StringBuffer实现。

欢迎交流,希望对使用者有用。

JAVA中this用法小结
(2011-07-17 09:12:23)
转载▼
分类:JAVASE
标签:
java
this
Java关键字this只能用于方法方法体内。

当一个对象创建后,Java虚拟机(JVM)就会给这个对象分配一个引用自身的指针,这个指针的名字就是this。

因此,this 只能在类中的非静
态方法中使用,静态方法和静态的代码块中绝对不能出现this,这在“Java关键字static、final 使用总结”一文中给出了明确解释。

并且this只和特定的对象关联,而不和类关联,同一个类的不同对象有不同的this。

package test;
public class ThisTest {
private int i=0;
//第一个构造器:有一个int型形参
ThisTest(int i){
this.i=i+1;//此时this表示引用成员变量i,而非函数参数i
System.out.println("Int constructor i——this.i:
"+i+"——"+this.i);
System.out.println("i-1:"+(i-1)+"this.i+1:"+(this.i+1));
//从两个输出结果充分证明了i和this.i是不一样的!
}
// 第二个构造器:有一个String型形参
ThisTest(String s){
System.out.println("String constructor: "+s);
}
// 第三个构造器:有一个int型形参和一个String型形参
ThisTest(int i,String s){
this(s);//this调用第二个构造器
//this(i);
this.i=i++;//this以引用该类的成员变量
System.out.println("Int constructor: "+i+"/n"+"String constructor: "+s);
}
public ThisTest increment(){
this.i++;
return this;//返回的是当前的对象,该对象属于(ThisTest) }
public static void main(String[] args){
ThisTest tt0=new ThisTest(10);
ThisTest tt1=new ThisTest("ok");
ThisTest tt2=new ThisTest(20,"ok again!");
System.out.println(tt0.increment().increment().increment().i);
//tt0.increment()返回一个在tt0基础上i++的ThisTest对象,
//接着又返回在上面返回的对象基础上i++的ThisTest对象!
}
}
运行结果:
Int constructor i——this.i: 10——11
String constructor: ok
String constructor: ok again!
Int constructor: 21
String constructor: ok again!
14
细节问题注释已经写的比较清楚了,这里不在赘述,只是总结一下,其实this主要要三种用法:
1、表示对当前对象的引用!
2、表示用类的成员变量,而非函数参数,
注意在函数参数和成员变量同名是进行区
分!其实这是第一种用法的特例,比较常
用,所以那出来强调一下。

3、用于在构造方法中引用满足指定参数类
型的构造器(其实也就是构造方法)。


是这里必须非常注意:只能引用一个构造
方法且必须位于开始!
还有就是注意:this不能用在static方法中!所以甚至有人给static方法的定义就是:没有this的方法!虽然夸张,但是却充分说明this不能在static方法中使用!
说明在什么情况下需要用到this:
第一、通过this调用另一个构造方法,用发是this(参数列表),这个仅仅在类的构造方法中,别的地方不能这么用。

第二、函数参数或者函数中的局部变量
和成员变量同名的情况下,成员变量被屏蔽,此时要访问成员变量则需要用“this.成员变
量名”的方式来引用成员变量。

当然,在没有同名的情况下,可以直接用成员变量的名字,而不用this,用了也不为错,呵呵。

第三、在函数中,需要引用该函所属类的当前对象时候,直接用this。

其实这些用法总结都是从对“this是指向对象本身的一个指针”这句话的更深入的理解而来的,死记不然容易忘记而且容易搞错,要理解!。

相关文档
最新文档