001 /* 002 * Copyright (C) 2010 The Guava Authors 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 017 package com.google.common.util.concurrent; 018 019 import static com.google.common.base.Preconditions.checkArgument; 020 import static com.google.common.base.Preconditions.checkNotNull; 021 022 import java.lang.Thread.UncaughtExceptionHandler; 023 import java.util.concurrent.Executors; 024 import java.util.concurrent.ThreadFactory; 025 import java.util.concurrent.atomic.AtomicLong; 026 027 /** 028 * A ThreadFactory builder, providing any combination of these features: 029 * <ul> 030 * <li> whether threads should be marked as {@linkplain Thread#setDaemon daemon} 031 * threads 032 * <li> a {@linkplain ThreadFactoryBuilder#setNameFormat naming format} 033 * <li> a {@linkplain Thread#setPriority thread priority} 034 * <li> an {@linkplain Thread#setUncaughtExceptionHandler uncaught exception 035 * handler} 036 * <li> a {@linkplain ThreadFactory#newThread backing thread factory} 037 * </ul> 038 * If no backing thread factory is provided, a default backing thread factory is 039 * used as if by calling {@code setThreadFactory(}{@link 040 * Executors#defaultThreadFactory()}{@code )}. 041 * 042 * @author Kurt Alfred Kluever 043 * @since 4 044 */ 045 public final class ThreadFactoryBuilder { 046 private String nameFormat = null; 047 private Boolean daemon = null; 048 private Integer priority = null; 049 private UncaughtExceptionHandler uncaughtExceptionHandler = null; 050 private ThreadFactory backingThreadFactory = null; 051 052 /** 053 * Creates a new {@link ThreadFactory} builder. 054 */ 055 public ThreadFactoryBuilder() {} 056 057 /** 058 * Sets the naming format to use when naming threads ({@link Thread#setName}) 059 * which are created with this ThreadFactory. 060 * 061 * @param nameFormat a {@link String#format(String, Object...)}-compatible 062 * format String, to which a unique integer (0, 1, etc.) will be supplied 063 * as the single parameter. This integer will be unique to the built 064 * instance of the ThreadFactory and will be assigned sequentially. 065 * @return this for the builder pattern 066 */ 067 public ThreadFactoryBuilder setNameFormat(String nameFormat) { 068 String.format(nameFormat, 0); // fail fast if the format is bad or null 069 this.nameFormat = nameFormat; 070 return this; 071 } 072 073 /** 074 * Sets daemon or not for new threads created with this ThreadFactory. 075 * 076 * @param daemon whether or not new Threads created with this ThreadFactory 077 * will be daemon threads 078 * @return this for the builder pattern 079 */ 080 public ThreadFactoryBuilder setDaemon(boolean daemon) { 081 this.daemon = daemon; 082 return this; 083 } 084 085 /** 086 * Sets the priority for new threads created with this ThreadFactory. 087 * 088 * @param priority the priority for new Threads created with this 089 * ThreadFactory 090 * @return this for the builder pattern 091 */ 092 public ThreadFactoryBuilder setPriority(int priority) { 093 // Thread#setPriority() already checks for validity. These error messages 094 // are nicer though and will fail-fast. 095 checkArgument(priority >= Thread.MIN_PRIORITY, 096 "Thread priority (%s) must be >= %s", priority, Thread.MIN_PRIORITY); 097 checkArgument(priority <= Thread.MAX_PRIORITY, 098 "Thread priority (%s) must be <= %s", priority, Thread.MAX_PRIORITY); 099 this.priority = priority; 100 return this; 101 } 102 103 /** 104 * Sets the {@link UncaughtExceptionHandler} for new threads created with this 105 * ThreadFactory. 106 * 107 * @param uncaughtExceptionHandler the uncaught exception handler for new 108 * Threads created with this ThreadFactory 109 * @return this for the builder pattern 110 */ 111 public ThreadFactoryBuilder setUncaughtExceptionHandler( 112 UncaughtExceptionHandler uncaughtExceptionHandler) { 113 this.uncaughtExceptionHandler = checkNotNull(uncaughtExceptionHandler); 114 return this; 115 } 116 117 /** 118 * Sets the backing {@link ThreadFactory} for new threads created with this 119 * ThreadFactory. Threads will be created by invoking #newThread(Runnable) on 120 * this backing {@link ThreadFactory}. 121 * 122 * @param backingThreadFactory the backing {@link ThreadFactory} which will 123 * be delegated to during thread creation. 124 * @return this for the builder pattern 125 * 126 * @see MoreExecutors 127 */ 128 public ThreadFactoryBuilder setThreadFactory( 129 ThreadFactory backingThreadFactory) { 130 this.backingThreadFactory = checkNotNull(backingThreadFactory); 131 return this; 132 } 133 134 /** 135 * Returns a new thread factory using the options supplied during the building 136 * process. After building, it is still possible to change the options used to 137 * build the ThreadFactory and/or build again. State is not shared amongst 138 * built instances. 139 * 140 * @return the fully constructed {@link ThreadFactory} 141 */ 142 public ThreadFactory build() { 143 return build(this); 144 } 145 146 private static ThreadFactory build(ThreadFactoryBuilder builder) { 147 final String nameFormat = builder.nameFormat; 148 final Boolean daemon = builder.daemon; 149 final Integer priority = builder.priority; 150 final UncaughtExceptionHandler uncaughtExceptionHandler = 151 builder.uncaughtExceptionHandler; 152 final ThreadFactory backingThreadFactory = 153 (builder.backingThreadFactory != null) 154 ? builder.backingThreadFactory 155 : Executors.defaultThreadFactory(); 156 final AtomicLong count = (nameFormat != null) ? new AtomicLong(0) : null; 157 return new ThreadFactory() { 158 @Override public Thread newThread(Runnable runnable) { 159 Thread thread = backingThreadFactory.newThread(runnable); 160 if (nameFormat != null) { 161 thread.setName(String.format(nameFormat, count.getAndIncrement())); 162 } 163 if (daemon != null) { 164 thread.setDaemon(daemon); 165 } 166 if (priority != null) { 167 thread.setPriority(priority); 168 } 169 if (uncaughtExceptionHandler != null) { 170 thread.setUncaughtExceptionHandler(uncaughtExceptionHandler); 171 } 172 return thread; 173 } 174 }; 175 } 176 }