`

Thread和Runnable的区别

阅读更多

             使用多线程时,大家大抵都有过疑惑,到底是继承Thread呢,还是实现Runnable接口,我也有过这样的疑惑,于是便上网搜了下,这边就记下我自己的见解,欢迎指正批评。

    

              一般使用中,很多有经验的老程序员大部分都是不约而同的使用Runnable多一些,为什么呢,我也问过公司的那些老员工,他们也说不出个所以然,只是使用经验让他们去这样选择,这里我去查看Thread和Runnable的源代码,发觉Thread内部也是实现的Runnable的接口的,慢慢的好像有些头绪,Thread类片段源码如下:

 /**
     * If this thread was constructed using a separate 
     * <code>Runnable</code> run object, then that 
     * <code>Runnable</code> object's <code>run</code> method is called; 
     * otherwise, this method does nothing and returns. 
     * <p>
     * Subclasses of <code>Thread</code> should override this method. 
     *
     * @see     #start()
     * @see     #stop()
     * @see     #Thread(ThreadGroup, Runnable, String)
     */
    public void run() {
	if (target != null) {
	    target.run();
	}
    }

            这里写到如果target不为空的话,则调用它的run方法,这个target,则是Runnable的实例对象,也就是说,继承于Thread并实现它的run方法能创建一个线程,实际这个run方法就是Runnable所提供的代理对象所有的,因此我们得到Runnable的优点之后,再拿过来跟Thread的优先一一对比,也就大体能知道,使用时,到底选用Thread好还是Runnable好点。

           经对比,Runnable的优势有以下几点:1.适合多个相同程序代码的线程去处理同一个资源(多线程内的数据共享) 2.避免java特性中的单根继承限制  3.增加程序健壮性,数据被共享时,仍然可以保持代码和数据的分离和独立  4.更能体现java面向对象的设计特点

 

           针对以上的特点2,3,4我们大体可以理解,而对于优势1可能不太深刻,这边,为了体现它在多线程内的数据共享方面的优势,我们用张孝祥所举的经典的买票例子来演示下,多个窗口同时卖票,这里多个窗口即多个线程,所卖的票则是需要共享的数据。

 

           以下是Thread的实现方式:

class hello extends Thread {
    public void run() {
        for (int i = 0; i < 7; i++) {
            if (count > 0) {
                System.out.println("count= " + count--);
            }
        }
    }
 
    public static void main(String[] args) {
        hello h1 = new hello();
        hello h2 = new hello();
        hello h3 = new hello();
        h1.start();
        h2.start();
        h3.start();
    }
 
    private int count = 5;
}

 运行结果为:

count= 5

count= 4

count= 3

count= 2

count= 1

count= 5

count= 4

count= 3

count= 2

count= 1

count= 5

count= 4

count= 3

count= 2

count= 1

          以上结果显示,假使这真是一个买票系统count为总车票数的话,以上代码并未实现线程内的数据共享

 

下面我们看下Runnable的实现,代码如下:

class MyThread implements Runnable{
 
    private int ticket = 5;  //5张票
 
    public void run() {
        for (int i=0; i<=20; i++) {
            if (this.ticket > 0) {
                System.out.println(Thread.currentThread().getName()+ "正在卖票"+this.ticket--);
            }
        }
    }
}
public class lzwCode {
     
    public static void main(String [] args) {
        MyThread my = new MyThread();
        new Thread(my, "1号窗口").start();
        new Thread(my, "2号窗口").start();
        new Thread(my, "3号窗口").start();
    }
}

 运行结果为:

count= 5

count= 4

count= 3

count= 2

count= 1

 

3
8
分享到:
评论
13 楼 freezingsky 2013-04-27  
文档说明中Thread本身就实现了Runnable,所以说比较嘛,其实没有什么可比性了。只是由于java本身的单继承机制,更多让开发者倾向于接口的实现。毕竟通过接口实现,比起通过继承实现,在业务上要方便得多。
12 楼 oezx 2013-04-27  
没看懂啊。没看出什么区别啊。。。。。还不是想怎么用怎么用??
11 楼 elgs 2013-04-27  
功能上没有任何区别,Thread本身也是实现了Runnable的。
10 楼 nestnall 2013-04-27  
接口有一个作用就是用来实现多继承.这个就是区别
9 楼 点点与麦兜 2013-04-26  
这个也能比较吗?
8 楼 smilerain 2013-04-26  
一个是类对象,一个是接口,如果Thread不适合,就用Runnable接口直接实现,这就是区别
7 楼 jackjobs 2013-04-25  
kjw247 写道
这例子写的
Thread类的count是类属性,new hello()调用了3次,3个对象当然是独立的,你把count加上static试试

即时加上static在不同的机器上执行结果肯定不一样的,单核的机器试试看,线程安全要加synchronized或锁的...


谢谢指正
6 楼 wxl24life 2013-04-25  

不管怎样,楼主这篇文章的讨论点很有价值。。

"implement Runnable" or "extends Thread"

http://stackoverflow.com/questions/541487/implements-runnable-vs-extends-thread

有时间也去理一理。。
5 楼 wxl24life 2013-04-25  

两个例子都不是标准的多线程写法

把第一个例子重写了一下,权当抛砖引玉

import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;

public class Hello extends Thread {  
	
	public Hello(String name) {
		super(name);
	}
	
    public void run() {  
        for (int i = 0; i < 7; i++) {  
        	synchronized (count) {
        		if (count.get() > 0) {  
                    System.out.println(Thread.currentThread().getName() + " " +
                    		"count= " + count.getAndDecrement());  
                }  
			}
        	
        	try {
				TimeUnit.SECONDS.sleep(1);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
        }  
    }  
   
    public static void main(String[] args) {
    	  Hello h1 = new Hello("hello-1");  
          Hello h2 = new Hello("hello-2");  
          Hello h3 = new Hello("hello-3");  
          h1.start();  
          h2.start();  
          h3.start();  
	}
    private static AtomicInteger count = new AtomicInteger(5);  
}  
4 楼 中国凉茶 2013-04-25  
根本无法比较。。。。
3 楼 kevincollins 2013-04-25  
售票系统是 一个典型的 线程不安全的场景。

implements Runnable接口 和继承 Thread 都能 实现这个需求。

为了说明“多线程内的数据共享”,你这个例子不恰当。
2 楼 kjw247 2013-04-25  
这例子写的
Thread类的count是类属性,new hello()调用了3次,3个对象当然是独立的,你把count加上static试试

即时加上static在不同的机器上执行结果肯定不一样的,单核的机器试试看,线程安全要加synchronized或锁的...
1 楼 zxylcy 2013-04-25  
楼主,你后面的Thread类和Runnable接口的测试,实现过程都不一样,怎么比较呢

相关推荐

    详解Java中Thread 和Runnable区别

    详解Java中Thread和Runnable区别 Java中Thread和Runnable是两个不同的概念,它们都是Java中的线程机制,但是它们之间有着本质的区别。 Thread是Java中的一个类,而Runnable是一个接口。Thread类实现了Runnable接口...

    Java中Runnable和Thread的区别

    在Java编程语言中,多线程的实现有多种方式,其中最常见的是通过继承`Thread`类和实现`Runnable`接口。这两种方法各有其特点和适用场景,理解它们之间的区别对于编写高效、可维护的多线程应用至关重要。 ### 继承...

    多线程的Runnable_和thread有什么区别

    ### 多线程的Runnable与Thread的区别 在Java中实现多线程有两种主要方式:一种是继承`Thread`类;另一种是实现`Runnable`接口。本文将深入探讨这两种方法的区别,帮助初学者更好地理解并选择适合场景的多线程实现...

    认识Thread和Runnable

    "认识Thread和Runnable" Thread 和 Runnable 是 Java 中实现多线程的两种方式。继承 Thread 类或者实现 Runnable 接口可以实现多线程。Runnable 接口建议使用,因为它可以实现多继承,并且只有一个 run 方法,很...

    JAVA中使用Thread类和Runnable接口创建多线程

    在Java中创建线程有两种方法:使用Thread类和使用Runnable接口。在使用Runnable接口时需要建立一个Thread实例。因此,无论是通过Thread类还是Runnable接口建立线程,都必须建立Thread类或它的子类的实例。

    thread 线程类 实现runnable接口

    Thread 线程类是 Java 中 Thread 和 Runnable 接口的实现,它提供了线程的基本操作和管理。下面是对 Thread 类的详细介绍: 1. Runnable 接口 Runnable 接口是 Thread 类的基类,它提供了 run() 方法,该方法是...

    【java学习记录】11.多线程实现阶乘计算(Thread类和Runnable接口)

    这两种方法都可以实现阶乘计算的多线程化,主要区别在于继承`Thread`时,线程行为直接与类绑定,而实现`Runnable`接口时,线程行为是通过对象实例传递的。选择哪种方式取决于具体需求和设计考虑,例如是否需要复用类...

    java线程_Thread_Runnable

    Java提供了两种创建线程的方式:继承Thread类和实现Runnable接口。 1. 继承Thread类创建线程: 当你选择继承Thread类时,你需要创建一个新的类,这个类扩展自Thread类。Thread类已经实现了Runnable接口,所以你不...

    JAVA多线程Thread和Runnable的实现

    有两种主要的方式可以实现多线程:继承`Thread`类和实现`Runnable`接口。这两种方法各有特点,适用于不同的场景。 **继承Thread类** 当一个类直接继承自`Thread`类时,这个类本身就成为一个线程实体。你可以重写`...

    java 多线程Thread与runnable的区别

    Thread 和 Runnable 的区别 在 Java 中,Thread 和 Runnable 是两种不同的多线程实现方法。Thread 是一个具体的线程类,而 Runnable 是一个接口。继承 Thread 类时,我们需要重写 run() 方法,而实现 Runnable 接口...

    Java中Runnable和Thread的区别分析.rar

    在创建线程时,我们通常会遇到两种主要的方式:实现`Runnable`接口和继承`Thread`类。虽然两者都能实现多线程,但它们之间存在着重要的区别,这些差异主要体现在设计模式、内存共享、线程的创建以及线程的生命周期...

    francistao#LearningNotes#Thread和Runnable实现多线程的区别1

    在第二种方法中,由于3个Thread对象共同执行一个Runnable对象中的代码,因此可能会造成线程的不安全,比如可能ticket会输出-1(如果我们Syste

    05 看若兄弟,实如父子—Thread和Runnable详解.pdf

    在Java中,有多种方式实现多线程,其中最常见的是通过继承`Thread`类或实现`Runnable`接口。这两种方式看似并列,但实际上它们都归结为一种根本实现:创建`Thread`对象并调用`start()`方法。 首先,当我们创建一个...

    Java中Runnable和Thread的区别分析

    在Java编程语言中,创建多线程有两种主要的方式:继承`Thread`类和实现`Runnable`接口。这两种方法都有其特点和适用场景,下面我们详细分析它们的优缺点。 首先,让我们看看继承`Thread`类的方法。`Thread`类是Java...

    浅析Java中Runnable和Thread的区别

    在Java编程语言中,创建和管理多线程有两种主要的方式:继承`Thread`类和实现`Runnable`接口。这两种方式都是为了实现并发执行任务,但它们之间存在一些关键的区别。 首先,当我们创建一个新的线程时,实际上是为了...

    【Java多线程】Thread和Runnable创建新线程区别

    这是一道面试题,创建多线程时,使用继承Thread类和实现Runnable接口有哪些区别呢? 一、Thread 先来看看Thread类和其中的start()方法 class Thread implements Runnable{.....} // 可见,Thread类实现了Runnable...

    Java多线程技术 线程的死锁,详细阐述了多线程的两种实现方法: 由Thread类派生子类;实现Runnable接口

    进程与线程的区别** - **进程**:是一个正在执行的程序实例,具有独立的地址空间和资源。每个进程都包括一个或多个线程。 - **线程**:是进程内的执行单位,是最小的调度单元。同一进程内的线程共享相同的地址空间...

    创建线程的三种方式(Thread、Runnable、Callable).docx

    在Java编程中,创建线程主要有三种方式:继承Thread类、实现Runnable接口以及实现Callable接口。下面我们将逐一探讨这些方式的细节。 **方式一:继承Thread类实现多线程** 这种方式是最直观的创建线程的方法。首先...

    Java有两种实现多线程的方式:通过Runnable接口、通过Thread直接实现,请掌握这两种实现方式,并编写示例程序。

    在Java中,实现多线程的两种主要方式分别是:**通过继承`Thread`类** 和 **实现`Runnable`接口**。下面将详细介绍这两种方式,并提供相应的示例程序。 #### 一、通过继承`Thread`类实现多线程 在Java中,可以通过...

    面向小白学习法——Thread和Runnable,扩展Callable!

    面试中,理解`Thread`和`Runnable`的区别很重要。`Thread`适合需要直接继承的场景,而`Runnable`更适合多态和资源共享。`Callable`则在需要从线程获取返回值的情况下更为适用。掌握这些基础,能帮助你在实际编程中更...

Global site tag (gtag.js) - Google Analytics