zoukankan      html  css  js  c++  java
  • java线程小结1

     1.创建线程的两种方法

      新线程的创建和启动都是通过java代码触发的。除了第一个线程(也就是启动程序的。运行main()方法的线程)是由java平台直接创建的之外,其余的线程都是在java代码中通过“创建Thread类的实例,启动线程”这种方式创建并启动的。

      当启动一个新的线程时,其过程是:由java代码通知java平台,java平台再启动线程。线程1启动线程2的过程实际就是线程1执行thread.start(),这对于线程1来说是一个很短的过程,因为启动线程的具体工作都是java平台做的,线程1只是通知而已,通知完了就继续执行代码,不等待线程2。

     a.继承java.lang.Thread类

      使用一个类去继承Thread类,然后为这个Threa类的子类添加一个run()方法,用来覆盖Thread类中原来的run()方法。

      使用继承自Thread类的子类的方法创建线程时,不需要向该子类中传入参数。如:MyThread thread = new MyThread();  //其中MyThread类是继承自Thread类的子类

     b.实现java.lang.Runnable接口

      使用一个实现了Runnable接口的类,在该类的内部有run()方法。

      使用实现了Runnable接口的类的方法创建线程时,需要向Thread中传入该类的实例。如:Thread thread = new Thread(MyRunnable);  //其中MyRunnable实现了Runnable接口的类的实例

     c.两种方式的比较

      实际中往往采用实现Runable接口,一方面因为java只支持单继承,继承了Thread类就无法再继续继承其它类,而且Runable接口只有一个run方法。

     2.start和run两种方法的区别

      run()方法:在本线程内调用该Runnable对象的run()方法,可以重复多次调用;
      start()方法:启动一个线程,调用该Runnable对象的run()方法,不能多次启动一个线程;

     a.start:

      用start方法来启动线程,真正实现了多线程运行,这时无需等待run方法体代码执行完毕而直接继续执行下面的代码。通过调用Thread类的start()方法来启动一个线程,这时此线程处于就绪(可运行)状态,并没有运行,一旦得到cpu时间片,就开始执行run()方法,这里方法 run()称为线程体,它包含了要执行的这个线程的内容,Run方法运行结束,此线程随即终止。

     b.run:

      run()方法只是类的一个普通方法而已,如果直接调用Run方法,程序中依然只有主线程这一个线程,其程序执行路径还是只有一条,还是要顺序执行,还是要等待run方法体执行完毕后才可继续执行下面的代码,这样就没有达到写线程的目的。

      run()方法中编写需要在独立线程内执行的代码,run()方法可以调用其他方法,但是执行的线程总是通过调用run()方法开始的。没有参数的run()方法是自动被调用的,而带参数的run()方法是被重载的,必须显式调用。

      总结:调用start方法方可启动线程,而run方法只是thread的一个普通方法调用,还是在主线程里执行。这两个方法应该都比较熟悉,把需要并行处理的代码放在run()方法中,start()方法启动线程将自动调用 run()方法,这是由jvm的内存机制规定的。并且run()方法必须是public访问权限,返回值类型为void。

     3.实例

     1 package com.ljq.test;
     2 
     3 public class ThreadTest {
     4     
     5     /**
     6      * 观察直接调用run()和用start()启动一个线程的差别 
     7      * 
     8      * @param args
     9      * @throws Exception
    10      */
    11     public static void main(String[] args){
    12         Thread thread=new ThreadDemo();
    13         //第一种
    14         //表明: run()和其他方法的调用没任何不同,main方法按顺序执行了它,并打印出最后一句
    15         //thread.run();
    16         
    17         //第二种
    18         //表明: start()方法重新创建了一个线程,在main方法执行结束后,由于start()方法创建的线程没有运行结束,
    19         //因此主线程未能退出,直到线程thread也执行完毕.这里要注意,默认创建的线程是用户线程(非守护线程)
    20         //thread.start();
    21         
    22         //第三种
    23         //1、为什么没有打印出100句呢?因为我们将thread线程设置为了daemon(守护)线程,程序中只有守护线程存在的时候,是可以退出的,所以只打印了七句便退出了
    24         //2、当java虚拟机中有守护线程在运行的时候,java虚拟机会关闭。当所有常规线程运行完毕以后,
    25         //守护线程不管运行到哪里,虚拟机都会退出运行。所以你的守护线程最好不要写一些会影响程序的业务逻辑。否则无法预料程序到底会出现什么问题
    26         //thread.setDaemon(true);
    27         //thread.start();
    28         
    29         //第四种
    30         //用户线程可以被System.exit(0)强制kill掉,所以也只打印出七句
    31         thread.start();
    32         System.out.println("main thread is over");
    33         System.exit(1);
    34     }
    35     
    36     public static class ThreadDemo extends Thread{
    37         @Override
    38         public void run() {
    39             for (int i = 0; i < 100; i++) {
    40                 System.out.println("This is a Thread test"+i);
    41             }
    42         }
    43     }
    44 }

      

  • 相关阅读:
    $GLOBALS超级全局变量
    归来
    Mscorlib.dll 里的 System.Internal 类是干嘛的?
    Query Composition using Functional Programming Techniques in C# 3.0
    反射之人千万不能错过的 EmitHelper
    给自己的Blog程序添加对Windows Live Writer的支持
    WebService的应用之winform身份验证
    c# static 的全部用法收集整理
    ASP.NET设置网站图标
    C# 2.0 之 static class (转)
  • 原文地址:https://www.cnblogs.com/Eason-S/p/5513733.html
Copyright © 2011-2022 走看看