Java结束线程

前言

       最近年底,项目紧很少写博客了,现在说说最近碰到的问题,笔者在做分布式JOB系统的时候,遇到一个比较棘手的问题:停止正在运行的线程。

1. 线程的生命周期

线程停止即Terminated状态是伴随run方法的结束而生,也就是run完成后由Thread类来决定线程停止了,销毁资源释放空间。

关于线程可以看我的另一篇文章:线程的实现,调度和生命周期,纯粹理论。

2. 手工停止

使用标记位法interrupt方法,我们知道有interrupt可以用来中断线程。

demo

public class StopThread {

    public static void main(String[] args) {
        Thread thread = new DemoThread();
        thread.start();
        thread.interrupt();
    }

    private static class DemoThread extends Thread{
        @Override
        public void run() {
            for (int i = 0; i < 1000000; i++) {
                System.out.println("i is " + i);
            }
        }
    }
}

运行,然而并没用,线程照常执行

看API

/**
     * Interrupts this thread.
     *
     * <p> Unless the current thread is interrupting itself, which is
     * always permitted, the {@link #checkAccess() checkAccess} method
     * of this thread is invoked, which may cause a {@link
     * SecurityException} to be thrown.
     *
     * <p> If this thread is blocked in an invocation of the {@link
     * Object#wait() wait()}, {@link Object#wait(long) wait(long)}, or {@link
     * Object#wait(long, int) wait(long, int)} methods of the {@link Object}
     * class, or of the {@link #join()}, {@link #join(long)}, {@link
     * #join(long, int)}, {@link #sleep(long)}, or {@link #sleep(long, int)},
     * methods of this class, then its interrupt status will be cleared and it
     * will receive an {@link InterruptedException}.
     *
     * <p> If this thread is blocked in an I/O operation upon an {@link
     * java.nio.channels.InterruptibleChannel InterruptibleChannel}
     * then the channel will be closed, the thread's interrupt
     * status will be set, and the thread will receive a {@link
     * java.nio.channels.ClosedByInterruptException}.
     *
     * <p> If this thread is blocked in a {@link java.nio.channels.Selector}
     * then the thread's interrupt status will be set and it will return
     * immediately from the selection operation, possibly with a non-zero
     * value, just as if the selector's {@link
     * java.nio.channels.Selector#wakeup wakeup} method were invoked.
     *
     * <p> If none of the previous conditions hold then this thread's interrupt
     * status will be set. </p>
     *
     * <p> Interrupting a thread that is not alive need not have any effect.
     *
     * @throws  SecurityException
     *          if the current thread cannot modify this thread
     *
     * @revised 6.0
     * @spec JSR-51
     */
    public void interrupt() {
        if (this != Thread.currentThread())
            checkAccess();

        synchronized (blockerLock) {
            Interruptible b = blocker;
            if (b != null) {
                interrupt0();           // Just to set the interrupt flag
                b.interrupt(this);
                return;
            }
        }
        interrupt0();
    }

    private native void interrupt0();

说了半天其实并不是终止线程,只是打了个status状态,且能被sleep join wait 清除掉,有了标记就好说了,我们可以在程序中判定标识,然后让线程快速return出栈,来停止线程。同时我们也可以自己通过标记位来提前终止线程。

public class StopThread {

    public static void main(String[] args) throws InterruptedException {
        Thread thread = new DemoThread();
        thread.start();
        Thread.sleep(1000);
        thread.interrupt();
    }

    private static class DemoThread extends Thread{
        @Override
        public void run() {
            for (int i = 0; i < 1000000; i++) {
                if (Thread.currentThread().isInterrupted())
                    return;
                System.out.println("i is " + i);
            }
        }
    }
}

可以看到

线程没运行完成结束了。

下面看看sleep方法的影响

public class StopThread {

    public static void main(String[] args) throws InterruptedException {
        Thread thread = new DemoThread();
        thread.start();
//        Thread.sleep(1000);
        thread.interrupt();
    }

    private static class DemoThread extends Thread{
        @Override
        public void run() {
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            for (int i = 0; i < 1000000; i++) {
                if (Thread.currentThread().isInterrupted())
                    return;
                System.out.println("i is " + i);
            }
        }
    }
}

看结果,标记位被清除了

为了应对这种情况,我们可以自己打标记位

public class StopThread {

    private static volatile boolean isStopped = false;

    public static void main(String[] args) throws InterruptedException {
        Thread thread = new DemoThread();
        thread.start();
        Thread.sleep(1000);
//        thread.interrupt();
        isStopped = true;
    }

    private static class DemoThread extends Thread{

