对于String,StringBuffer,StringBuilder作为HashMap的Key的探讨

探讨了String和StringBuilder/StringBuffer作为HashMap键的区别,解释了为何String更适合作为键,涉及不可变性和hashCode的实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文: http://www.itkeyword.com/doc/1200285227501001789/string-vs-stringbuffer-as-hashmap-key

以下内容为对该文章的理解性翻译,部分认为无用的内容被删除

我试图理解为什么当用作HashMap密钥时,String和StringBuilder / StringBuffer被区别对待。让我通过以下内容使我的困惑更加清楚:

例1:使用String:

String s1 = new String("abc");
String s2 = new String("abc");
HashMap<String,Object> hm = new HashMap<>();
hm.put(s1, 1);
hm.put(s2, 2);
System.out.println(hm.size());
  • 上面的代码片段打印为“ 1”;

例2:使用StringBuilder(或StringBuffer):

StringBuilder sb1 = new StringBuilder("abc");
StringBuilder sb2 = new StringBuilder("abc");
HashMap<StringBuilder,Object> hm = new HashMap<>();
hm.put(sb1, 1);
hm.put(sb2, 2);
System.out.println(hm.size());
  • 上面的代码片段显示为“ 2”;

接下来是一些讨论(只翻译了其中的部分):

  • 问:造成这种不一致的确切区别是什么。两者都是内部char []。不是吗?

  • 答:

    • String是不可变的。StringBuilder是可变的,这意味着仅因为两个StringBuilder碰巧现在包含相同的文本,并不意味着它们将来会包含相同的文本

    • 理解:测试以下代码

      String s1 = new String("abc");
      String s2 = new String("abc");
      System.out.println(s1.hashCode());
      System.out.println(s2.hashCode());
      System.out.println("========================================");
      StringBuilder sb1 = new StringBuilder("abc");
      StringBuilder sb2 = new StringBuilder("abc");
      System.out.println(sb1.hashCode());
      System.out.println(sb2.hashCode()); 
      
    • StringBuilder / Buffer没有重写hashCode()和equals()函数。这意味着该对象的每个实例都应该是唯一的哈希码并且其值或状态无关紧要

    • 你应该使用String作为Key。StringBuilder / Buffer是可变的,将其用作HashMap的键通常不是一个好主意,因为在其下存储值可能会修改,导致以后无法访问原始值

    • StringBuilder使用Object默认的hashCode()函数实现,而String则通过比较字符串中的字符进行判断是否相等。Map的工作方式(特别是HashMap)是利用对象的hashcode,而不是类的内容(理解:其中保存的内容);

