线程同步

2.1在实际生活中,售票时出票也是需要时间的,所以在出售一张票的时候,需要一点时间的延迟。

 

卖票会出现的问题

  相同的票出现多次

  出现了负数的票

 

问题原因

  线程执行的随机性导致的

 

2.2卖票案例数据安全问题的解决

为什么出现问题?(这也是我们判断多线程程序是否会有数据安全问题的标准)

  是否是多线程环境

  是否有共享数据

  是否有多条语句操作共享数据

 

如何解决多线程安全问题呢?

  基本思想:让程序没有安全问题的环境

 

怎么实现呢?

  把多条语句操作共享数据的代码给锁起来,让任意时刻只能有一个线程执行即可

 

2.3同步代码块

锁多条语句操作共享数据,可以使用同步代码块实现

  格式:

  synchronized(任意对象){

    多条语句操作共享数据的代码

  }

  synchronized(任意对象):就相当于给代码枷锁了,任意对象就可以看成是一把锁

同步的好处和弊端

  好处:解决了多线程的数据安全问题

  弊端:当线程很多时,因为每个线程都会去判断同步上的锁,这是很耗费资源的,无形中会降低程序的运行效率

 

2.4同步方法

同步方法:就是synchronized关键字加到方法上

  格式:

  修饰符 synchronized 返回值类型 方法名(方法参数){  }

 

同步方法的锁对象是是什么呢?

  this

同步静态方法:就是把synchronized关键字加到静态方法上

  格式:

  修饰符 static synchronized 返回值类型 方法名(方法参数){  }

同步静态方法的锁对象是什么呢?

  类名.class

 

2.5线程安全的类

StringBuffer

  线程安全,可变的字符序列

  从版本JDK5开始,被StringBuilder替代。通常应该使用StringBuilder类,因为它支持所以相同的操作,但它更快,因为它不执行同步。

 

Vector

  从Java2平台v1.2开始,该类改进了List接口,使其成为Java Collections Framework的成员。与新的集合实现不同,Vector被同步。如果不需要线程安全的实现,建议使用ArrayList代替Vector

 

Hashtable

  该类实现了一个哈希表,它将键映射到值。任何非null对象都可以用作键或者值

  从Java2平台v1.2开始,该类进行了改进,实现了Map接口,使其成为Java Collections Framework的成员。与新的集合实现不同,Hashtable被同步。如果不需要线程安全的实现,建议使用HashMap代替Hashtable。

 

2.6Lock锁

虽然我们可以理解同步代码块和同步方法的锁对象问题,但是我们并没有直接看到哪里加上了锁,在哪里释放了锁,为了更清晰的表达如何加锁和释放锁,JDK5以后提供了一个新的锁对象Lock

 

Lock实现提供比使用synchronized方法和语句可以获得更广泛的锁定操作

Lock中提供了获得锁和释放锁的方法

  void lock():获得锁

  void unlock():释放锁

 

Lock是接口不能直接实例化,这里采用它的实现类ReentrantLock来实例化

ReentrantLock的构造方法

  ReentrantLock():创建一个ReentrantLock的实例

 

Object类的等待和唤醒方法:

方法名 说明
void wait() 导致当前线程等待,直到另一个线程调用该对象的notify()方法或notifyAll()方法
void notify() 唤醒正在等待对象监视器的单个线程
void notifyAll() 唤醒正在等待对象监视器的所有线程
posted @ 2022-03-07 20:39  隳世  阅读(41)  评论(0)    收藏  举报