String构造方法初始化和常量赋值初始化区别

下面的代码是一个String对象的两种不同的初始化方式,关于这两种不同初始化方式的区别,本文通过画内存图来进行解释,首先代码如下:

public class Test {
public static void main(String[] args) {
String s1 = new String("hello");
String s2 = "hello";
System.out.println(s1 == s2);
System.out.println(s1.equals(s2)); } }

s1是通过String类的构造方法进行初始化的,s2是通过字符串常量进行赋值初始化的,该程序在内存中的图如下:

字符串常量都是存储在方法区的字符串常量池中,所以第一种方式的初始化是先在堆中建立一个对象,然后该对象的值来自方法区的“hello”,所以方式1的初始化方式最多会创建2个对象,最少创建1个;方式2的初始化由于是字符串常量直接赋值,所以直接去方法区里面查找,最后返回已有对象,所以方式2的初始化方式最多创建1个对象,最少创建0个,例如本例中,由于方法区中已有“hello”,所以创建0个。

String常量值不可以变,但引用可以变

直接上代码来解释了:

public class Test {
public static void main(String[] args) {
String s = "hello";
String s1 = s;
System.out.println("s: " + s);
System.out.println("s1: " + s1);
System.out.println("---------------");
s = "world";
System.out.println("s: " + s);
System.out.println("s1: " + s1);
}
}

结果:

s: hello
s1: hello
---------------
s: world
s1: hello Process finished with exit code 0

可以看出,对s进行重新复制,并没有改变原来的hello,只是将s指向了新的“world”。

String中常量拼接和变量拼接效果不同

首先看下面这段代码中的helloworld的不同拼接方式:

public class Test {
public static void main(String[] args) {
String s1 = "hello";
String s2 = "world";
String s3 = "helloworld";
System.out.println(s3 == s1 + s2);
System.out.println(s3.equals(s1 + s2));
System.out.println("-------------------");
System.out.println(s3 == "hello" + "world");
System.out.println(s3.equals("hello" + "world"));
}
}

程序运行的结果是:

false
true
-------------------
true
true Process finished with exit code 0

equals的比较由于内容相同,为true并不奇怪,但是变量拼接和s3比较的结果为false,而常量拼接结果却为true。这就是String中常量拼接和变量拼接的区别了。

  • 变量拼接:先开辟空间,再去拼接;
    所以s1+s2结果存储在新开辟的空间中,地址自然和s3不一样。反编译的代码是:

    System.out.println(s3 == (new StringBuffer(String.valueOf(s1)).append(String.valueOf(s2))).toString());
  • 常量拼接:先拼接,再在常量池中查找,如果有,则返回找到的地址,否则创建新的常量。
    所以”hello“+”world“拼接后再常量池中找到了s3,所以返回s3的地址,所以结果为true。反编译的代码是:
    System.out.println(s3 == "helloworld");//所以可以看出,编译器对变量的拼接直接做了优化
    
    

内容为空和null的区别

直接上代码看结果解释:

public class Test {
public static void main(String[] args) {
String s1 = "";
String s2 = null;
System.out.println(s1.isEmpty());
System.out.println(s2.isEmpty());
}
}
true
Exception in thread "main" java.lang.NullPointerException
at Test.main(Test.java:6)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:606)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140) Process finished with exit code 1

很明显,s1是可以调用String对象的方法的,而s2由于是null的,所以s2没有指向任何字符串对象,所以调用对象的方法会报空指针错误。

