【java多线程】多线程的创建三种方式--笔记

申明:线程的概念以及进程的相关概念,可以参考网络上其他资料,这里只讨论多线程是怎么实现。

一、多线程的简单理解

明白什么是多线程,小生通俗一点的理解为:在一个程序里,我想同时让这个程序完成多个任务。

比如:让主函数 main 在打印1~100之间的所有整数的时候,要求在主函数打印到 20 的时候,再运行另一个类里的程序,让它打印10~100之间的所有整数。

这里忽略同进程内的多线程之间的抢占时间问题,上面的举例需求是要求一个程序只要有发生同时运行俩个程序的情况就行,即不准出现无论程序跑多少次都是先把 main 函数的1~100 的所有整数打印完再打印10~100 之间的所有整数(PS:如果是这样的话,那就是 main 调用其他类的方法,属于单线程了)。

二、创建多线程的三种方式(小生写的比较繁缀罗嗦,建议阅读完之后,参考代码再阅读一次。)

第一种方式:

创建:编写一个类 MyThread1 (这里只是小生的自定义命名,方便与代码里名称一致)让它继承 Thread 类,

并把需要多线程运行的程序放到 public void run() 方法里。

启动:在主函数中,new 出 MyThread1 类的实例。

运行:调用 MyThread1 类的实例的 start() 方法即可。

第二种方式:

创建:编写一个类 MyThread2 让它实现 Runnable 接口,并且要重写 run() 方法(把需要多线程运行的程序放到 public void run() 方法里)。

启动:在主函数中,new 出 MyThread1 类的实例,

new 出Thread 类(带有 target 的构造方法),

把MyThread1 类的实例作为参数传入Thread 类的构造方法里。

运行:调用 Thread 类的实例的 start() 方法即可。

第三种方式:

创建:实现 Callable 接口(小生定义这个类为 MyCallable),并且实现 call() 方法,注意 call() 方法是有返回值的。

启动:new 出Callable 接口的实现类MyCallable,

new 出 FutureTask 类的实例 task,

把call() 方法的返回值放入FutureTask 类的构造方法里,

把 task 放入 new 出的 Thread 构造方法里。

运行:调用 Thread 类的实例的 start() 方法即可。

三种方式的利弊总结:

  第一种方法好处:创建已经继承 Thread 类的类的实例,调用 start() 方法就能运行,代码写起来很方便,

    当然缺点就是继承了Thread 父类,就没有办法继承其他的类了,扩展性不好。所以一般不建议用这种方式创建多线程。

  第二种方法好处:继承接口,扩展性好。

    弊端是相对的,第一种方式不好也是相对于第二种来说的,而第二种方式相比第三种方式来说,run() 方法没有返回值,并且不能申明抛出异常。

  第三种方式好处上面已经说明,不好的就是编码很麻烦(小生太笨看了半天才搞明白)

四、创建多线程的三种方式的代码演示

说明:小生会对每种方式的每条线程进行重命名,只是为了方便代码运行效果演示。

java 默认线程的名称规则:java 中的 main 函数本身就是一个默认的主线程,名字为:main(在下面例子可以证实 main 线程在 jvm 中的默认名称)

如果自己定义的线程没有定义名称,则系统默认按照 Thread-0,Thread-1,……的命名方式进行命名。也可以通过对线程的构造方法进行重命名。

