toString()、new String()和(String) 、String.ValueOf()的区别《转载》
先来看段代码:
String str="hello"; String base64Encoded=Hex.encodeToString(str.getBytes()); String str2=new String(Hex.decode(base64Encoded)); String str3=(Hex.decode(base64Encoded)).toString(); //str3的值为[B@2d65ca94 System.out.println(str2.equals(str)); System.out.println(str3.equals(str));
第一个输出结果为true,第二个输出结果为false这是为什么呢?
原来,toString()方法是调用了这个object对象的toString方法。一般是返回这么一个String:[class name]@[hashCode],即对象在内存中的地址。在这种使用方法中,因为java.lang.Object类里已有public方法.toString(),所以对任何严格意义上的java对象都可以调用此方法。但在使用时要注意,必须保证object不是null值,否则将抛出NullPointerException异常。采用这种方法时,通常派生类会覆盖Object里的toString()方法。
而new String(b)是根据parameter是一个字节数组,使用java虚拟机默认的编码格式,将这个字节数组decode为对应的字符。若虚拟机默认的编码格式是ISO-8859-1,按照ascii编码表即可得到字节对应的字符。所以**new String()**方法返回的才是真实的值
再来看(String)
Object obj = new Integer(100);
String strVal = (String)obj;
在运行时将会出错,因为将Integer类型强制转换为String类型,无法通过。如下图:
这是因为,这种标准的类型转换,将object转成String类型的值。使用这种方法时,需要注意的是类型必须能转成String类型。因此最好先用instanceof做个类型检查,以判断是否可以转换。否则容易抛出ClassCastException异常。此外,需特别小心的是因定义为Object 类型的对象在转成String时语法检查并不会报错,这将可能导致潜在的错误存在。这时要格外小心。
最后再来看String.valueOf()
这个方法是静态的,直接通过String调用,可以说是完美,只是平时使用不多,这样的实现避免了前面两个的不足和缺点。首先来看看他内部的实现机制:
public static String valueOf(Object obj){ return (obj==null) ? "null" : obj.toString() };
从这可以看出,其在内部就是作了为空的判断的,所以就不会报出空指针异常。
但我们也要很清醒地认识到,当object为null 时,String.valueOf(object)的值是字符串”null”,而不是null!!!在使用过程中切记要注意。我们可以使用以下代码进行检验:
Object obj = null; System.out.println(String.valueOf(obj) + "->此处null的类型是" + String.valueOf(obj).getClass()); System.out.println(obj);
我们看到的输出将是一模一样的东西:null,但它们意义其实并不相同。
原文链接:https://blog.csdn.net/shenxiaomo1688/article/details/81063823
补充一点:
String.valueOf()传入null值的问题 报空指针
原因:
原来,valueof方法被重载多次,其中String.valueof(null)进入了valueof(char[] data)方法,valueof(char[] data)方法直接进入String构造器,内部会获取char[]的length,因此返回空指针异常。String.valueOf(map.get(“id”))却进入了valueOf(Object obj)方法,参数为null时,返回了一个字符串“null”,所以出现了不同的结果。
那重载方法是如何被选择的呢?
多个重载方法均能匹配的条件下,优先会选择精度高的那个,或者说范围小的那个,那上面的问题就明朗了,char[]和object均能匹配null值,而char[]是继承自object的(java中数组也是一种特殊的object),因此string。valueOf(null)优先选择精度高的char[],而String.valueOf(map.get(“id”))中map的v值已经声明为了object,所以它只能进入valueOf(Object obj)方法。
总结:java编译器在选择重载方法时,如果重载方法参数个数一致,且具备多个重载方法可以匹配到所传递的参数,此时,会优先选择精度相对较高的,即java继承树种树的深度较深的那个。