4

在 Java 中,有 3 个线程想要访问(只读)一个不可变的 hashmap 来做某事。是否SynchronizedMap低于为此目的最快的解决方案?如果没有,那么使用什么会更快?

import com.carrotsearch.hppc.IntObjectMap;
import com.carrotsearch.hppc.IntObjectOpenHashMap;

public class abc {
    public static void main(String[] args) {

        final IntObjectMap<int[]> map = new IntObjectOpenHashMap<int[]>();
        for (int i = 0; i < 4; i++) {
            map.put(i, new int[] {1, 2, 3, 4, 5});
        }

        Thread[] threads = new Thread[3];

        class SynchronizedMap {

            private final Object syncObject = new Object();

            public final int[] read(int i) {

                final int[] value;

                synchronized (syncObject) {

                    // code that reads-only immutable map object
                    value = map.get(i);

                }

                return value;

            }
        }

        final SynchronizedMap syncMap = new SynchronizedMap();

        class AccessMap implements Runnable {

            private int id;
            AccessMap(int index) { id = index; }

            public void run() {

                // code that reads-only immutable map object like this:
                for (int i = 0; i < 4; i++) {

                    final int[] array = syncMap.read(i);

                    for (int j = 0; j < array.length; j++)
                        System.out.println(id + ": " + array[j] + " ");

                }

            }
        }

        for (int i = 0; i < threads.length; i++) {
            threads[i] = new Thread(new AccessMap(i) {});
            threads[i].start();
        }

        for (int i = 0; i < threads.length; i++) {
            try {
                threads[i].join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

    }
}
4

1 回答 1

5

SynchronizedMap 类是否低于为此目的最快的解决方案?

不,如果HashMap是真正不可变/只读的,那么 avolatile Map<...>就是要走的路。

volatile IntObjectMap<int[]> readOnlyMap = new IntObjectOpenHashMap<int[]>();

如果您在构建地图启动线程,那么您甚至不需要volatile. 唯一需要的volatile是,如果您正在交换当前正在运行的线程正在访问的新映射。

final IntObjectMap<int[]> readOnlyMap = new IntObjectOpenHashMap<int[]>();
于 2013-04-01T20:27:01.900 回答