4.1 继承 Thread 类

  偷窥一下 API 发现 Thread 类实现了 Runnable 接口,currentThread() 静态方法返回当前线程这个实例,再 getName() 一下就得到当前线程的名字了。

   

  

  MyThread1 线程:

 1 package com.test.threadDemo1;
 2
 3 public class MyThread1 extends Thread {
 4
 5     public MyThread1(String name) {
 6         this.setName(name);
 7     }
 8
 9     public void run() {
10         for(int i=20;i<=100;i++) {
11             System.out.println(getName()+"..."+i);
12         }
13     }
14 }

  main 线程:

 1 package com.test.threadDemo1;
 2 /**
 3  * 多线程的实现之一:继承 Thread 类
 4  * @author Administrator
 5  *
 6  */
 7 public class ThreadDemo1 {
 8     public static void main(String[] args) {
 9         for (int i=1;i<=100;i++) {
10             System.out.println(Thread.currentThread().getName()+"..."+i);
11
12             if(i==30) {
13                 MyThread1 myThread = new MyThread1("新线程");
14                 myThread.start();
15             }
16         }
17     }
18 }

  运行结果:(有交叉运行的情况发生,就是多线程了)

  从结果可以看出,当 main 线程跑到 30 的时候还在“抢占” CPU

  

4.2 实现 Runable 接口

  MyThread 线程:

 1 package com.test.threadDemo1;
 2
 3 public class MyThread2 implements Runnable {
 4
 5     public MyThread2(String name) {
 6         Thread.currentThread().setName(name);
 7     }
 8
 9     @Override
10     public void run() {
11         for(int i=20;i<=100;i++) {
12             System.out.println(Thread.currentThread().getName()+"..."+i);
13         }
14     }
15
16 }

  main 线程:

 1 package com.test.threadDemo1;
 2 /**
 3  * 多线程的实现之二:实现 Runable 接口
 4  * @author Administrator
 5  *
 6  */
 7 public class ThreadDemo2 {
 8     public static void main(String[] args) {
 9         for (int i=1;i<=100;i++) {
10             System.out.println(Thread.currentThread().getName()+"..."+i);
11
12             if(i==30) {
13                 MyThread2 myThread = new MyThread2("新线程");
14
15                 Thread thread = new Thread(myThread);
16
17                 thread.start();
18             }
19         }
20     }
21 }

  运行结果:

  

4.3 实现 Callable 接口和 Future 接口

  Future接口的实现类是 FutureTask,它实现了 Runable 接口,所以能够作为 target 参数传到 Thread 类的构造方法里。

  FutureTask(Callable<V> callable)

  从构造方法里可以看出 FutureTask 的构造方法里传入 callable 的实现类就行了。

  MyThread3 线程:

 1 package com.test.threadDemo1;
 2
 3 import java.util.concurrent.Callable;
 4
 5 public class MyThread3 implements Callable<Integer> {
 6
 7     @Override
 8     public Integer call() throws Exception {
 9         int i = 20;
10         for(;i<=100;i++) {
11             System.out.println(Thread.currentThread().getName()+"..."+i);
12         }
13         return i;
14     }
15
16 }

  main 线程:

 1 package com.test.threadDemo1;
 2
 3 import java.util.concurrent.Callable;
 4 import java.util.concurrent.FutureTask;
 5
 6 /**
 7  * 多线程的实现之三:实现 Callable 接口和 Future 接口
 8  * @author Administrator
 9  *
10  */
11 public class ThreadDemo3 {
12     public static void main(String[] args) {
13
14         Callable<Integer> myThread3 = new MyThread3();
15         FutureTask<Integer> task = new FutureTask<Integer>(myThread3);
16
17         for (int i=1;i<=100;i++) {
18             System.out.println(Thread.currentThread().getName()+"..."+i);
19
20             if(i==30) {
21
22                 Thread thread = new Thread(task,"新线程");
23                 thread.start();
24             }
25         }
26     }
27 }

  运行结果:

  

时间: 2024-11-06 16:11:53

【java多线程】多线程的创建三种方式--笔记的相关文章

java实现HTTP请求的三种方式

目前JAVA实现HTTP请求的方法用的最多的有两种:一种是通过HTTPClient这种第三方的开源框架去实现.HTTPClient对HTTP的封装性比较不错,通过它基本上能够满足我们大部分的需求,HttpClient3.1 是 org.apache.commons.httpclient下操作远程 url的工具包,虽然已不再更新,但实现工作中使用httpClient3.1的代码还是很多,HttpClient4.5是org.apache.http.client下操作远程 url的工具包,最新的:另一