【Java基础】关于String的总结的更多相关文章

  1. Java基础之String、StringBuffer、StringBuilder浅析

    Java基础之String.StringBuffer.StringBuilder浅析 一.前言: 位于java.lang包下的String.StringBuilder.StringBuffer一般都是 ...

  2. 【Java基础】String 相关知识点总结

    String 相关知识点总结 字符串的不可变性 概述 String 被声明为 final,因此它不可继承 在 Java8 中,String 内部使用 char 数组存储数据 public final ...

  3. Java基础之String中equals,声明方式,等大总结

    无论你是一个编程新手还是老手,提到String你肯定感觉特别熟悉,因为String类我们在学习java基础的时候就已经学过,但是String类型有我们想象的那么简单吗?其实不然,String类型的知识 ...

  4. Java 基础之 String 类

    String String 被声明为 final,因此不能被继承.(Integer 等包装类也不能被继承) 在 java8 中,String 内部使用 char 数组 来存储数据 public fin ...

  5. java基础(六)-----String性质深入解析

    本文将讲解String的几个性质. 一.String的不可变性 对于初学者来说,很容易误认为String对象是可以改变的,特别是+链接时,对象似乎真的改变了.然而,String对象一经创建就不可以修改 ...

  6. Java基础之String、StringBuffer、StringBuilder

    1:String类:字符串(重点) (1)多个字符组成的一个序列,叫字符串. 生活中很多数据的描述都采用的是字符串的.而且我们还会对其进行操作. 所以,java就提供了这样的一个类供我们使用. (2) ...

  7. Java基础-关键字-String

    1.String的本质 线程安全 打开String的源码,类注释中有这么一段话“Strings are constant; their values cannot be changed after t ...

  8. Java基础(36):String与基本数据类型之间的双向转换(Wrapper类)

    Java 中基本类型和字符串之间的转换 在程序开发中,我们经常需要在基本数据类型和字符串之间进行转换. 其中,基本类型转换为字符串有三种方法: 1. 使用包装类的 toString() 方法 2. 使 ...

  9. Java基础(32):String与StringBuilder、StringBuffer的区别(String类)

    在Java中,除了可以使用 String 类来存储字符串,还可以使用 StringBuilder 类或 StringBuffer 类存储字符串,那么它们之间有什么区别呢? String 类具有是不可变 ...

  10. Java基础(30):String对象的常用方法与实例(String类)

    Java 中 String 类的常用方法 Ⅰ String 类提供了许多用来处理字符串的方法,例如,获取字符串长度.对字符串进行截取.将字符串转换为大写或小写.字符串分割等,下面我们就来领略它的强大之 ...

随机推荐

  1. RAC 安装完成后 节点间通信不依赖于SSH

    RAC 安装完成后,想修改ssh 的端口.google了一下.原文https://community.oracle.com/thread/2444594?tstart=0 原文说的是11g,10g也好 ...

  2. python AES 加密与解密

    #用aes加密,再用base64 encode def aes_encrypt(data): from Crypto.Cipher import AES import base64 key=setti ...

  3. 【一起学OpenFOAM】04 OpenFOAM的学习资源

    OpenFOAM的学习资料并不多,个人猜测也许是与软件的类型有关系. 对于商用软件来讲,由于要占领市场,软件开发商自然是巴不得会用软件的人越多越好,因为他们卖的是软件,会用的人越多,软件卖得越好.他们 ...

  4. php分页类的二种调用方法(转载)

    php分页类的二种调用方法 原文地址:http://www.xfcodes.com/php/fenye/25584.htm 导读:php分页类的二种调用用法,ajax调用php分页类,非ajax方式调 ...

  5. tomcat+apache 实现负载均衡之一:同一台电脑部署2个以上tomcat

    1.  下载tomcat 8.0.17 http://apache.fayea.com/tomcat/tomcat-8/v8.0.17/bin/apache-tomcat-8.0.17.tar.gz ...

  6. about mobile web

    http://blog.csdn.net/kavensu/article/details/8722268 http://cavenfeng.iteye.com/blog/1551516 http:// ...

  7. Ajax.BeginForm返回方法OnSuccess

    在MVC3里面——程序集 System.Web.Mvc.dll, v4.0.30319有这么一个Ajax.BeginForm异步登录验证的类型,我们在下面给出一个例子:在登录页面Logion.csht ...

  8. 体验Openstack--感觉这个比HADOOP还要基础啊

    当然,这两者应该是可以整到一起的..这个和ESX是什么关系叫?? 今天闲时慢慢按文档整了一个大概出来.. http://www.linuxidc.com/Linux/2013-01/78253p3.h ...

  9. 139. Word Break

    题目: Given a string s and a dictionary of words dict, determine if s can be segmented into a space-se ...

  10. WinForm实现简单的拖拽功能(C#)

    用到了ListBox和TreeView两个控件,ListBox作为数据源,通过拖拽其中的数据放置到TreeView上,自动添加一个树节点 ListBox控件的MouseDown用于获取要拖拽的值并调用 ...