推荐:Java中String,StringBuffer,StringBuilder简述及区别

  • 查看源码:

    • 如果查看StringBuffer,它将发现它使用了Object.equals:
    public boolean equals(Object obj) {
        return (this == obj);
    }
    
    • 而String的实现是:
        public boolean equals(Object anObject) {
            if (this == anObject) {
                return true;
            }
            if (anObject instanceof String) {
                String anotherString = (String)anObject;
                int n = value.length;
                if (n == anotherString.value.length) {
                    char v1[] = value;
                    char v2[] = anotherString.value;
                    int i = 0;
                    while (n-- != 0) {
                        if (v1[i] != v2[i])
                            return false;
                        i++;
                    }
                    return true;
                }
            }
            return false;
        }
    
  • 接下来是一些争论:争论的中心点在于String的intern()那个技术点;

    • 你可以参考我的文章: https://blog.youkuaiyun.com/Hubz131/article/details/105158957
    • 如果我的文章不能是你满意,也可以自行搜索;
  • 现在,想象一下,如果我有两个具有相同哈希码值的对象,将会发生什么?在这种情况下,HashMap首先需要知道两个对象是否相同,因为如果它们相同,则意味着映射中只有一个条目,并且与该键关联的现有值将被新值替换。但是,仅具有相同的哈希码并不意味着两个密钥都相等。因此,是否相等是通过在关键对象上调用equals()方法来确定的。如果equals()返回true,则对象相等。在这种情况下,哈希映射需要更新现有条目的值。但是如果equals()返回false怎么办?在这种情况下,两个对象都不相同,应将其存储为单独的条目。因此,它们都将存储在HashMap中。 在检索值时,将使用输入值的哈希码到达存储它的隔离专区(:位桶数组的相应位置),然后,如果隔离专区包含多个对象(由于使用拉链法解决哈希碰撞的原因),则将检查输入键是否与隔离专区中的某个对象相等,如果匹配,则返回关联的值;

    • 仅具有相同的哈希码并不意味着两个密钥都相等:

    •   String s1 = new String("uP");
        String s2 = new String("v1");
        System.out.println(s1.hashCode());
        System.out.println(s2.hashCode());
        HashMap<String,Object> hm = new HashMap<>();
        hm.put(s1, 1);
        hm.put(s2, 2);
        System.out.println(hm.size());
      
  • 现在,让我们将上述理论应用于您拥有的代码。如果字符串对象的内容相等,则它们相等。并且根据规则,如果两个对象相等,则它们应返回相同的哈希码。但是请记住,相反是不正确的。如果两个对象返回相同的哈希码,则内容不一定相等。具有相同内容的两个字符串相等,即使它们在物理上是不同的对象,它们也会返回相同的哈希码,因此在HashMap中用作键时,它们始终会映射到同一条目。因此,就会出现以上现象(指的是例1)。

  • String类会覆盖默认的equals()方法,该方法表示两个对象如果具有相同的引用,而这些引用依赖于内容的相等性,则它们相等。之所以可以这样做是因为String是不可变的。但是,StringBuffer不会这样做。它仍然依赖于引用相等(地址)。

总结

三者相比,String更适合作为HashMap的键

1、StringBuffer/Builder作为键的话,如果后期内容发生了更改,就无法使用历史插入的那个值进行查询;

2、StringBuffer/Builder如果出现hashCode一样的话,不会对内容进行比较,直接回替换另一个相同hashCode的值,违背了正常思维(我是这么理解的,希望指正)

