第50条:如果其他类型更合适,则尽量避免使用字符串

1.字符串不适合代替其他的值类型。只有当数据确实是文本信息时,才应该使用字符串,如果是数值,就应该被转换为适当的数值类型,如果是一个“是-或-否”的问题答案,应该被转换为boolean类型,如果是一个对象,应该使用对象引用来引用它。

 

2.字符串不适合代替枚举类型:枚举类型比字符串更加适合用来表示枚举类型的常量。

 

3.字符串不适合代替聚集类型。如果一个实体有多个组件,用字符串来表示这个实体通常不恰当,String compundKey = className + "#" + i.next();

更好的做法是编写一个类来描述这个数据集,通常是一个私有的静态成员类。

 

4.字符串不适合代替能力表。有时候,字符串被用于对某种功能进行授权访问,考虑设计一个提供线程局部变量的机制,这个机制提供的变量在每个线程中都有自己的值。

public class ThreadLocal {
    private ThreadLocal() {}
    public static void set(String key, Object value);
    public static Object get(String key);
}

这种方法的问题在于,字符串键代表一个共享的全局命名空间,要使这种办法可行,客户端提供的字符串必须是唯一的,如果使用了相同的字符串,实际上就共享了这个变量。

 

修正,使用一个不可伪造的键,有时被称为能力表来代替字符串:

public class ThreadLocal {
    private ThreadLocal() {}
    public static class Key {
        Key() {}
    }
    public static Key getKey() {
        return new Key();
    }
    public static void set(Key key, Object value);
    public static Object get(Key key);
}

虽然解决基于字符串API的问题,但实际上可以使用ThreadLocal本身作为键,就不再需要额外的键了。

java.util.ThreadLocal提供的API正是这样的:

public T get() {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);//将自身作为键来取得线程局部变量值
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;
                return result;
            }
        }
        return setInitialValue();
}

 

总之:如果有更合适的数据类型,避免使用字符串来表示对象。

posted @ 2016-08-16 17:32  没有梦想的小灰灰  阅读(490)  评论(0编辑  收藏  举报