        @Override
        public void run() {
            for (int i = 0; i < 1000000; i++) {
                if (isStopped)
                    return;
                System.out.println("i is " + i);
            }
        }
    }
}

结果符合预期

注意变量被volatile修饰,方便线程共享。

也可以使用抛异常方式结束线程,可以打印出堆栈信息

3. stop

线程提供stop方法,虽然废弃了,但迫不得已的时候仍然可以留着备用。

分析一下

/**
     * Forces the thread to stop executing.
     * <p>
     * If there is a security manager installed, its <code>checkAccess</code>
     * method is called with <code>this</code>
     * as its argument. This may result in a
     * <code>SecurityException</code> being raised (in the current thread).
     * <p>
     * If this thread is different from the current thread (that is, the current
     * thread is trying to stop a thread other than itself), the
     * security manager's <code>checkPermission</code> method (with a
     * <code>RuntimePermission("stopThread")</code> argument) is called in
     * addition.
     * Again, this may result in throwing a
     * <code>SecurityException</code> (in the current thread).
     * <p>
     * The thread represented by this thread is forced to stop whatever
     * it is doing abnormally and to throw a newly created
     * <code>ThreadDeath</code> object as an exception.
     * <p>
     * It is permitted to stop a thread that has not yet been started.
     * If the thread is eventually started, it immediately terminates.
     * <p>
     * An application should not normally try to catch
     * <code>ThreadDeath</code> unless it must do some extraordinary
     * cleanup operation (note that the throwing of
     * <code>ThreadDeath</code> causes <code>finally</code> clauses of
     * <code>try</code> statements to be executed before the thread
     * officially dies).  If a <code>catch</code> clause catches a
     * <code>ThreadDeath</code> object, it is important to rethrow the
     * object so that the thread actually dies.
     * <p>
     * The top-level error handler that reacts to otherwise uncaught
     * exceptions does not print out a message or otherwise notify the
     * application if the uncaught exception is an instance of
     * <code>ThreadDeath</code>.
     *
     * @exception  SecurityException  if the current thread cannot
     *               modify this thread.
     * @see        #interrupt()
     * @see        #checkAccess()
     * @see        #run()
     * @see        #start()
     * @see        ThreadDeath
     * @see        ThreadGroup#uncaughtException(Thread,Throwable)
     * @see        SecurityManager#checkAccess(Thread)
     * @see        SecurityManager#checkPermission
     * @deprecated This method is inherently unsafe.  Stopping a thread with
     *       Thread.stop causes it to unlock all of the monitors that it
     *       has locked (as a natural consequence of the unchecked
     *       <code>ThreadDeath</code> exception propagating up the stack).  If
     *       any of the objects previously protected by these monitors were in
     *       an inconsistent state, the damaged objects become visible to
     *       other threads, potentially resulting in arbitrary behavior.  Many
     *       uses of <code>stop</code> should be replaced by code that simply
     *       modifies some variable to indicate that the target thread should
     *       stop running.  The target thread should check this variable
     *       regularly, and return from its run method in an orderly fashion
     *       if the variable indicates that it is to stop running.  If the
     *       target thread waits for long periods (on a condition variable,
     *       for example), the <code>interrupt</code> method should be used to
     *       interrupt the wait.
     *       For more information, see
     *       <a href="{@docRoot}/../technotes/guides/concurrency/threadPrimitiveDeprecation.html">Why
     *       are Thread.stop, Thread.suspend and Thread.resume Deprecated?</a>.
     */
    @Deprecated
    public final void stop() {
        SecurityManager security = System.getSecurityManager();
        if (security != null) {
            checkAccess();
            if (this != Thread.currentThread()) {
                security.checkPermission(SecurityConstants.STOP_THREAD_PERMISSION);
            }
        }
        // A zero status value corresponds to "NEW", it can't change to
        // not-NEW because we hold the lock.
        if (threadStatus != 0) {
            resume(); // Wake up thread if it was suspended; no-op otherwise
        }

        // The VM can handle all thread states
        stop0(new ThreadDeath());
    }

Stopping a thread with Thread.stop causes it to unlock all of the monitors that it  has locked (as a natural consequence of the unchecked <code>ThreadDeath</code> exception propagating up the stack).

Stopping a thread with Thread.stop causes it to unlock all of the monitors that it  has locked

停止会对monitors造成unlock,造成对资源控制失控,推荐我们对run方法return来提前结束线程,interrupt用来中断wait

 

总结

       其实线程的停止就是让线程提前结束,即run方法通过标识提前返回,这是官方推荐的做法。运用线程运行完成自动结束的生命周期来停止线程,stop方法已废弃,不推荐使用,除非及其特殊的场景。

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值