<think>好的,我现在需要帮助用户解决如何将Java中的HashMap转换为字符串,并使用自定义分隔符进行拼接的问题。首先,我得理解用户的需求。他们可能有一个HashMap,里面的键值对需要转换成字符串,键和值之间用某个分隔符,比如等号,然后每个键值对之间用另一个分隔符,比如逗号或者分号连接起来。例如,把{name: "John", age: 30}转换成"name=John,age=30"这样的形式。 接下来,我需要回忆一下Java中处理这种情况的常用方法。首先,HashMap的entrySet()方法可以获取所有键值对,然后遍历每个Entry,处理键和值。然后,使用StringJoiner或者StringBuilder来拼接字符串,这样可以更高效地处理字符串连接,避免多次创建字符串对象。 用户提到自定义分隔符,所以可能需要两个分隔符:一个用于键值之间的分隔,比如“=”,另一个用于键值对之间的分隔,比如“,”。这时候,使用StringJoiner可能比较方便,因为它允许指定分隔符和前后缀。不过,如果用户需要更灵活的处理,比如处理null值或者特殊字符,可能需要更多的逻辑。 另外,Java 8引入的Stream API也是一个不错的选择,可以通过stream().map()来处理每个Entry,然后collect(Collectors.joining(delimiter))来拼接。这种方法代码简洁,可读性高,适合现代Java版本。 然后,我需要考虑不同的Java版本。如果用户使用的是Java 8及以上,推荐使用Stream和StringJoiner;如果是旧版本,可能需要用StringBuilder手动拼接。同时,要注意处理可能出现的null值,比如键或值为null时,是否需要替换成空字符串或者其他占位符。 接下来,查看用户提供的引用内容。引用[2]提到优化StringBuffer的使用,预先分配大小。虽然现在更常用StringBuilder,但原理相同,预先估算大小可以减少扩容次数,提高性能。所以在使用StringBuilder时,可以考虑初始化容量,尤其是当HashMap较大时。 引用[1]中的代码示例涉及Groovy类,可能与问题无关,但里面有一个moveToNewPlace方法,处理输入映射和电话号码,可能涉及到如何操作Map,但具体和当前问题关联不大。引用[3]提到Oracle数据库相关的内容,可能也不相关。所以主要参考引用[2]关于字符串处理的优化建议。 现在,我需要整理步骤:遍历HashMap的Entry,将每个键值对转换成字符串,用键分隔符连接键和值,然后用条目分隔符连接所有键值对字符串。使用Java 8的Stream API或者StringJoiner来实现会更简洁。 可能的陷阱包括:键或值本身包含分隔符,导致解析时出错。但用户的问题可能不涉及这种情况,所以暂时不需要处理,但可以提醒用户注意。另外,处理null值的方式也需要明确,比如使用Objects.toString()方法避免NullPointerException。 然后,代码示例需要展示不同的方法,比如使用StringJoiner、Stream API和StringBuilder。每个方法的优缺点也需要说明,比如Stream API简洁但可能性能稍差于StringBuilder,但对于大多数情况足够。 最后,生成相关问题,可能涉及性能优化、处理空值、其他集合类型的转换等。确保问题与主题相关,帮助用户深入理解。 总结步骤: 1. 遍历HashMap的entrySet。 2. 对每个Entry,拼接键和值,使用键分隔符。 3. 将所有拼接后的字符串用条目分隔符连接。 4. 选择合适的方法(StringJoiner、Stream、StringBuilder)实现。 5. 处理可能的null值和性能优化。</think>### 如何将Java HashMap转换为字符串并自定义拼接格式 #### 核心方法 1. **使用`StringJoiner`(Java 8+)** ```java public static String mapToString(Map<?, ?> map, String keyValueDelimiter, String entryDelimiter) { StringJoiner result = new StringJoiner(entryDelimiter); for (Map.Entry<?, ?> entry : map.entrySet()) { result.add(entry.getKey() + keyValueDelimiter + entry.getValue()); } return result.toString(); } ``` - **优点**:代码简洁,自动处理分隔符。 - **示例调用**: ```java Map<String, String> map = new HashMap<>(); map.put("name", "John"); map.put("age", "30"); String output = mapToString(map, "=", ";"); // 输出 "name=John;age=30" ``` 2. **使用Stream API(Java 8+)** ```java public static String mapToStringStream(Map<?, ?> map, String keyValueDelimiter, String entryDelimiter) { return map.entrySet().stream() .map(entry -> entry.getKey() + keyValueDelimiter + entry.getValue()) .collect(Collectors.joining(entryDelimiter)); } ``` - **优点**:函数式编程风格,适合链式调用[^2]。 3. **手动拼接`StringBuilder`** ```java public static String mapToStringBuilder(Map<?, ?> map, String keyValueDelimiter, String entryDelimiter) { StringBuilder sb = new StringBuilder(); for (Map.Entry<?, ?> entry : map.entrySet()) { if (sb.length() > 0) { sb.append(entryDelimiter); } sb.append(entry.getKey()) .append(keyValueDelimiter) .append(entry.getValue()); } return sb.toString(); } ``` - **优点**:兼容旧版本Java,性能可控。 #### 扩展优化 - **处理`null`值**:使用`Objects.toString()`避免空指针: ```java entry.getKey() == null ? "null" : entry.getKey() ``` - **预分配容量**:若Map较大,初始化`StringBuilder`时指定容量(如`new StringBuilder(256)`)可减少扩容次数[^2]。 #### 示例输出对比 | 输入Map | 方法 | 输出结果 | |-----------------------|-----------------------|------------------------------| | `{name=John, age=30}` | `StringJoiner` | `name=John;age=30` | | `{id=100, valid=true}`| Stream API | `id=100,valid=true` |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值