• Java多线程——<六>更方便的线程


    一、概述

      标题很抽象,什么叫更方便?更是相比谁来说的呢?

      原来,我们定义任务,都是实现自Runnable或者Callable接口,但是这样必然需要你将新定义的任务附着给线程,然后再调用线程启动。在不考虑Executor的情况下,你会感觉这么做很呆板,如果直接new一个任务对象,他自己就能启动的话,是不是更好?这就是更方便的线程这个标题的由来。

    二、实现

      1.继承自Thread

      因为Thread实现了Runnable接口,所以继承自Thread的类,实现其run方法,其中定义的也就是任务。

      这样做的好处是,可以直接调用父类的start方法,可以实现概述中要实现的效果

    public class TaskExtendsThread extends Thread{
        private int countDown = 5;
        private static int threadCount = 0;
        /*
         * 可以在它的构造方法里直接调用run方法
         */
        public TaskExtendsThread(){
            super(Integer.toString(++threadCount));
            start();
        }
        public String toString(){
            return "#"+getName()+"("+countDown+").";
        }
        public void run()    {
            while(true){
                System.out.println(this);
                if(--countDown == 0 )
                    return ;
            }
        }
        public static void main(String[] args){
            for(int i = 0 ;i < 5 ;i++)
                new TaskExtendsThread();
        }
    }

      这样,在new一个对象时就会让你的任务自动附着给线程并启动。

      2.自管理的Runnable,实现Runnable接口。例如SelfManagedThread

      既然已经有了Thread类的实现,为什么还要有自管理Runnable呢?这样做和继承Thread没有区别,只是这样做会让你的类更灵活,实现了该类还可以继承其他你想让他继承的类。

    public class SelfManagedThread implements Runnable{
        private int countDown = 5;
        private Thread t = new Thread(this);
        /*
         * 在构造器中启动线程可能会变得很有问题
         * 因为另一个任务可能会在构造器结束之前开始执行,这意味着该任务能够访问处于不稳定状态的对象
         */
        public SelfManagedThread(){
            t.start();
        }
        public String toString(){
            return Thread.currentThread().getName()+"("+countDown+").";
        }
        @Override
        public void run() {
            while(true){
                System.out.println(this);
                if(--countDown==0)
                    return;
            }
        }
        public static void main(String[] args){
            for(int i = 0; i < 5 ;i++)
                new SelfManagedThread();
        }
    }

    三、然并卵

      这个大标题起的很好,因为目前为止,只知道这两种方式,都是在构造器中启动线程的,这可能会变的很有问题,因为另一个任务可能会在构造器结束之前开始执行,这意味着该任务可能访问处于不稳定状态的对象.故:优选Executor而不是显示地创建Thread对象的一个原因

      但是也不能说这样的解决方式完全没有用,待有朋友看到文章后给些提示……

    四、线程的术语

      至此,关于线程的实现,大部分总结完了,剩下的就是竞争资源、线程同步等并发方面的问题了。所以,在此总结下线程的术语

    术语
       |——线程并不是任务,线程不做任何事情只是驱动附着给它的任务执行
       |——Runnable接口理解:可以理解为该方法的反型封装,即它执行能做的事情
       |——其实任务应该实现一个名为Task的接口,这样更容易理解java中的线程
       |——从概念上讲:我们希望创建独立于其他任务运行的任务,因此我们应该能够定义任务,然后说“开始”,并且不用操心细节。
       |——java的线程机制基于来自c的低级的p线程方式,这事你必须深入研究,并且需要完全理解其所有事务的所有细节的方式,这种低级特性部分地渗入           了java的实现中,因此为了处于更高的抽象级别,在编写代码时,你必须遵循规则
       |——任务:在描述将要执行的工作时使用术语“任务”;在引用到驱动任务的具体机制时,使用属于“线程”

      注:以上代码均来自《Thinking in java》,总结及理解均来自个人,如有错误请大家批评指正。

  • 相关阅读:
    苹果推送APNS自己总结
    Windows下LDAP服务器配置
    service mysql start出错,mysql启动不了,解决mysql: unrecognized service错误
    PHP 中的 9 个魔术方法
    使用 Equinox 开发 OSGi 应用程序
    使用 Equinox 框架进行 OSGi 环境下的 Web 开发
    探索 OSGi 框架的组件运行机制
    基于 OSGi 和 Spring 开发 Web 应用
    Equinox OSGI ServletBridge 原理与实践
    利用 Eclipse 开发基于 OSGi 的 Bundle 应用
  • 原文地址:https://www.cnblogs.com/brolanda/p/4712630.html
Copyright © 2020-2023  润新知