前面已经介绍过,在操作系统中,进程是资源分配的单位,线程是调度运行的单位。一个进程中可以存在多个线程,这多个线程共享进程的资源。线程具有并发性,也就是说,一个进程内的多个线程在完成任务的时候,是有可能同时使用到进程内的同一个资源,这个时候就会产生竞争,这也是多线程程序中的一个普遍问题,在java中,这种现象叫做竞争状态。
- synchronized关键字
为避免竞争状态,应该防止多个线程同时进入程序的某一特定部分,程序中的这部分称为临界区,而这个临界区多数情况下通常为一个方法,或者是一个方法中的一段代码块。
一个同步方法在执行之前需要加锁。对于实例方法,要给调用该方法的对象加锁;对于静态方法,要给这个类加锁。如果一个线程调用一个对象上的同步实例方法(静态方法),首先给该对象(类)加锁,然后执行该方法,最后解锁。在解锁之前,另一个调用那个对象(类)中方法的线程将被阻塞,直到解锁。
比如:一个账户中有0块钱,100个线程向该账户中存钱,每个线程执行一次,每次向账户中存1块钱。那么在100个线程执行完之后,账户中的余额应该是100。但是由于多线程同步执行,会发生并发操作,导致最后账户余额小于0。假如向账户中存钱的方法为:
public void deposit(int amount){
int newBalance = balance + amount;
}
为了避免出现上述情况,可对该方法进行同步:
public synchronized void deposit(int amount){
int newBalance = balance + amount;
}
- 同步语句