关于java中String类!!!!

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于java中String类!!!!相关的知识,希望对你有一定的参考价值。

ava字符串类(java.lang.String)是Java中使用最多的类,也是最为特殊的一个类,很多时候,我们对它既熟悉又陌生。
  一、从根本上认识java.lang.String类和String池
  首先,我建议先看看String类的源码实现,这是从本质上认识String类的根本出发点。从中可以看到:
  1、String类是final的,不可被继承。public final class String。
  2、String类是的本质是字符数组char[], 并且其值不可改变。private final char value[];
  然后打开String类的API文档,可以发现:
  3、String类对象有个特殊的创建的方式,就是直接指定比如String x = "abc","abc"就表示一个字符串对象。而x是"abc"对象的地址,也叫
  做"abc"对象的引用。
  4、String对象可以通过“+”串联。串联后会生成新的字符串。也可以通过concat()来串联,这个后面会讲述。
  6、Java运行时会维护一个String Pool(String池),JavaDoc翻译很模糊“字符串缓冲区”。String池用来存放运行时中产生的各种字符串,
  并且池中的字符串的内容不重复。而一般对象不存在这个缓冲池,并且创建的对象仅仅存在于方法的堆栈区。
  5、创建字符串的方式很多,归纳起来有三类:
  其一,使用new关键字创建字符串,比如String s1 = new String("abc");
  其二,直接指定。比如String s2 = "abc";
  其三,使用串联生成新的字符串。比如String s3 = "ab" + "c";
  二、String对象的创建
  String对象的创建也很讲究,关键是要明白其原理。
  原理1:当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个X在String池中找是否存在内容相同的字符串对象,
  如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
  原理2:Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
  原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢
  了!但绝不会在堆栈区再去创建该String对象。
  原理4:使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。
  另外,String的intern()方法是一个本地方法,定义为public native String intern(); intern()方法的价值在于让开发者能将注意力集中到
  String池上。当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池
  中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。
  三、不可变类
  不可改变的字符串具有一个很大的优点:编译器可以把字符串设置为共享。
  不可变类String有一个重要的优点-它们不会被共享引用。
  是这样的,JAVA为了提高效率,所以对于String类型进行了特别的处理---为string类型提供了串池
  定义一个string类型的变量有两种方式:
  string name= "tom ";
  string name =new string( "tom ")
  使用第一种方式的时候,就使用了串池,
  使用第二中方式的时候,就是一种普通的声明对象的方式
  如果你使用了第一种方式,那么当你在声明一个内容也是 "tom "的string时,它将使用串池里原来的那个内存,而不会重新分配内存,也就是说,string saname= "tom ",将会指向同一块内存
  另外关于string类型是不可改变的问题:
  string类型是不可改变的,也就是说,当你想改变一个string对象的时候,比如name= "madding "
  那么虚拟机不会改变原来的对象,而是生成一个新的string对象,然后让name去指向它,如果原来的那个 "tom "没有任何对象去引用它,虚拟机的垃圾回收机制将接收它。
  据说这样可以提高效率!!!
参考技术A '+'运算符不会改变字符串本身,而是创建并返回一个包含改变后内容的新字符串对象 参考技术B 你要记住,String也是一个变量,所谓String的不变,是指你已经给某个String赋值了,然后向修改其中的某一个字符,这样说不行的~
在JAVA中,对String做+运算,就是将两个字符链接起来~
参考技术C 这个问题最好到博客园或csdn上去找,很详细的!

Java关于一个java.util.TreeSet的问题

import java.util.TreeSet;

class Drink implements Comparable
public String name;

public int compareTo(Object o)
return 0;


public class A
public static void main(String[] args)
Drink one = new Drink();
Drink two = new Drink();
one.name= "Coffee";
two.name= "Tea";
TreeSet set = new TreeSet();
System.out.println(set.add(one));
System.out.println(set.size());
System.out.println(set.add(two));
System.out.println(set.size());
for(Object d:set.toArray())
System.out.println(((Drink)d).name);




为什么第二次给set添加的时候失败
越详细越好

你好,看到这种问题,你最好能看看源码,我截取TreeSet代码如下:
public boolean add(E e)
return m.put(e, PRESENT)==null;

public TreeSet()
this(new TreeMap<E,Object>()); //默认构造

HashSet的内部实现就是一个new TreeMap<E,Object>(),继续看TreeMap:
public V put(K key, V value)
Entry<K,V> t = root;
if (t == null)
// TBD:
// 5045147: (coll) Adding null to an empty TreeSet should
// throw NullPointerException
//
// compare(key, key); // type check
root = new Entry<K,V>(key, value, null);
size = 1;
modCount++;
return null;

int cmp;
Entry<K,V> parent;
// split comparator and comparable paths
Comparator<? super K> cpr = comparator;
if (cpr != null)
do
parent = t;
cmp = cpr.compare(key, t.key);
if (cmp < 0)
t = t.left;
else if (cmp > 0)
t = t.right;
else
return t.setValue(value);
while (t != null);

else
if (key == null)
throw new NullPointerException();
Comparable<? super K> k = (Comparable<? super K>) key;
do
parent = t;
cmp = k.compareTo(t.key);
if (cmp < 0)
t = t.left;
else if (cmp > 0)
t = t.right;
else
return t.setValue(value);
while (t != null);

Entry<K,V> e = new Entry<K,V>(key, value, parent);
if (cmp < 0)
parent.left = e;
else
parent.right = e;
fixAfterInsertion(e);
size++;
modCount++;
return null;

看到这里我想你也明白了吧,这里就对调用对象的compareTo方法,而你的那个方法返回的正好是相等也就是0.所以在put的时候每次的键相同是不可以继续添加的,所以返回是false。
参考技术A 你的这个方法改成这样:
public int compareTo(Object o)
// TODO Auto-generated method stub
return o.hashCode();

就行了。。
因为Set是不能存放相同的值的。。而它是怎么判断值的呢。就是靠的上面的方法。。
你原代码都是返回0的,所以第二次的时候。Set认为你添加的是相同的值。。本回答被提问者采纳

以上是关于关于java中String类!!!!的主要内容,如果未能解决你的问题,请参考以下文章

关于java,String类,输出结果是啥?为啥?

Java—— 一点关于String的转换

关于java的String

Java关于一个java.util.TreeSet的问题

关于java中TreeSet类的一些问题

关于java的ThreadLocal的问题