org.rhq.core.pc.util.LoggingThreadFactory.java Source code

Java tutorial

Introduction

Here is the source code for org.rhq.core.pc.util.LoggingThreadFactory.java

Source

/*
 * RHQ Management Platform
 * Copyright (C) 2005-2008 Red Hat, Inc.
 * All rights reserved.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License, version 2, as
 * published by the Free Software Foundation, and/or the GNU Lesser
 * General Public License, version 2.1, also as published by the Free
 * Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License and the GNU Lesser General Public License
 * for more details.
 *
 * You should have received a copy of the GNU General Public License
 * and the GNU Lesser General Public License along with this program;
 * if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 */
package org.rhq.core.pc.util;

import java.lang.Thread.UncaughtExceptionHandler;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

/**
 * This thread factory just adds a {@link UncaughtExceptionHandler} to threads it creates so that there are logs of the
 * exceptions that cause threads to die.
 *
 * <p>Use this class when providing a thread factory to threadpools.</p>
 *
 * <p>This factory's constructor can be told to create daemon or non-daemon threads.</p>
 *
 * @author Greg Hinkle
 */
public class LoggingThreadFactory implements ThreadFactory, Thread.UncaughtExceptionHandler {
    private final Log log;
    private final String poolName;
    private final AtomicInteger threadNumber = new AtomicInteger(1);
    private final ThreadGroup group;
    private final boolean daemon;

    /**
     * Creates a factory that will produce either daemon or non-daemon threads. Be careful if you pass in <code>
     * false</code> for the <code>daemon</code> parameter; if you do, make sure your thread pool and all its threads
     * created by this factory are properly shutdown; otherwise you could prevent the VM from exiting properly.
     *
     * @param poolName the name of the thread pool that will be using this factory
     * @param daemon   if <code>true</code>, the factory will create daemon threads; <code>false</code> for non-daemon
     *                 threads
     */
    public LoggingThreadFactory(String poolName, boolean daemon) {
        this.poolName = poolName;
        this.log = LogFactory.getLog("org.rhq.threadpools." + poolName);
        this.daemon = daemon;

        SecurityManager s = System.getSecurityManager();
        group = (s != null) ? s.getThreadGroup() : Thread.currentThread().getThreadGroup();
    }

    /**
     * @see java.util.concurrent.ThreadFactory#newThread(Runnable)
     */
    public Thread newThread(Runnable r) {
        Thread t = new Thread(group, r, poolName + "-" + threadNumber.getAndIncrement());

        t.setDaemon(this.daemon);

        if (t.getPriority() != Thread.NORM_PRIORITY) {
            t.setPriority(Thread.NORM_PRIORITY);
        }

        t.setUncaughtExceptionHandler(this);

        return t;
    }

    /**
     * This simply logs the exception via this factory's logger.
     *
     * @see UncaughtExceptionHandler#uncaughtException(Thread, Throwable)
     */
    public void uncaughtException(Thread t, Throwable e) {
        log.error("Uncaught exception on scheduled thread [" + t.getName() + "]", e);
    }
}