ThreadUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3;

  18. import java.time.Duration;
  19. import java.util.Collection;
  20. import java.util.Collections;
  21. import java.util.Objects;
  22. import java.util.function.Function;
  23. import java.util.function.Predicate;
  24. import java.util.stream.Collectors;
  25. import java.util.stream.Stream;

  26. import org.apache.commons.lang3.time.DurationUtils;

  27. /**
  28.  * Helpers for {@link Thread} and {@link ThreadGroup}.
  29.  *
  30.  * <p>
  31.  * #ThreadSafe#
  32.  * </p>
  33.  *
  34.  * @see Thread
  35.  * @see ThreadGroup
  36.  * @since 3.5
  37.  */
  38. public class ThreadUtils {

  39.     /**
  40.      * A predicate implementation which always returns true.
  41.      *
  42.      * @deprecated Use a {@link Predicate}.
  43.      */
  44.     @Deprecated
  45.     private static final class AlwaysTruePredicate implements ThreadPredicate, ThreadGroupPredicate {

  46.         private AlwaysTruePredicate() {
  47.         }

  48.         @Override
  49.         public boolean test(final Thread thread) {
  50.             return true;
  51.         }

  52.         @Override
  53.         public boolean test(final ThreadGroup threadGroup) {
  54.             return true;
  55.         }
  56.     }

  57.     /**
  58.      * Used internally, consider private.
  59.      * <p>
  60.      * A predicate implementation which matches a thread or thread group name.
  61.      * </p>
  62.      *
  63.      * @deprecated Use a {@link Predicate}.
  64.      */
  65.     @Deprecated
  66.     public static class NamePredicate implements ThreadPredicate, ThreadGroupPredicate {

  67.         private final String name;

  68.         /**
  69.          * Constructs an instance.
  70.          *
  71.          * @param name thread or thread group name
  72.          * @throws NullPointerException if the name is {@code null}
  73.          */
  74.         public NamePredicate(final String name) {
  75.             Objects.requireNonNull(name, "name");
  76.             this.name = name;
  77.         }

  78.         @Override
  79.         public boolean test(final Thread thread) {
  80.             return thread != null && thread.getName().equals(name);
  81.         }

  82.         @Override
  83.         public boolean test(final ThreadGroup threadGroup) {
  84.             return threadGroup != null && threadGroup.getName().equals(name);
  85.         }
  86.     }

  87.     /**
  88.      * A predicate for selecting thread groups.
  89.      *
  90.      * @deprecated Use a {@link Predicate}.
  91.      */
  92.     @Deprecated
  93.     // When breaking BC, replace this with Predicate<ThreadGroup>
  94.     @FunctionalInterface
  95.     public interface ThreadGroupPredicate {

  96.         /**
  97.          * Evaluates this predicate on the given thread group.
  98.          * @param threadGroup the thread group
  99.          * @return {@code true} if the threadGroup matches the predicate, otherwise {@code false}
  100.          */
  101.         boolean test(ThreadGroup threadGroup);
  102.     }

  103.     /**
  104.      * A predicate implementation which matches a thread id.
  105.      *
  106.      * @deprecated Use a {@link Predicate}.
  107.      */
  108.     @Deprecated
  109.     public static class ThreadIdPredicate implements ThreadPredicate {

  110.         private final long threadId;

  111.         /**
  112.          * Predicate constructor
  113.          *
  114.          * @param threadId the threadId to match
  115.          * @throws IllegalArgumentException if the threadId is zero or negative
  116.          */
  117.         public ThreadIdPredicate(final long threadId) {
  118.             if (threadId <= 0) {
  119.                 throw new IllegalArgumentException("The thread id must be greater than zero");
  120.             }
  121.             this.threadId = threadId;
  122.         }

  123.         @Override
  124.         public boolean test(final Thread thread) {
  125.             return thread != null && thread.getId() == threadId;
  126.         }
  127.     }

  128.     /**
  129.      * A predicate for selecting threads.
  130.      *
  131.      * @deprecated Use a {@link Predicate}.
  132.      */
  133.     @Deprecated
  134.     // When breaking BC, replace this with Predicate<Thread>
  135.     @FunctionalInterface
  136.     public interface ThreadPredicate {

  137.         /**
  138.          * Evaluates this predicate on the given thread.
  139.          * @param thread the thread
  140.          * @return {@code true} if the thread matches the predicate, otherwise {@code false}
  141.          */
  142.         boolean test(Thread thread);
  143.     }

  144.     /**
  145.      * Predicate which always returns true.
  146.      *
  147.      * @deprecated Use a {@link Predicate}.
  148.      */
  149.     @Deprecated
  150.     public static final AlwaysTruePredicate ALWAYS_TRUE_PREDICATE = new AlwaysTruePredicate();

  151.     private static final Predicate<?> ALWAYS_TRUE = t -> true;

  152.     @SuppressWarnings("unchecked")
  153.     private static <T> Predicate<T> alwaysTruePredicate() {
  154.         return (Predicate<T>) ALWAYS_TRUE;
  155.     }

  156.     /**
  157.      * Finds the active thread with the specified id.
  158.      *
  159.      * @param threadId The thread id
  160.      * @return The thread with the specified id or {@code null} if no such thread exists
  161.      * @throws IllegalArgumentException if the specified id is zero or negative
  162.      * @throws  SecurityException
  163.      *          if the current thread cannot access the system thread group
  164.      *
  165.      * @throws  SecurityException  if the current thread cannot modify
  166.      *          thread groups from this thread's thread group up to the system thread group
  167.      */
  168.     public static Thread findThreadById(final long threadId) {
  169.         if (threadId <= 0) {
  170.             throw new IllegalArgumentException("The thread id must be greater than zero");
  171.         }
  172.         final Collection<Thread> result = findThreads((Predicate<Thread>) t -> t != null && t.getId() == threadId);
  173.         return result.isEmpty() ? null : result.iterator().next();
  174.     }

  175.     /**
  176.      * Finds the active thread with the specified id if it belongs to a thread group with the specified group name.
  177.      *
  178.      * @param threadId The thread id
  179.      * @param threadGroupName The thread group name
  180.      * @return The threads which belongs to a thread group with the specified group name and the thread's id match the specified id.
  181.      * {@code null} is returned if no such thread exists
  182.      * @throws NullPointerException if the group name is null
  183.      * @throws IllegalArgumentException if the specified id is zero or negative
  184.      * @throws  SecurityException
  185.      *          if the current thread cannot access the system thread group
  186.      *
  187.      * @throws  SecurityException  if the current thread cannot modify
  188.      *          thread groups from this thread's thread group up to the system thread group
  189.      */
  190.     public static Thread findThreadById(final long threadId, final String threadGroupName) {
  191.         Objects.requireNonNull(threadGroupName, "threadGroupName");
  192.         final Thread thread = findThreadById(threadId);
  193.         if (thread != null && thread.getThreadGroup() != null && thread.getThreadGroup().getName().equals(threadGroupName)) {
  194.             return thread;
  195.         }
  196.         return null;
  197.     }

  198.     /**
  199.      * Finds the active thread with the specified id if it belongs to the specified thread group.
  200.      *
  201.      * @param threadId The thread id
  202.      * @param threadGroup The thread group
  203.      * @return The thread which belongs to a specified thread group and the thread's id match the specified id.
  204.      * {@code null} is returned if no such thread exists
  205.      * @throws NullPointerException if {@code threadGroup == null}
  206.      * @throws IllegalArgumentException if the specified id is zero or negative
  207.      * @throws  SecurityException
  208.      *          if the current thread cannot access the system thread group
  209.      *
  210.      * @throws  SecurityException  if the current thread cannot modify
  211.      *          thread groups from this thread's thread group up to the system thread group
  212.      */
  213.     public static Thread findThreadById(final long threadId, final ThreadGroup threadGroup) {
  214.         Objects.requireNonNull(threadGroup, "threadGroup");
  215.         final Thread thread = findThreadById(threadId);
  216.         if (thread != null && threadGroup.equals(thread.getThreadGroup())) {
  217.             return thread;
  218.         }
  219.         return null;
  220.     }

  221.     /**
  222.      * Finds all active thread groups which match the given predicate.
  223.      *
  224.      * @param predicate the predicate
  225.      * @return An unmodifiable {@link Collection} of active thread groups matching the given predicate
  226.      * @throws NullPointerException if the predicate is null
  227.      * @throws  SecurityException
  228.      *          if the current thread cannot access the system thread group
  229.      * @throws  SecurityException  if the current thread cannot modify
  230.      *          thread groups from this thread's thread group up to the system thread group
  231.      * @since 3.13.0
  232.      */
  233.     public static Collection<ThreadGroup> findThreadGroups(final Predicate<ThreadGroup> predicate) {
  234.         return findThreadGroups(getSystemThreadGroup(), true, predicate);
  235.     }

  236.     /**
  237.      * Finds all active thread groups which match the given predicate and which is a subgroup of the given thread group (or one of its subgroups).
  238.      *
  239.      * @param threadGroup the thread group
  240.      * @param recurse if {@code true} then evaluate the predicate recursively on all thread groups in all subgroups of the given group
  241.      * @param predicate the predicate
  242.      * @return An unmodifiable {@link Collection} of active thread groups which match the given predicate and which is a subgroup of the given thread group
  243.      * @throws NullPointerException if the given group or predicate is null
  244.      * @throws  SecurityException  if the current thread cannot modify
  245.      *          thread groups from this thread's thread group up to the system thread group
  246.      * @since 3.13.0
  247.      */
  248.     public static Collection<ThreadGroup> findThreadGroups(final ThreadGroup threadGroup, final boolean recurse, final Predicate<ThreadGroup> predicate) {
  249.         Objects.requireNonNull(threadGroup, "threadGroup");
  250.         Objects.requireNonNull(predicate, "predicate");

  251.         int count = threadGroup.activeGroupCount();
  252.         ThreadGroup[] threadGroups;
  253.         do {
  254.             threadGroups = new ThreadGroup[count + count / 2 + 1]; //slightly grow the array size
  255.             count = threadGroup.enumerate(threadGroups, recurse);
  256.             //return value of enumerate() must be strictly less than the array size according to Javadoc
  257.         } while (count >= threadGroups.length);
  258.         return Collections.unmodifiableCollection(Stream.of(threadGroups).limit(count).filter(predicate).collect(Collectors.toList()));
  259.     }

  260.     /**
  261.      * Finds all active thread groups which match the given predicate and which is a subgroup of the given thread group (or one of its subgroups).
  262.      *
  263.      * @param threadGroup the thread group
  264.      * @param recurse if {@code true} then evaluate the predicate recursively on all thread groups in all subgroups of the given group
  265.      * @param predicate the predicate
  266.      * @return An unmodifiable {@link Collection} of active thread groups which match the given predicate and which is a subgroup of the given thread group
  267.      * @throws NullPointerException if the given group or predicate is null
  268.      * @throws  SecurityException  if the current thread cannot modify
  269.      *          thread groups from this thread's thread group up to the system thread group
  270.      * @deprecated Use {@link #findThreadGroups(ThreadGroup, boolean, Predicate)}.
  271.      */
  272.     @Deprecated
  273.     public static Collection<ThreadGroup> findThreadGroups(final ThreadGroup threadGroup, final boolean recurse, final ThreadGroupPredicate predicate) {
  274.         return findThreadGroups(threadGroup, recurse, (Predicate<ThreadGroup>) predicate::test);
  275.     }

  276.     /**
  277.      * Finds all active thread groups which match the given predicate.
  278.      *
  279.      * @param predicate the predicate
  280.      * @return An unmodifiable {@link Collection} of active thread groups matching the given predicate
  281.      * @throws NullPointerException if the predicate is null
  282.      * @throws  SecurityException
  283.      *          if the current thread cannot access the system thread group
  284.      * @throws  SecurityException  if the current thread cannot modify
  285.      *          thread groups from this thread's thread group up to the system thread group
  286.      * @deprecated Use {@link #findThreadGroups(Predicate)}.
  287.      */
  288.     @Deprecated
  289.     public static Collection<ThreadGroup> findThreadGroups(final ThreadGroupPredicate predicate) {
  290.         return findThreadGroups(getSystemThreadGroup(), true, predicate);
  291.     }

  292.     /**
  293.      * Finds active thread groups with the specified group name.
  294.      *
  295.      * @param threadGroupName The thread group name
  296.      * @return the thread groups with the specified group name or an empty collection if no such thread group exists. The collection returned is always unmodifiable.
  297.      * @throws NullPointerException if group name is null
  298.      * @throws  SecurityException
  299.      *          if the current thread cannot access the system thread group
  300.      *
  301.      * @throws  SecurityException  if the current thread cannot modify
  302.      *          thread groups from this thread's thread group up to the system thread group
  303.      */
  304.     public static Collection<ThreadGroup> findThreadGroupsByName(final String threadGroupName) {
  305.         return findThreadGroups(predicateThreadGroup(threadGroupName));
  306.     }

  307.     /**
  308.      * Finds all active threads which match the given predicate.
  309.      *
  310.      * @param predicate the predicate
  311.      * @return An unmodifiable {@link Collection} of active threads matching the given predicate
  312.      *
  313.      * @throws NullPointerException if the predicate is null
  314.      * @throws  SecurityException
  315.      *          if the current thread cannot access the system thread group
  316.      * @throws  SecurityException  if the current thread cannot modify
  317.      *          thread groups from this thread's thread group up to the system thread group
  318.      * @since 3.13.0
  319.      */
  320.     public static Collection<Thread> findThreads(final Predicate<Thread> predicate) {
  321.         return findThreads(getSystemThreadGroup(), true, predicate);
  322.     }

  323.     /**
  324.      * Finds all active threads which match the given predicate and which belongs to the given thread group (or one of its subgroups).
  325.      *
  326.      * @param threadGroup the thread group
  327.      * @param recurse if {@code true} then evaluate the predicate recursively on all threads in all subgroups of the given group
  328.      * @param predicate the predicate
  329.      * @return An unmodifiable {@link Collection} of active threads which match the given predicate and which belongs to the given thread group
  330.      * @throws NullPointerException if the given group or predicate is null
  331.      * @throws  SecurityException  if the current thread cannot modify
  332.      *          thread groups from this thread's thread group up to the system thread group
  333.      * @since 3.13.0
  334.      */
  335.     public static Collection<Thread> findThreads(final ThreadGroup threadGroup, final boolean recurse, final Predicate<Thread> predicate) {
  336.         Objects.requireNonNull(threadGroup, "The group must not be null");
  337.         Objects.requireNonNull(predicate, "The predicate must not be null");
  338.         int count = threadGroup.activeCount();
  339.         Thread[] threads;
  340.         do {
  341.             threads = new Thread[count + count / 2 + 1]; //slightly grow the array size
  342.             count = threadGroup.enumerate(threads, recurse);
  343.             //return value of enumerate() must be strictly less than the array size according to javadoc
  344.         } while (count >= threads.length);
  345.         return Collections.unmodifiableCollection(Stream.of(threads).limit(count).filter(predicate).collect(Collectors.toList()));
  346.     }

  347.     /**
  348.      * Finds all active threads which match the given predicate and which belongs to the given thread group (or one of its subgroups).
  349.      *
  350.      * @param threadGroup the thread group
  351.      * @param recurse if {@code true} then evaluate the predicate recursively on all threads in all subgroups of the given group
  352.      * @param predicate the predicate
  353.      * @return An unmodifiable {@link Collection} of active threads which match the given predicate and which belongs to the given thread group
  354.      * @throws NullPointerException if the given group or predicate is null
  355.      * @throws  SecurityException  if the current thread cannot modify
  356.      *          thread groups from this thread's thread group up to the system thread group
  357.      * @deprecated Use {@link #findThreads(ThreadGroup, boolean, Predicate)}.
  358.      */
  359.     @Deprecated
  360.     public static Collection<Thread> findThreads(final ThreadGroup threadGroup, final boolean recurse, final ThreadPredicate predicate) {
  361.         return findThreads(threadGroup, recurse, (Predicate<Thread>) predicate::test);
  362.     }

  363.     /**
  364.      * Finds all active threads which match the given predicate.
  365.      *
  366.      * @param predicate the predicate
  367.      * @return An unmodifiable {@link Collection} of active threads matching the given predicate
  368.      *
  369.      * @throws NullPointerException if the predicate is null
  370.      * @throws  SecurityException
  371.      *          if the current thread cannot access the system thread group
  372.      * @throws  SecurityException  if the current thread cannot modify
  373.      *          thread groups from this thread's thread group up to the system thread group
  374.      * @deprecated Use {@link #findThreads(Predicate)}.
  375.      */
  376.     @Deprecated
  377.     public static Collection<Thread> findThreads(final ThreadPredicate predicate) {
  378.         return findThreads(getSystemThreadGroup(), true, predicate);
  379.     }

  380.     /**
  381.      * Finds active threads with the specified name.
  382.      *
  383.      * @param threadName The thread name
  384.      * @return The threads with the specified name or an empty collection if no such thread exists. The collection returned is always unmodifiable.
  385.      * @throws NullPointerException if the specified name is null
  386.      * @throws  SecurityException
  387.      *          if the current thread cannot access the system thread group
  388.      *
  389.      * @throws  SecurityException  if the current thread cannot modify
  390.      *          thread groups from this thread's thread group up to the system thread group
  391.      */
  392.     public static Collection<Thread> findThreadsByName(final String threadName) {
  393.         return findThreads(predicateThread(threadName));
  394.     }

  395.     /**
  396.      * Finds active threads with the specified name if they belong to a thread group with the specified group name.
  397.      *
  398.      * @param threadName The thread name
  399.      * @param threadGroupName The thread group name
  400.      * @return The threads which belongs to a thread group with the specified group name and the thread's name match the specified name,
  401.      * An empty collection is returned if no such thread exists. The collection returned is always unmodifiable.
  402.      * @throws NullPointerException if the specified thread name or group name is null
  403.      * @throws  SecurityException
  404.      *          if the current thread cannot access the system thread group
  405.      *
  406.      * @throws  SecurityException  if the current thread cannot modify
  407.      *          thread groups from this thread's thread group up to the system thread group
  408.      */
  409.     public static Collection<Thread> findThreadsByName(final String threadName, final String threadGroupName) {
  410.         Objects.requireNonNull(threadName, "threadName");
  411.         Objects.requireNonNull(threadGroupName, "threadGroupName");
  412.         return Collections.unmodifiableCollection(findThreadGroups(predicateThreadGroup(threadGroupName)).stream()
  413.             .flatMap(group -> findThreads(group, false, predicateThread(threadName)).stream()).collect(Collectors.toList()));
  414.     }

  415.     /**
  416.      * Finds active threads with the specified name if they belong to a specified thread group.
  417.      *
  418.      * @param threadName The thread name
  419.      * @param threadGroup The thread group
  420.      * @return The threads which belongs to a thread group and the thread's name match the specified name,
  421.      * An empty collection is returned if no such thread exists. The collection returned is always unmodifiable.
  422.      * @throws NullPointerException if the specified thread name or group is null
  423.      * @throws  SecurityException
  424.      *          if the current thread cannot access the system thread group
  425.      *
  426.      * @throws  SecurityException  if the current thread cannot modify
  427.      *          thread groups from this thread's thread group up to the system thread group
  428.      */
  429.     public static Collection<Thread> findThreadsByName(final String threadName, final ThreadGroup threadGroup) {
  430.         return findThreads(threadGroup, false, predicateThread(threadName));
  431.     }

  432.     /**
  433.      * Gets all active thread groups excluding the system thread group (A thread group is active if it has been not destroyed).
  434.      *
  435.      * @return all thread groups excluding the system thread group. The collection returned is always unmodifiable.
  436.      * @throws  SecurityException
  437.      *          if the current thread cannot access the system thread group
  438.      *
  439.      * @throws  SecurityException  if the current thread cannot modify
  440.      *          thread groups from this thread's thread group up to the system thread group
  441.      */
  442.     public static Collection<ThreadGroup> getAllThreadGroups() {
  443.         return findThreadGroups(alwaysTruePredicate());
  444.     }

  445.     /**
  446.      * Gets all active threads (A thread is active if it has been started and has not yet died).
  447.      *
  448.      * @return all active threads. The collection returned is always unmodifiable.
  449.      * @throws  SecurityException
  450.      *          if the current thread cannot access the system thread group
  451.      *
  452.      * @throws  SecurityException  if the current thread cannot modify
  453.      *          thread groups from this thread's thread group up to the system thread group
  454.      */
  455.     public static Collection<Thread> getAllThreads() {
  456.         return findThreads(alwaysTruePredicate());
  457.     }

  458.     /**
  459.      * Gets the system thread group (sometimes also referred as "root thread group").
  460.      * <p>
  461.      * This method returns null if this thread has died (been stopped).
  462.      * </p>
  463.      *
  464.      * @return the system thread group
  465.      * @throws SecurityException if the current thread cannot modify thread groups from this thread's thread group up to the
  466.      *         system thread group
  467.      */
  468.     public static ThreadGroup getSystemThreadGroup() {
  469.         ThreadGroup threadGroup = Thread.currentThread().getThreadGroup();
  470.         while (threadGroup != null && threadGroup.getParent() != null) {
  471.             threadGroup = threadGroup.getParent();
  472.         }
  473.         return threadGroup;
  474.     }

  475.     /**
  476.      * Waits for the given thread to die for the given duration. Implemented using {@link Thread#join(long, int)}.
  477.      *
  478.      * @param thread The thread to join.
  479.      * @param duration How long to wait.
  480.      * @throws InterruptedException if any thread has interrupted the current thread.
  481.      * @see Thread#join(long, int)
  482.      * @since 3.12.0
  483.      */
  484.     public static void join(final Thread thread, final Duration duration) throws InterruptedException {
  485.         DurationUtils.accept(thread::join, duration);
  486.     }

  487.     private static <T> Predicate<T> namePredicate(final String name, final Function<T, String> nameGetter) {
  488.         return (Predicate<T>) t -> t != null && Objects.equals(nameGetter.apply(t), Objects.requireNonNull(name));
  489.     }

  490.     private static Predicate<Thread> predicateThread(final String threadName) {
  491.         return namePredicate(threadName, Thread::getName);
  492.     }

  493.     private static Predicate<ThreadGroup> predicateThreadGroup(final String threadGroupName) {
  494.         return namePredicate(threadGroupName, ThreadGroup::getName);
  495.     }

  496.     /**
  497.      * Sleeps the current thread for the given duration. Implemented using {@link Thread#sleep(long, int)}.
  498.      *
  499.      * @param duration How long to sleep.
  500.      * @throws InterruptedException if any thread has interrupted the current thread.
  501.      * @see Thread#sleep(long, int)
  502.      * @since 3.12.0
  503.      */
  504.     public static void sleep(final Duration duration) throws InterruptedException {
  505.         DurationUtils.accept(Thread::sleep, duration);
  506.     }

  507.     /**
  508.      * Sleeps for the given duration while ignoring {@link InterruptedException}.
  509.      * <p>
  510.      * The sleep duration may be shorter than duration if we catch a {@link InterruptedException}.
  511.      * </p>
  512.      *
  513.      * @param duration the length of time to sleep.
  514.      * @since 3.13.0
  515.      */
  516.     public static void sleepQuietly(final Duration duration) {
  517.         try {
  518.             sleep(duration);
  519.         } catch (final InterruptedException ignore) {
  520.             // Ignore & be quiet.
  521.         }
  522.     }

  523.     /**
  524.      * ThreadUtils instances should NOT be constructed in standard programming. Instead, the class should be used as {@code ThreadUtils.getAllThreads()}
  525.      * <p>
  526.      * This constructor is public to permit tools that require a JavaBean instance to operate.
  527.      * </p>
  528.      *
  529.      * @deprecated TODO Make private in 4.0.
  530.      */
  531.     @Deprecated
  532.     public ThreadUtils() {
  533.         // empty
  534.     }
  535. }