目录
1.多线程的创建方式
2.线程的生命周期
3.线程安全
4.线程通信
5.常见问题
1.多线程的创建有四种方法
①继承Tread类
class ThreadTest extends Thread{
public void run() {}
}
public class java2 {
public static void main(String[] args) {
ThreadTest threadTest = new ThreadTest();
threadTest.start();
}
}
②实现Runnable接口
class ThreadTest implements Runnable{
public void run() {}
}
public class java2 {
public static void main(String[] args) {
Thread thread = new Thread(new ThreadTest());
thread.start();
}
}
③实现Callable接口
class ThreadTest implements Callable {
public Object call() throws Exception {return null;}
}
public class java2 {
public static void main(String[] args) {
FutureTask futureTask = new FutureTask(new ThreadTest());
new Thread(futureTask).start();
}
}
④使用线程池
class ThreadTest implements Runnable {
public void run() {}
}
public class java2 {
public static void main(String[] args) {
ExecutorService service = Executors.newFixedThreadPool(10);
service.execute(new ThreadTest());
service.shutdown();
}
}
2.线程的生命周期包含五个状态:新建、就绪、运行、阻塞、死亡
3
3.1当多个线程同时操作共享数据时就会出现线程安全问题,java给出了如下方法来解决线程安全问题
方法一:同步代码块
* synchronized(同步监视器){
* //需要同步的数据
* }
* 说明:
* 1.操作共享数据的代码,即为需要同步的代码
* 2.共享数据:多个线程共同的操作的变量
* 3.同步监视器,俗称:锁。任何一个类的对象,都可以充当锁。
* 要求:多个线程必须要共用一把锁
* 4.代码块只能括住操作共享数据的代码(不能括多也不能括少)
*方法二:同步方法
public synchronized void test(){}
说明:
1.同步方法仍然涉及到同步监视器,只是不需要我们显示的声明
2.非静态的同步方法,同步监视器是:this
3.静态的同步方法,同步监视器是:当前类本身
3.2死锁
1.死锁的理解:不同的线程分别占用对方需要的同步资源不放弃,都在等待对方放弃自己需要的同步资源,就形成了线程的死锁
2.说明:
1)出现死锁后,不会出现异常,不会出现提示,只是所有的线程都处于阻塞状态,无法继续
2)我们使用同步时,要避免出现死锁
4.线程通信
涉及到的三个方法:
wait():一旦执行此方法,当前线程并进入阻塞状态,并释放同步监视器
notify():一旦执行此方法,就会唤醒wait的一个线程。如果有多个线程被wait,就唤醒优先级高的
notifyAll():一旦执行此方法,就会唤醒所有被wait的线程。
说明:
1.wait(),notify(),notifyAll()三个方法必须使用在同步代码块或同步方法中
2.wait(),notify(),notifyAll()三个方法的调用者必须时同步代码块或同步方法中的同步监视器,否则会出异常
5.常见问题
sleep()和wait()的异同?
1.相同点:一旦执行方法,都可以使得当前线程进入阻塞状态
2.不同点:①两个方法声明的位置不同:Thread类中声明sleep(),Object类中声明wait()
②调用的要求不同:sleep()可以在热河需要的场景下调用,wait()必须使用在同步代码块或同步方法中
③关于是否释放同步监视器:如果两个方法都使用在同步代码块或同步方法中,sleep()不会释放同步监视器,而wait()会