why there are two way of using thread in java?
我知道在Java中使用线程有两种方法:
我也知道实现runable比扩展线程要好。
但为什么有两种方法——为什么不止一种?
如果实现可运行性是一种更好的方法,为什么还有其他选择?
只有一个选择会有什么问题?
extends Thread 号:您的线程创建唯一的对象并与之关联
implements Runnable 号:它与多个线程共享同一对象
还有一点要注意,因为在Java中只能扩展一个类,如果EDCOX1为4,则不能扩展另一个类。如果选择
从技术上讲,只有一种方法:实现
我们使用每种线程的另一个原因。
扩展
因此,根据您的设计要求,您可以使用任何一种方法。
即使您实现了可运行的接口,您也需要创建线程来让您的任务作为线程运行。实现可运行性的明显优势是
扩展线程类只是一个选项,因为它在内部实现了可运行的,所以您最终间接实现了可运行的接口。只是这个类并不是为了防止开发人员扩展它而最终确定的。正如Joshua Bloch在"有效Java"中提到的,通常不需要扩展线程的理由。
两种方法都有不同的方法。实现可运行接口并不能给线程本身任何控制权。如果我们扩展线程类,那么派生类就不能扩展任何其他的基类。
因此,如果用户希望完全控制程序,那么线程类的扩展是更好的选择;如果用户希望扩展其他基类的灵活性,那么实现可运行接口是很好的选择。
将线程用作任务提供了创建和运行并行线程的简洁方法
1 2 3 4 5 |
子类化
此外,子类化线程类有时会产生更可读的代码,在子类中,您可以拥有自己的自定义API。可以想象类,如downloadingthread、renderingthread等扩展线程。
当我们通过实现runnable创建线程时,在线程创建期间不会初始化任何值。但是,当我们扩展线程类以创建线程时,如果需要,我们可以使用构造函数的优点初始化一些值,如下所示
1 2 3 4 5 6 7 8 9 10 11 | public class MyThread extends Thread { int aValue; public Mythread(int aValue) { this.aValue = aValue; } ............................ ............................ } |
号
当我们创建线程时,我们可以如下初始化
1 2 | MyThread t = new MyThread(7); t.start(); |
因为Java不支持多重继承,所以如果我们扩展了线程类,那么我们就失去了扩展另一个类的机会。因此,在这个场景中,可运行接口对创建线程非常有帮助。
通过扩展线程,每个线程都有一个与之相关联的唯一对象,而实现可运行,许多线程可以共享同一个可运行实例。
大多数时候,我们使用可运行的接口。因为这使得我们在结构和功能上更加灵活。Java允许对接口进行多重继承。这个概念在软件设计中被大量使用。
顺便说一下,还有另一个称为Callable的接口。可调用类可以返回类型v中的值,也可以抛出检查过的异常。