在Java中复制一个Map对象

5

我试图按照<Java如何复制一个对象?>中的方式进行操作。

但是在下面的代码中,这种方式不能用于Map对象。 我想将原始映射数据复制到currMap。当前输出为:

0
1
2
3
null
null
null

我希望它能够变成:

0
1
2
3
0
2
3

我错过了什么?

import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Queue;

class mapCopy{
    private Map<Character, Queue<Integer>> map;
    mapCopy(Map<Character, Queue<Integer>> map){
        this.map=map;
    }
    
    mapCopy(mapCopy mapcopy){
        this.map=mapcopy.map;
    }
    
    Map<Character, Queue<Integer>> getMap(){
        return this.map;
    }
}

public class Test {

    static Map<Character, Queue<Integer>> BuildMap(){
        String toMatch="able";
        Map<Character, Queue<Integer>> map = new HashMap<>();
        
        int i=0;
        for(var c:toMatch.toCharArray()) {
            Queue<Integer> q = map.get(c);
            if(q==null)
                q=new ArrayDeque<Integer>();
            
            q.add(i);
            map.put(c, q);          
            i++;
        }
        
        return map;
    }
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Map<Character, Queue<Integer>> map = BuildMap();
        
        List<String> dic = Arrays.asList("able", "ale");        
                        
        for(var d:dic) {
            var copy1 = new mapCopy(map);
            var copy2 = new mapCopy(copy1);
            
            var currMap = copy2.getMap();
            
            for(var c:d.toCharArray()) {        
                    System.out.println(currMap.get(c).poll());
            }
        }
    }

}

更新1:

iota的回答就是我所寻找的。以下是通过添加copyMap函数并将其添加到currMap (var currMap = copyMap(map);) 实现的实际代码。

不需要mapCopy类。

static Map<Character, Queue<Integer>> copyMap(Map<Character, Queue<Integer>> mapcopy){
    Map<Character, Queue<Integer>> map = new HashMap<>(mapcopy.size());
    mapcopy.forEach((k,v)->{
        map.put(k, new ArrayDeque<>(v));
    });
    
    return map;
}

更新2:
添加完整代码
import java.util.ArrayDeque;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Queue;

public class Test {

    static Map<Character, Queue<Integer>> BuildMap(){
        String toMatch="able";
        Map<Character, Queue<Integer>> map = new HashMap<>();
        
        int i=0;
        for(var c:toMatch.toCharArray()) {
            Queue<Integer> q = map.get(c);
            if(q==null)
                q=new ArrayDeque<Integer>();
            
            q.add(i);
            map.put(c, q);          
            i++;
        }
        
        return map;
    }
    
    static Map<Character, Queue<Integer>> copyMap(Map<Character, Queue<Integer>> mapcopy){
        Map<Character, Queue<Integer>> map = new HashMap<>(mapcopy.size());
        mapcopy.forEach((k,v)->{
            map.put(k, new ArrayDeque<>(v));
        });
        
        return map;
    }
    
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Map<Character, Queue<Integer>> map = BuildMap();
        
        List<String> dic = Arrays.asList("able", "ale");        
                        
        for(var d:dic) {
            var currMap = copyMap(map);

            for(var c:d.toCharArray()) {        
                    System.out.println(currMap.get(c).poll());
            }
        }
    }
}

3
你所指出的问题有一个被接受的答案,但它只适用于原始类型Map<Character, Queue<Integer>>不是原始类型,因此你不能简单地创建一个带有副本的虚拟类,它将包含对你创建的对象的引用。如果你需要深度克隆map变量,那么请创建一个新实例 - Arthur Attout
谢谢Arthur Attout。但是,如果我按照您链接中的流行示例,“var currMap = new HashMap<Character, Queue<Integer>>(map);”将无法工作。 - Don
这个回答解决了你的问题吗?Java HashMap - 深度拷贝 - CryptoFool
谢谢Steve!我不想使用库,只需要一些简单的代码。iota的解决方案就是我需要的。 - Don
1个回答

3

您可以直接遍历Map并将每个值复制到新的Map中。

mapCopy(mapCopy mapcopy){
    this.map = new HashMap<>(mapcopy.map.size());
    mapcopy.map.forEach((k,v)->{
        map.put(k, new ArrayDeque<>(v));
    });
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接