Java基础:synchronized实现线程同步

线程安全问题

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;
}}}}

Jdk5.0新增方式:Lock锁

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

King'sLanding

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值