• 线程的同步机制


    1 线程安全问题的原因:由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程有参与进来,导致共享数据存在安全问题

    2 解决方法:必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作

    3 java如何实现线程的安全,现成的同步机制

    synchronized(同步监视器){  //需要被同步的代码块(操作共同数据的代码)}

    同步监视器:由任何一个类的对象充当,哪个线程获取此监视器,就执行大括号里被同步的代码

    1)同步代码块

    class Window2 implements Runnable{
        private int ticket = 100;  //不用static,只创建一个对象
        public void run(){
            while(true){
                synchronized(this){ //注意一个线程进入同步代码块中,在执行的其中的代码过程中其他线程不能进入,所以称为同步锁。synchronized一定要放在共享数据前后
                if(ticket>0){  //ticket是共享数据,所以同步从这个地方开始,如果放在while前,那么线程1一直占用cpu直到结束,线程2,3就进不来了
                    try {
                        Thread.currentThread().sleep(20);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
                    System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
                }
                else{
                    break;
                }}
    class Window extends Thread{
        static int ticket = 100;
        static Object obj = new Object();
        public void run(){
            while(true){
                synchronized (obj) {
                    if (ticket > 0) {
                        try {
                            Thread.currentThread().sleep(20);
                        } catch (InterruptedException e) {
                            // TODO Auto-generated catch block
                            e.printStackTrace();
                        }
                        System.out.println(Thread.currentThread().getName() + ":"
                                + ticket--);
                    } else {
                        break;
                    }
                }

    2)同步方法: 将操作共享数据的方法声明为synchronized,此方法称为同步方法,能够保证当其中一个线程执行此方法时,其他线程在外等待直到此线程执行完此方法。只能在实现方式中能使用,继承方式3个对象,有3个this,不能用同步方法

    同步方法的锁:this,不用显式指明。

    package lianxi1;
    //方式二:实现方式创建多线程
    class Window3 implements Runnable{
        private int ticket = 100;  //不用static,只创建一个对象
        public void run(){
            while(true){
               show();
            }
            }
        public synchronized void show(){ //不是显式调用this,因为只有一个对象,所以只有this就是同步监视器
            if(ticket>0){  //ticket是共享数据,所以从这个地方开始
                try {
                    Thread.currentThread().sleep(20);
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
                System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
            }
        }
    }
    public class TestThreadWindow3 {
    
        public static void main(String[] args) {
            Window3 w2 = new Window3();
            Thread t1 = new Thread(w2); //将Runnable实现类对象作为形参传人Thread构造器,执行start方法
            Thread t2 = new Thread(w2);
            Thread t3 = new Thread(w2);
            t1.setName("窗口1");
            t2.setName("窗口2");
            t3.setName("窗口3");
            t1.start();
            t2.start();
            t3.start();
    
        }
    
    }
  • 相关阅读:
    c#9
    Jmeter--自己写并发压测脚本一定会遇到的类
    Jmeter从数据库获取测试数据, 作为下一个接口参数方法
    Jmeter-从数据库中获取数据并作为变量传输
    Java文件操作
    Java读取文件
    Java字符串处理——String类常用方法
    Eclipse导出可执行jar包步骤
    批处理常用DOS命令之for简单 002
    批处理常用DOS命令之for简单 001
  • 原文地址:https://www.cnblogs.com/yjtm53/p/4160620.html
Copyright © 2020-2023  润新知