提交 c2f53c52 编写于 作者: M martin

6797480: Remove synchronization bottleneck in logger

Reviewed-by: swamyv
Contributed-by: jeremymanson@google.com
上级 070fe7ac
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
package java.util.logging; package java.util.logging;
import java.util.*; import java.util.*;
import java.util.concurrent.CopyOnWriteArrayList;
import java.security.*; import java.security.*;
import java.lang.ref.WeakReference; import java.lang.ref.WeakReference;
...@@ -165,10 +166,11 @@ public class Logger { ...@@ -165,10 +166,11 @@ public class Logger {
private static final int offValue = Level.OFF.intValue(); private static final int offValue = Level.OFF.intValue();
private LogManager manager; private LogManager manager;
private String name; private String name;
private ArrayList<Handler> handlers; private final CopyOnWriteArrayList<Handler> handlers =
new CopyOnWriteArrayList<Handler>();
private String resourceBundleName; private String resourceBundleName;
private boolean useParentHandlers = true; private volatile boolean useParentHandlers = true;
private Filter filter; private volatile Filter filter;
private boolean anonymous; private boolean anonymous;
private ResourceBundle catalog; // Cached resource bundle private ResourceBundle catalog; // Cached resource bundle
...@@ -180,9 +182,9 @@ public class Logger { ...@@ -180,9 +182,9 @@ public class Logger {
private static Object treeLock = new Object(); private static Object treeLock = new Object();
// We keep weak references from parents to children, but strong // We keep weak references from parents to children, but strong
// references from children to parents. // references from children to parents.
private Logger parent; // our nearest parent. private volatile Logger parent; // our nearest parent.
private ArrayList<WeakReference<Logger>> kids; // WeakReferences to loggers that have us as parent private ArrayList<WeakReference<Logger>> kids; // WeakReferences to loggers that have us as parent
private Level levelObject; private volatile Level levelObject;
private volatile int levelValue; // current effective level value private volatile int levelValue; // current effective level value
/** /**
...@@ -438,7 +440,7 @@ public class Logger { ...@@ -438,7 +440,7 @@ public class Logger {
* @exception SecurityException if a security manager exists and if * @exception SecurityException if a security manager exists and if
* the caller does not have LoggingPermission("control"). * the caller does not have LoggingPermission("control").
*/ */
public synchronized void setFilter(Filter newFilter) throws SecurityException { public void setFilter(Filter newFilter) throws SecurityException {
checkAccess(); checkAccess();
filter = newFilter; filter = newFilter;
} }
...@@ -448,7 +450,7 @@ public class Logger { ...@@ -448,7 +450,7 @@ public class Logger {
* *
* @return a filter object (may be null) * @return a filter object (may be null)
*/ */
public synchronized Filter getFilter() { public Filter getFilter() {
return filter; return filter;
} }
...@@ -465,10 +467,9 @@ public class Logger { ...@@ -465,10 +467,9 @@ public class Logger {
if (record.getLevel().intValue() < levelValue || levelValue == offValue) { if (record.getLevel().intValue() < levelValue || levelValue == offValue) {
return; return;
} }
synchronized (this) { Filter theFilter = filter;
if (filter != null && !filter.isLoggable(record)) { if (theFilter != null && !theFilter.isLoggable(record)) {
return; return;
}
} }
// Post the LogRecord to all our Handlers, and then to // Post the LogRecord to all our Handlers, and then to
...@@ -476,12 +477,8 @@ public class Logger { ...@@ -476,12 +477,8 @@ public class Logger {
Logger logger = this; Logger logger = this;
while (logger != null) { while (logger != null) {
Handler targets[] = logger.getHandlers(); for (Handler handler : logger.handlers) {
handler.publish(record);
if (targets != null) {
for (int i = 0; i < targets.length; i++) {
targets[i].publish(record);
}
} }
if (!logger.getUseParentHandlers()) { if (!logger.getUseParentHandlers()) {
...@@ -1182,13 +1179,10 @@ public class Logger { ...@@ -1182,13 +1179,10 @@ public class Logger {
* @exception SecurityException if a security manager exists and if * @exception SecurityException if a security manager exists and if
* the caller does not have LoggingPermission("control"). * the caller does not have LoggingPermission("control").
*/ */
public synchronized void addHandler(Handler handler) throws SecurityException { public void addHandler(Handler handler) throws SecurityException {
// Check for null handler // Check for null handler
handler.getClass(); handler.getClass();
checkAccess(); checkAccess();
if (handlers == null) {
handlers = new ArrayList<Handler>();
}
handlers.add(handler); handlers.add(handler);
} }
...@@ -1201,14 +1195,11 @@ public class Logger { ...@@ -1201,14 +1195,11 @@ public class Logger {
* @exception SecurityException if a security manager exists and if * @exception SecurityException if a security manager exists and if
* the caller does not have LoggingPermission("control"). * the caller does not have LoggingPermission("control").
*/ */
public synchronized void removeHandler(Handler handler) throws SecurityException { public void removeHandler(Handler handler) throws SecurityException {
checkAccess(); checkAccess();
if (handler == null) { if (handler == null) {
return; return;
} }
if (handlers == null) {
return;
}
handlers.remove(handler); handlers.remove(handler);
} }
...@@ -1217,11 +1208,8 @@ public class Logger { ...@@ -1217,11 +1208,8 @@ public class Logger {
* <p> * <p>
* @return an array of all registered Handlers * @return an array of all registered Handlers
*/ */
public synchronized Handler[] getHandlers() { public Handler[] getHandlers() {
if (handlers == null) { return handlers.toArray(emptyHandlers);
return emptyHandlers;
}
return handlers.toArray(new Handler[handlers.size()]);
} }
/** /**
...@@ -1235,7 +1223,7 @@ public class Logger { ...@@ -1235,7 +1223,7 @@ public class Logger {
* @exception SecurityException if a security manager exists and if * @exception SecurityException if a security manager exists and if
* the caller does not have LoggingPermission("control"). * the caller does not have LoggingPermission("control").
*/ */
public synchronized void setUseParentHandlers(boolean useParentHandlers) { public void setUseParentHandlers(boolean useParentHandlers) {
checkAccess(); checkAccess();
this.useParentHandlers = useParentHandlers; this.useParentHandlers = useParentHandlers;
} }
...@@ -1246,7 +1234,7 @@ public class Logger { ...@@ -1246,7 +1234,7 @@ public class Logger {
* *
* @return true if output is to be sent to the logger's parent * @return true if output is to be sent to the logger's parent
*/ */
public synchronized boolean getUseParentHandlers() { public boolean getUseParentHandlers() {
return useParentHandlers; return useParentHandlers;
} }
...@@ -1354,9 +1342,12 @@ public class Logger { ...@@ -1354,9 +1342,12 @@ public class Logger {
* @return nearest existing parent Logger * @return nearest existing parent Logger
*/ */
public Logger getParent() { public Logger getParent() {
synchronized (treeLock) { // Note: this used to be synchronized on treeLock. However, this only
return parent; // provided memory semantics, as there was no guarantee that the caller
} // would synchronize on treeLock (in fact, there is no way for external
// callers to so synchronize). Therefore, we have made parent volatile
// instead.
return parent;
} }
/** /**
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册