zoukankan      html  css  js  c++  java
  • Java-多线程(下)

    线程的同步

        线程的同步与死锁是多线程里面最需要重点理解的概念。这种操作的核心问题在于:每一个线程对象轮番强占资源

    带来的问题。

        关于窗口卖票问题,若不采用线程同步,当三个窗口同时开始卖票,因为多线程之间的资源是共享的,当三个窗口都获取到票仅剩一张的时候,三个窗口都满足可以卖票这一条件,故对票ticket做三次-1操作,则ticket的最终结果是-1;从而需要线程同步

        窗口卖票链接

    synchronized处理同步问题

        使用synchronized关键字处理有两种模式:同步代码块、同步方法

    1.对象锁

    使用同步代码块 :
        如果要使用同步代码块必须设置一个要锁定的对象,所以一般可以锁定当前对象:this。

        synchronized(this){ ... }

    使用同步方法:

        在方法前加synchronized修饰

    public synchronized void fun()

    同步虽然可以保证数据的完整性(线程安全操作),但是其执行的速度会很慢。

    2.全局锁

        a) 使用synchronized锁这个类对应的Class对象

            synchronized(类名.class) {}

        b) static synchronized方法,static方法可以直接类名加方法名调用,方法中无法使用this,所以它锁的不是this,而是
    类的Class对象,所以,static synchronized方法也相当于全局锁,相当于锁住了代码段。

    Lock锁的使用

        1.在类中声明并实例化一个Lock类的属性    private Lock ticketLock = new ReentrantLock() ;

        2.使用ticketLock.lock(){ ... } 实现同步代码

    package www.bit.java.testdemo;
    
    import java.util.concurrent.locks.Lock;
    import java.util.concurrent.locks.ReentrantLock;
    
    class MyThread implements Runnable {
    	private int ticket = 500;
    	private Lock ticketLock = new ReentrantLock();
    	@Override
    	public void run() {
    		for (int i = 0; i < 500; i++) {
    			ticketLock.lock();
    			try {
    				if (this.ticket > 0) { // 还有票
    					try {
    						Thread.sleep(20);
    					} catch (InterruptedException e) {
    						e.printStackTrace();
    					} // 模拟网络延迟
    					System.out.println(Thread.currentThread().getName() + ",还有" + this.ticket - -+" 张票");
    				}
    			} finally {
    				ticketLock.unlock();
    			}
    		}
    	}
    }
    
    public class TestDemo {
    	public static void main(String[] args) {
    		MyThread mt = new MyThread();
    		Thread t1 = new Thread(mt, "黄牛A");
    		Thread t2 = new Thread(mt, "黄牛B");
    		Thread t3 = new Thread(mt, "黄牛C");
    		t1.setPriority(Thread.MIN_PRIORITY);
    		t2.setPriority(Thread.MAX_PRIORITY);
    		t3.setPriority(Thread.MAX_PRIORITY);
    		t1.start();
    		t2.start();
    		t3.start();
    	}
    }

    wait()方法  (wait()方法就是使线程停止运行)

    1. 方法wait()的作用是使当前执行代码的线程进行等待,wait()方法是Object类的方法,该方法是用来将当前线
        程置入“预执行队列”中,并且在wait()所在的代码处停止执行,直到接到通知或被中断为止。
    2. wait()方法只能在同步方法中或同步块中调用。如果调用wait()时,没有持有适当的锁,会抛出异常。

    3. wait()方法执行后,当前线程释放锁,线程与其它线程竞争重新获取锁。

    notify()方法  (notify方法就是使停止的线程继续运行)

    1. 方法notify()也要在同步方法或同步块中调用,该方法是用来通知那些可能等待该对象的对象锁的其它线程,
    对其发出通知notify,并使它们重新获取该对象的对象锁。如果有多个线程等待,则有线程规划器随机挑选出
    一个呈wait状态的线程。
    2. 在notify()方法后,当前线程不会马上释放该对象锁,要等到执行notify()方法的线程将程序执行完,也就是退

    出同步代码块之后才会释放对象锁。

    notifyAll()方法

        notify方法只是唤醒某一个等待线程,那么如果有多个线程都在等待中怎么办呢,这个时候就可以使用

    notifyAll方法可以一次唤醒所有的等待线程。

  • 相关阅读:
    爬弹幕
    写了这么多行就给我30,呜呜呜
    ticket
    yield求平均数
    协程原理
    爬取一类字二类字的信息和笔顺gif图片
    关于CRF的相关阅读
    embedding size与vocabulary size之间的关系: e = v**0.25
    pandas多个值取数
    转 pandas pivot
  • 原文地址:https://www.cnblogs.com/yongtaochang/p/13615346.html
Copyright © 2011-2022 走看看