在总结JDBC数据库连接池的时候,发现Java多线程这块掌握得不是很好,因此回头看了下多线程的内容。做一下多线程模块的学习和总结,稳固一下多线程这块的基础。关于多线程的一些理论知识,这里不想啰嗦太多,可以进行下搜索了解。
1. 如何使用Java创建多线程
使用Java多线程,总的来说方法有两种:①继承Thread类,重写run方法 ②把相关的类实现Runnable(可运行)接口,重写run方法。③实现Callable接口(相对用得较少)
1 package com.scl.thread; 2 3 public class TestThread 4 { 5 public static void main(String[] args) throws Exception 6 { 7 String mainName = Thread.currentThread().getName(); 8 System.out.println(mainName + "线程开始运行"); 9 10 // 方法1 使用继承对线程进行调用 11 MyThread myThread = new MyThread(); 12 myThread.setName("wft"); 13 myThread.start(); 14 myThread.join(); 15 16 // 方法2 继承Runnable接口进行调用 17 Thread thread = new Thread(new MyRunnable()); 18 thread.start(); 19 20 System.out.println(mainName + "线程结束"); 21 } 23 } 24 25 // 使用runnable开启线程 26 class MyRunnable implements Runnable 27 { 28 @Override 29 public void run() 30 { 31 // 获取调用线程的名字 32 String curName = Thread.currentThread().getName(); 33 System.out.println(curName + "线程启动"); 34 try 35 { 36 Thread.sleep(1000); 37 System.out.println(curName + "正在运行"); 38 } 39 catch (InterruptedException e) 40 { 41 e.printStackTrace(); 42 } 43 System.out.println(curName + "线程结束"); 44 } 45 } 46 47 // 使用Thread进行run方法的编写,开启线程 48 class MyThread extends Thread 49 { 50 // Thread.currentThread()调用这个方法的线程名称。 51 String s = Thread.currentThread().getName(); // 调用这个类的是main方法调用,run方法交由新线程操作 52 53 public MyThread() 54 { 55 System.out.println("Thread.currentThread().getname()=" + Thread.currentThread().getName()); 56 // 线程没运行前初始化的名字.因为类的初始化跟线程运行无关,这里返回的是处世话的名字 57 System.out.println("This.getName=" + this.getName()); 58 } 59 60 @Override 61 public void run() 62 { 63 // run方法交由新线程操作,所以调用的名称不是main 64 String strName = Thread.currentThread().getName(); 65 System.out.println("this.getName " + this.getName()); 66 // main 67 System.out.println("do u no s?" + s); 68 69 System.out.println(strName + " 开始数数"); 70 for (int i = 0; i < 100; i++) 71 { 72 System.out.println(strName + " " + i); 73 } 74 System.out.println(strName + " 数数完毕"); 75 } 76 77 // 返回设置的name 78 public String bName() 79 { 80 return this.getName(); 81 } 82 }
最应该注意的是给线程标注名称的时候,应该多使用Thread.currentThread().getname() 来获取线程名称。使用this.getName方法很容易出错。
主要该注意的有如下两个:
① 获取名称不在run方法里面,永远返回名称为main。因为在构造线程类的时候,主线程main会进行代码的初始化。run方法由新创建起来的线程去调用,这样run方法里面的名称才能被正确获取到
② 在继承类的构造函数内,如上面MyThread的构造函数内调用getName()方法。该方法是由JDK 源码Thread类继承而来。new一个Thread实例时,系统会调用Thread
下的方法,代码如下
//Thread 类构造函数 public Thread() { init(null, null, "Thread-" + nextThreadNum(), 0); } // 私有的init方法 private void init(ThreadGroup g, Runnable target, String name,long stackSize) { init(g, target, name, stackSize, null); } //init 方法. 更详细内容见JDK源码Thread类363行 private void init(ThreadGroup g, Runnable target, String name,long stackSize, AccessControlContext acc)
所以在构造器内的初始化的时候调用init方法给线程设置name字段。导致参数名称修改且以"Thread-"开头。初始化完毕后,如果用户设置过相关的线程名称,系统会把该字段进行修改,可以运用上述代码中的bName()函数进行验证。
在使用Java多线程的时候,不要直接调用某个实例下的run方法,如果直接使用run方法。则直接调用,不会开辟新的线程进行功能操作。
继承Thread和实现Runnable接口的区别:
① 继承Thread能够简单地实现线程开辟,但由于Java只支持单继承。继承Thread方法实现线程会有比较多的局限性
② 实现Runnable接口能够很好的将线程代码和业务逻辑代码分离,但代码结构相对复杂。一般都会使用实现Runnable方法创建线程。
2. Java多线程下常用方法
① join( ) 多线程是有cpu分配执行时间片段,如果要控制线程操作流程,则需要使用到join方法。join方法会阻塞主线程,主线程会等待join方法的线程执行完毕再继续执行后续的操作。同时join方法可以添加时间,让主线程等待一段时间,如果时间过了。则不再等待继续执行。如:thread.join( ),当主线程编译到该代码时,主线程会停止下来等待thread线程执行完毕后,主线程再继续后续操作
② sleep( ) 不释放锁资源,让调用的线程进行等待。如:thread.sleep( ) ,但代码执行到这一句时,thread会停止下来等待,直到被唤醒或者等到到sleep到的时间点后再进行后续操作。
③ wait( )和notify( )、notifyAll( ) 。这三个方法主要处理线程同步数据时使用。数据同步,也就是几条线程在同时修改一份共享数据。如火车站5名销售员同时在售卖火车票,火车票是固定数量的前提下,如何实现这5名销售员同时知道火车票库存、同一路段火车票售卖、车票信息共享等。
wait方法使当前线程暂停执行并释放对象锁标志。notify和notifyall方法会唤醒几个等待的线程,让线程重新获取锁代码以便后续进行。线程状态图表如下:
关于使用锁以及线程状态内容,将在下一篇博文进行总结,上述内容若有问题,烦请指出纠正。