第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(); }
总之:如果有更合适的数据类型,避免使用字符串来表示对象。