每天一道Java题[10]

题目

阐述创建线程最常用的两种方法及其对比。

解答

方法一:继承Thread类实现

步骤:

  1. 创建Thread类的子类,如MyThread。
  2. 重写Thread类的run()方法。
  3. 实例化MyThread类,对象名如myThread。
  4. 运用Thread类的start()方法启动线程,如myThread.start()。

方法二:实现Runnable接口

步骤:

  1. 创建一个类,如MyRunnableThread,实现Runnable接口。
  2. 创建MyRunnableThread类的对象。
  3. 实例化Thread类,对象名如thread,并向其构造函数传入MyRunnableThread类和线程名两个参数。
  4. 运用Thread类的start()方法启动线程,如thread.start()。

继承Thread类创建线程与实现Runnable接口创建线程的不同之处在于,当用同一个类创建多个线程的时候,前者实际上是创建了多个不同的Thread对象,它内部的run()方法执行的时候是在各自对象中执行,互不干扰,如同多个线程执行多个任务;而后者实际上是使用同一个对象来创建多个线程,所以对象内的属性会公用,那就相当于多个线程在执行同一个任务一样。

另外,在使用上,继承了Thread类的类就不能继承其它类了,而实现了Runnable接口的类,还可以继承其他类,前者相对局限。

这样子说法可能有点抽象,下面那用代码讲解一下。

参考代码

MyThread类

package me.huangzijian;

public class MyThread extends Thread {

    private int num = 10;
    private String name;

    public MyThread(String name) {
        this.name = name;
    }

    @Override
    public void run() {
        int count = num;
        for (int i = 0; i < count; i++) {
            System.out.println(this.name + ":" + num);
            num--;
        }
    }
}

MyRunnable类

package me.huangzijian;

public class MyRunnableThread implements Runnable {

    private int num = 10;

    @Override
    public void run() {
        int count = num;
        for (int i = 0; i < count; i++) {
            System.out.println(Thread.currentThread().getName() + ":" + num);
            num--;
        }
    }
}

TheadCreation类

package me.huangzijian;

public class ThreadCreation {

    public static void main(String[] args) {
        // 继承Thread类实现
        MyThread myThread1 = new MyThread("MyThread1");
        MyThread myThread2 = new MyThread("MyThread2");
        MyThread myThread3 = new MyThread("MyThread3");
        myThread1.start();
        myThread2.start();
        myThread3.start();

        // 实现Runnable接口
        MyRunnableThread myRunnableThread = new MyRunnableThread();
        Thread t1 = new Thread(myRunnableThread, "MyRunnableThread1");
        Thread t2 = new Thread(myRunnableThread, "MyRunnableThread2");
        Thread t3 = new Thread(myRunnableThread, "MyRunnableThread3");
        t1.start();
        t2.start();
        t3.start();
    }
}

运行结果:

运行ThreadCreation后,我们会看到,继承Thread类实现的结果如下:

可以看到,三个线程对于字段num的操作是各操作各的,都将num从10减到1。

而实现Runnable接口的就诶过如下:

可以看到,三个线程对同一个myRunnableThread对象的num进行共同的操作。根据程序,每个线程循环10次,所以num从10一直减到负数。有朋友可能会问,为什么一开始三个线程都会获取到10,这就是线程同步的问题了,需要用到synchronized等关键字进行修饰。

时间: 2024-08-25 06:19:25

每天一道Java题[10]的相关文章

每天一道Java题[11]

