黑马程序员技术交流社区
标题:
求解释,关于StringBuilder的打印问题。
[打印本页]
作者:
刘源
时间:
2012-7-12 11:10
标题:
求解释,关于StringBuilder的打印问题。
本帖最后由 刘源 于 2012-7-14 10:25 编辑
class StringBuilderDemoop
{
public static void main(String[] args)
{
StringBuilder s = new StringBuilder("haha");
System.out.println(s.toString());
//打印结果都是haha
System.out.println(s);
//打印结果都是haha
}
}
这个是看毕老师视频的时候,StringBuilder是一个容器,为什么不用把它转换成字符串也能被打印啊,而且打印的也不是这个对象的地址而是内容,难度也像基本数据类型一样有一个自动装箱的过程吗。
我看了毕老师的打印都是把他转换成字符串再打印的。有明白人解释下吗?
或者说只有这个打印语句是特殊的不用转换成字符串,如果有其他的应用也不用转换的字符串的话,希望有人能举出其他的例子。万分感谢。
作者:
韦念欣
时间:
2012-7-12 11:13
System.out.println(s)这段代码,自动调用了s.toString()方法,所以两个效果是一样的。
作者:
张天天
时间:
2012-7-12 11:27
呵呵,这个问题貌似有点复杂哈。
首先System.out.println(s),println方法是PrintStream类的一个方法:
public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}
复制代码
这里调用了String类的valueOf方法对s进行处理
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}
复制代码
这里执行的显然是 obj.toString()
这里体现出了面向对象的多态了哈。
obj会指向子类型的引用,即StringBuilder中的toString方法:
public String toString() {
// Create a copy, don't share the array
return new String(value, 0, count);
}
复制代码
看到返回的是什么了么?我就不再解释了哈
作者:
王达
时间:
2012-7-12 11:32
class StringBuilderDemoop
{
public static void main(String[] args)
{
StringBuilder s = new StringBuilder("haha");
System.out.println(s.toString());
//打印结果都是haha
System.out.println(s);
//打印结果都是haha
}
}
而在你打印s的时候他会自动调用s的toString方法,而你的StringBuilder中放的是“haha”是一个字符串,所以就打印出haha了。
其实这个toString就是对你StringBuilder中放置的元素逐个进行toString如果有未覆盖toString方法的元素,这个元素所显示的打印值就是他的地址值
如:
package Test;
public class MainArgs {
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
StringBuilder s = new StringBuilder("hah"+"\r\n");
s.append(new Object());//
加入一个未覆盖toString()方法的对象。
System.out.println(s.toString());
System.out.println(s);
}
}
打印出来的值是
hah
java.lang.Object@16e1fb1
hah
java.lang.Object@16e1fb1
String类型的为其本身
而new Object()是他的地址值。
作者:
万宝东
时间:
2012-7-12 11:34
System.out.println(s.toString());//打印结果都是haha
System.out.println(s);//打印结果都是haha
这两句话的输出结果一样楼上的韦版主已经解释了,System.out.println(s);打印时会自动调用toString方法。
还有楼主的另一个疑问为什么打印出来的不是hashcode值,而是内容?这是因为StringBuilder s = new StringBuilder("haha");new出来的s是StringBuilder类型的,而StringBuilder类复写了Object类的toString方法,返回的是此字符序列的字符串表示形式。(可以查看Api文档)。
至于楼主看到的那些打印地址值的是因为打印的是Object类型的数据,他们调用的是Object的toString方法,Object的toString方法打印的是hashcode值。
作者:
苑占丽
时间:
2012-7-12 11:45
String 字符串常量
StringBuilder 字符串变量(非线程安全)
StringBuilder也是一个类呀。
一个可变的字符序列。此类提供一个与 StringBuffer 兼容的 API,但不保证同步。该类被设计用作 StringBuffer 的一个简易替换,用在字符串缓冲区被单个线程使用的时候(这种情况很普遍)。如果可能,建议优先采用该类,因为在大多数实现中,它比 StringBuffer 要快。 在 StringBuilder 上的主要操作是 append 和 insert 方法,可重载这些方法,以接受任意类型的数据。每个方法都能有效地将给定的数据转换成字符串,然后将该字符串的字符添加或插入到字符串生成器中。append 方法始终将这些字符添加到生成器的末端;而 insert 方法则在指定的点添加字符。 例如,如果 z 引用一个当前内容为“start”的字符串生成器对象,则该方法调用 z.append("le") 将使字符串生成器包含“startle”,而 z.insert(4, "le") 将更改字符串生成器,使之包含“starlet”。 通常,如果 sb 引用 StringBuilder 的实例,则 sb.append(x) 和 sb.insert(sb.length(), x) 具有相同的效果。 每个字符串生成器都有一定的容量。只要字符串生成器所包含的字符序列的长度没有超出此容量,就无需分配新的内部缓冲区。如果内部缓冲区溢出,则此容量自动增大。 将 StringBuilder 的实例用于多个线程是不安全的。如果需要这样的同步,则建议使用 StringBuffer。StringBuilder类可以用于在无需创建一个新的字符串对象情况下修改字符串。
StringBuilder类:
一个String对象的长度是固定的,不能改变它的内容,或者是附加新的字符至String对象中。您也许会使用+来串联字符串以达到附加新字符或字符串的目的,但+会产生一个新的String实例。如果程序对这种附加字符串的需求很频繁,并不建议使用+来进行字符串的串联。在面向对象程序设计中,最好是能重复运用已生成的对象,对象的生成需要内存空间与时间,不断地产生String实例是一个没有效率的行为。
J2SE 5.0提供java.lang.StringBuilder类,使用这个类所产生的对象默认会有16个字符的长度,您也可以自行指定初始长度。如果附加的字符超出可容纳的长度,则StringBuilder对象会自动增加长度以容纳被附加的字符。如果有频繁作字符串附加的需求,使用StringBuilder会让程序的效率大大提高。通过下面的简单测试程序就可以知道效能差距有多大。
作者:
刘源
时间:
2012-7-14 10:25
高手无处不在啊,都看懂了
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/)
黑马程序员IT技术论坛 X3.2