JAVA实现Base64编码的三种方式

摘要: Javabase64编码的三种方式   有如下三种方式: 方式一:commons-codec.jar Java代码  1. String base64String="whuang123"; 2. byte[] result = Base64. Javabase64编码的三种方式 有如下三种方式: 方式一:commons-codec.jar Java代码 1. String base64String="whuang123"; 2. byte[] result

java连接access数据库的三种方式以及远程连接

连接access数据库,网上的内容很多,尝试的过程中也会遇到各种问题,没有特别好的介绍,所以自己想总结一下,日后备用. 这里需要提前说明下,win7系统32bit和64bit默认安装的access都是32bit的,但是如果是64bit的系统连接access数据库时需要安装64bit的office软件,下面会说到. 以64bit操作系统为列 第一种:电脑上配置结合java代码实现 a.打开目录C:\Windows\System32找到odbcad32.exe文件,双击打开 b.点击右侧添加按钮,在

java中遍历集合的三种方式

集合遍历操作的三种方式 Iterator迭代器方式 增强for循环 普通for循环 代码如下: public static void test3(){ ArrayList list = new ArrayList(); list.add(123); list.add("AAAA"); list.add("bb"); list.add(new String("JavaEE")); list.add(new Date()); list.add(fal

java加载jdbc驱动三种方式的比较

平时连接数据库的时候首先要加载jdbc驱动,这一步骤其实有三种方式,他们的区别?优劣? 快速了解三种加载方式 Class.forName("com.mysql.jdbc.Driver"); DriverManager.registerDriver(new com.mysql.jdbc.Driver()) System.setProperty("jdbc.drivers","com.mysql.jdbc.Driver"); 注释: 第二种与第三种注

java中终止线程的三种方式

在java中有三种方式可以终止线程.分别为: 1.  使用退出标志,使线程正常退出,也就是当run方法完成后线程终止.  2.  使用stop方法强行终止线程(这个方法不推荐使用,因为stop和suspend.resume一样,也可能发生不可预料的结果). 3.  使用interrupt方法中断线程. 下面我们来详细的介绍这三种方法. 1. 使用退出标志终止线程 当run方法执行完后,线程就会退出.但有时run方法是永远不会结束的.如在服务端程序中使用线程进行监听客户端请求,或是其他的需要循环处

Java 数组元素倒序的三种方式

将数组元素反转有多种实现方式,这里介绍常见的三种. 直接数组元素对换 @Test public void testReverseSelf() throws Exception { System.out.println("use ReverseSelf"); String[] strings = { "ramer", "jelly", "bean", "cake" }; System.out.println(

线程的状态以及创建多线程的三种方式

首先了解一下线程的五种状态: 新建状态: 新建状态是指new之后,即新创建了一个线程的时候,此时并未运行任何线程方法体内的程序代码. 就绪状态: 简单来说就是指程序调用了start()之后,线程就得到了启动,代表线程进入了就绪状态,但是此时并不代表它会立刻去执行run()方法体内的程序代码,而是随时等待cpu的调度. 运行状态: 获得cpu的时间后,调用run()方法,进入运行状态. 阻塞状态: 由于某种原因放弃了cpu的会用权力,暂时停止运行,等待再次被调用. 死亡状态: 线程正常执行完毕,或

JAVA多线程实现的三种方式

JAVA多线程实现方式主要有三种:继承Thread类.实现Runnable接口.使用ExecutorService.Callable.Future实现有返回结果的多线程. 当中前两种方式线程运行完后都没有返回值,仅仅有最后一种是带返回值的. 1.继承Thread类实现多线程 继承Thread类的方法虽然被我列为一种多线程实现方式,但Thread本质上也是实现了Runnable接口的一个实例,它代表一个线程的实例,而且,启动线程的唯一方法就是通过Thread类的start()实例方法.start(