线程安全问题
package thread;
/**
* 三个线程共同递减输出100-0的数字:多线程的安全问题
* 1.问题:买票过程中(遍历100-1)的过程中出现了重票、错票----即线程的安全问题
* 2.问题的原因:当某个线程操作车票的过程中,尚未操作完成时,其他线程参与了进来,
* 前一个线程还没完全执行,后一个线程也拿到了资源
* 3.解决方案:规定(上锁)线程a在操作共享数据ticket时,其他线程被锁住无法参与进来
* 直到a线程操作完,拿走了ticket的数据时,其他线程才开始操作ticket。
* 且即使a出现了阻塞,也不能改变。
*
*
* Created by KingsLanding on 2022/3/11 11:25
*/
public class ThreadTest04 {
public static void main(String[] args) {
MyThread03 m1 = new MyThread03();
MyThread03 m2 = new MyThread03();
MyThread03 m3 = new MyThread03();
m1.setName("分线程-1**");
m1.start();
m2.setName("分线程-2**");
m2.start();
m3.setName("分线程-3**");
m3.start();
}
}
class MyThread03 extends Thread{
//共享ticket;但是会暴露线程安全问题-->同一个数字多次出现(理论上只能出现一次)
private static int ticket=100;
public void run(){
while (true){
if(ticket>0){
System.out.println(getName()+"票号:"+ticket);
ticket--;
}else {
break;
}
}
}
}
线程同步(安全机制)
方法一
synchronized(同步监视器){
//需要同步(上锁)的代码块
}
继承Thread类的方式的多线程
- 慎用this充当同步监视器
package thread;
/**
* 继承方式创建的多线程
* 同步机制synchronized (同步监视器){
* }
*
* 说明:这种方式中慎用this充当同步监视器,考虑用当前类来充当同步监视器
*
* Created by KingsLanding on 2022/3/14 19:31
*/
public class ThreadTest07 {
public static void main(String[] args) {
MyThread07 m1 = new MyThread07();
MyThread07 m2 = new MyThread07();
m1.start();
m2.start();
}
}
class MyThread07 extends Thread {
//private int ticket = 100;
private static int ticket = 100;
// Lock01 L1=new Lock01();//这种方式依然不行,因为多个对象反复调用这个结构,并不是唯一的锁
//相当于其他线程有其他钥匙
private static Lock01 L1=new Lock01();
@Override
public void run() {
while (true) {
//类也是对象
//synchronized(ThreadTest07.class){//Class cla = ThreadTest07.class只会加载一次
// }
synchronized (L1) {
//synchronized (this) {//继承方式中不能用this,因为创建了多个MyThread07对象
if (ticket > 0) {
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
}
System.out.println(Thread.currentThread().getName() + "票号:" + ticket);
ticket--;
} else {
break;
}}}}}
class Lock01{//同步监视器类
}
实现Runnable接口的方法创建的多线程
package thread;
/**
* 说明:
* 1.操作共享数据的代码块就是需要被同步的地方
* 2.同步监视器:俗称“锁”任何一个类的对象,都可以充当锁:多个线程只能用一个锁
*
* 5.同步的方式能解决线程的安全问题;但是操作同步代码的只能有一个线程参与,其他线程只能
* 在“门”外等待。相当于是一个单线程过程,效率低;
*
* Created by KingsLanding on 2022/3/14 10:37
*/
public class ThreadTest06 {
public static void main(String[] args) {
MyThread06 m1 = new MyThread06();
Thread t1 = new Thread(m1);
Thread t2 = new Thread(m1);
t1.start();
t2.start();
}
}
class MyThread06 implements Runnable{
private int ticket=100;
Lock L1=new Lock();//同步监视器类对象
@Override
public void run() {
while (true){
synchronized (L1) {//上锁
if (ticket > 0) {
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
}
System.out.println(Thread.currentThread().getName() + "票号:" + ticket);
ticket--;
} else {
break;
}}}}}
class Lock{//同步监视器的类
}
方法二
同步方法
- 1.同步方法仍然涉及到同步监视器,只是不需要显示定义
- 2.非静态的同步方法,同步监视器就是this
- 3.静态的同步方法中,同步监视器是:当前类本身(类本身也是对象)
同步方法解决实现Runnable接口的线程方法
package thread;
/**
* 方式二:同步方法解决实现Runnable接口的线程方法
* 如果操作共享数据的代码完整的声明在一个方法中,不妨将该方法声明为同步的
*
* Created by KingsLanding on 2022/3/16 15:48
*/
public class ThreadTest08 {
public static void main(String[] args) {
MyThread08 m1 = new MyThread08();
Thread t1 = new Thread(m1);
Thread t2 = new Thread(m1);
t1.start();
t2.start();
}
}
class MyThread08 implements Runnable{
private int ticket=100;
@Override
public void run() {
Run();
}
private synchronized void Run(){
while (true) {
if (ticket > 0) {
System.out.println(Thread.currentThread().getName() + "票号:" + ticket);
ticket--;
}}}}
同步方法处理继承Thread类的方式创建的线程的安全问题
package thread;
/**
* 方式二:同步方法处理继承Thread类的方式创建的线程的安全问题
*
* Created by KingsLanding on 2022/3/16 16:06
*/
public class ThreadTest09 {
public static void main(String[] args) {
MyThread09 m1 = new MyThread09();
MyThread09 m2 = new MyThread09();
m1.start();
m2.start();
}
}
class MyThread09 extends Thread {
private static int ticket = 100;
public void run() {
Run();
}
private static synchronized void Run(){//同步监视器:MyThread09.class
//private synchronized void Run(){//这种方式同步监视器m1,m2,而锁只能有一把
while (true){
if (ticket > 0){
System.out.println(Thread.currentThread().getName() + "票号:" + ticket);
ticket--;
}else {
break;
}}}}