题目 synchronized怎么实现线程同步?请修改<每天一道Java题[10]>中的MyRunnableThread类以解决三个线程都获取到10的问题. 解答 方法一: 采用synchronized关键字包裹需要保证线程安全的代码块,来实现线程同步.语法格式为: Synchronized(expression){ //需同步的代码 } <每天一道Java题[10]>中的MyRunnableThread类修改为: package me.huangzijian; public cl

每天一道Java题[3]

问题 为什么在重写equals()方法的同时,必须重写hashCode()方法? 解答 在<每天一道Java题[2]>中,已经对hashCode()能否判断两个对象是否相等做出了解释.equals()方法与hashCode()方法的关系如下: 如果两个对象的hashCode()返回值不一样,则equals()返回的结果必为false. 如果两个对象的hashCode()返回值一样的时候,equals()返回的结果未知. 如果两个对象的equals()返回的结果为true,则两个对象的hashC

每天一道Java题[9]

题目 native关键字的作用是什么? 解答 首先,需了解JNI(Java Native Interface),它是连接Java平台与本地C代码的一个API. 其次,用native关键字声明的方法,是告诉JVM调用的方法是一个外部定义的方法,也就是本地C代码定义的一个方法. 总结来说,native关键字的具体作用是,用它声明的方法,并不需要Java代码自己实现.而是JVM通过JNI来加载本地系统C/C++的DLL,然后调用其中的方法来实现. 发散思维 1.说一下,我们平时用到的哪个方法,是用na

每天一道Java题[6]

题目 String字符串怎么转换为Date,Date又怎么转换成String字符串 解答 String->Date 主要用到类SimpleDateFormat及其抽象父类DateFormat中的方法parse(). 如下: String dateStr = "2017-05-18 16:00:00"; SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); Date date =

每天一道Java题[1]

问题:char[]与String相比,有什么优胜的地方? 回答: 针对安全保密高的信息,char[]比String做得更好.因为String是不可变得,即使你修改原先的变量,实际上也是在内存中新建一个对象,原数据还是保留在内存中,等待回收.而char[]中的元素是可以更改的.这就意味着,如密码等保密信息用完之后,你可以马上修改它而不能痕迹.从而相对于String有更好的安全性.可从下面例子中看出,char[]变更内容后,仍是那个对象.而String已经不是原来的String了. 引用: 这个对比

每天一道Java题[5]

题目 String.StringBuilder.StringBuffer有什么异同? 解答 相同点:String.StringBuilder.StringBuffer都可以用来存储字符串. 不同点: 1.String与StringBuilder.StringBuffer的不同点主要在于,String对象创建之后,是不可改变的,平时我们对同一个String变量赋值,实际上是创建了个新的对象.而后两者是可变的,这意思就是说,他们可以在同一内存地址上更改它的值,而无需创建新的对象. 2.StringB

每天一道Java题[8]

以下题目及解答属于个人见解,欢迎大家也分享和补充一下解答的内容,互相促进,共同进步! 题目 RESTful WebService与SOAP WebService有什么异同? 解答 SOAP是一个协议,而REST实际上是一个互联网软件的架构原则,并不是一个协议.它更像是对Http协议的设计初衷作诠释. SOAP发展到现在,相对REST来说,已经有相当的成熟度. SOAP WebService对于消息体.消息头等内容有统一的规范,通用性相对更强.而RESTful WebSerivce实际上只是一个设

每天一道Java题[7]

题目 什么是REST原则,请解释RESTful架构,以及其设计思想? 解答 REST,全称为Representation State Transfer,是一种互联网软件的架构原则.凡是满足REST原则的,我们都称它为RESTful架构. 对RESTful架构的理解,有以下几点: 资源,网络上的一个实体,或者是一个具体的信息.通常使用一个URI来表示一种资源. Representation,可理解为资源的表现层,资源的具体表现形式.在http请求的头信息Accept和Content-Type字段指

每天一道Java题[2]

问题 可以直接根据hashCode()方法产生的值判断两个对象是否相等吗? 解答 不能!根据Wikipedia(https://en.wikipedia.org/wiki/Java_hashCode())上对hashCode()方法的解释,它会根据这个对象内存储的数据及对象的一些特征来做散列,并返回一个有符号的32位哈希值.从这解释我们就可以看到,hashCode()方法返回的是一个散列值,而对于一个散列来说,不同的内容也是可能会出现相同的散列值的.所以即使两个对象的hashCode()返回的值