Author: ivaynberg
Date: Mon May 17 21:22:32 2010
New Revision: 945374

URL: http://svn.apache.org/viewvc?rev=945374&view=rev
Log:
WICKET-2741 non-performant Collections.synchronizedMap() should be replaced 
with ConcurrentMap

Added:
    
wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java
   (with props)
Modified:
    
wicket/trunk/wicket-ioc/src/main/java/org/apache/wicket/injection/Injector.java

Modified: 
wicket/trunk/wicket-ioc/src/main/java/org/apache/wicket/injection/Injector.java
URL: 
http://svn.apache.org/viewvc/wicket/trunk/wicket-ioc/src/main/java/org/apache/wicket/injection/Injector.java?rev=945374&r1=945373&r2=945374&view=diff
==============================================================================
--- 
wicket/trunk/wicket-ioc/src/main/java/org/apache/wicket/injection/Injector.java 
(original)
+++ 
wicket/trunk/wicket-ioc/src/main/java/org/apache/wicket/injection/Injector.java 
Mon May 17 21:22:32 2010
@@ -18,14 +18,11 @@ package org.apache.wicket.injection;
 
 import java.lang.reflect.Field;
 import java.util.ArrayList;
-import java.util.Collections;
 import java.util.List;
-import java.util.Map;
-import java.util.WeakHashMap;
-import java.util.concurrent.ConcurrentHashMap;
 
 import org.apache.wicket.Application;
 import org.apache.wicket.MetaDataKey;
+import org.apache.wicket.util.collections.ClassMetaCache;
 
 /**
  * Injector scans fields of an object instance and checks if the specified
@@ -42,7 +39,7 @@ public abstract class Injector
                private static final long serialVersionUID = 1L;
        };
 
-       private final Map<ClassLoader, ConcurrentHashMap<String, Field[]>> 
cache = Collections.synchronizedMap(new WeakHashMap<ClassLoader, 
ConcurrentHashMap<String, Field[]>>());
+       private final ClassMetaCache<Field[]> cache = new 
ClassMetaCache<Field[]>();
 
        /**
         * Binds current instance of the injector to the Application. After 
this method is called this
@@ -88,20 +85,15 @@ public abstract class Injector
                Field[] fields = null;
 
                // try cache
-               ConcurrentHashMap<String, Field[]> container = 
cache.get(clazz.getClassLoader());
-               if (container != null)
-               {
-                       fields = container.get(clazz.getName());
-               }
+               fields = cache.get(clazz);
 
                if (fields == null)
                {
+                       // cache miss, discover fields
                        fields = findFields(clazz, factory);
 
                        // write to cache
-                       container = new ConcurrentHashMap<String, Field[]>();
-                       container.put(clazz.getName(), fields);
-                       cache.put(clazz.getClassLoader(), container);
+                       cache.put(clazz, fields);
                }
 
                for (int i = 0; i < fields.length; i++)

Added: 
wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java
URL: 
http://svn.apache.org/viewvc/wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java?rev=945374&view=auto
==============================================================================
--- 
wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java
 (added)
+++ 
wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java
 Mon May 17 21:22:32 2010
@@ -0,0 +1,101 @@
+package org.apache.wicket.util.collections;
+
+import java.util.Collections;
+import java.util.Map;
+import java.util.WeakHashMap;
+import java.util.concurrent.ConcurrentHashMap;
+
+/**
+ * This class wraps a WeakHashMap that holds one ConcurrentHashMap per 
ClassLoader. In the rare
+ * event of a previously unmapped ClassLoader, the WeakHashMap is replaced by 
a new one. This avoids
+ * any synchronization overhead, much like a {...@link 
java.util.concurrent.CopyOnWriteArrayList}
+ * 
+ * @param <T>
+ *            type of objects stored in cache
+ */
+public class ClassMetaCache<T>
+{
+       private volatile Map<ClassLoader, ConcurrentHashMap<String, T>> cache = 
Collections.emptyMap();
+
+       /**
+        * Puts value into cache
+        * 
+        * @param key
+        * @param value
+        * @return value previously stored in cache for this key, or {...@code 
null} if none
+        */
+       public T put(Class<?> key, T value)
+       {
+               ConcurrentHashMap<String, T> container = 
getClassLoaderCache(key.getClassLoader(), true);
+               return container.put(key(key), value);
+       }
+
+       /**
+        * Gets value from cache or returns {...@code null} if not in cache
+        * 
+        * @param key
+        * @return value stored in cache or {...@code null} if none
+        */
+       public T get(Class<?> key)
+       {
+               ConcurrentHashMap<String, T> container = 
getClassLoaderCache(key.getClassLoader(), false);
+               if (container == null)
+               {
+                       return null;
+               }
+               else
+               {
+                       return container.get(key(key));
+               }
+       }
+
+       /**
+        * @param classLoader
+        * @param create
+        * @return a {...@link ConcurrentHashMap} mapping class names to 
injectable fields, never
+        *         <code>null</code>
+        */
+       private ConcurrentHashMap<String, T> getClassLoaderCache(ClassLoader 
classLoader, boolean create)
+       {
+               ConcurrentHashMap<String, T> container = cache.get(classLoader);
+               if (container == null)
+               {
+                       if (!create)
+                       {
+                               return container;
+                       }
+
+                       // only lock in rare event of unknown ClassLoader
+                       synchronized (this)
+                       {
+                               // check again inside lock
+                               container = cache.get(classLoader);
+                               if (container == null)
+                               {
+                                       container = new 
ConcurrentHashMap<String, T>();
+
+                                       /*
+                                        * don't write to current cache, copy 
instead
+                                        */
+                                       Map<ClassLoader, 
ConcurrentHashMap<String, T>> newCache = new WeakHashMap<ClassLoader, 
ConcurrentHashMap<String, T>>(
+                                               cache);
+                                       newCache.put(classLoader, container);
+                                       cache = 
Collections.unmodifiableMap(newCache);
+                               }
+                       }
+               }
+               return container;
+       }
+
+       /**
+        * converts class into a key used by the cache
+        * 
+        * @param clazz
+        * 
+        * @return string representation of the clazz
+        */
+       private static String key(Class<?> clazz)
+       {
+               return clazz.getName();
+       }
+}
\ No newline at end of file

Propchange: 
wicket/trunk/wicket-util/src/main/java/org/apache/wicket/util/collections/ClassMetaCache.java
------------------------------------------------------------------------------
    svn:mime-type = text/plain


Reply via email to