在开发当中使用多线程的,经常会用到synchronized和volatitle。接下来就讲讲他们的使用场景。
synchronized
java关键字,方法用到这个关键字则对这个方法进行加锁。一次只能进入一个线程,其他线程只能等待或执行其他没有加锁的方法。等到这个线程完成后,下个线程才能进去。这样保证数据的原子性,同时性能也就下降。
volatile
java关键字,在声明变量的时候加入该关键字,这样保证了数据的可见性。比如说一个线程对这个值进行了修改,其他线程获取的值就是更改之后的,这样就实现数据的共享。但不保证数据的原子性。当两个线程对这个变量同时做更改的时候,就会出现并发问题。
synchronized例子:
package com.company;
public class Main extends Thread {
//剩余票数
private int ticket = 1000;
//购买票数
private int buy = 0;
@Override
public void run() {
long startTime = System.currentTimeMillis();
while (ticket > 0) {
synchronized (this) {
if (ticket > 0) {
ticket--;
buy++;
System.out.println("线程名" + Thread.currentThread().getName() + "余票:" + ticket + "," + "出售:" + buy);
}
}
}
System.out.println("消耗:" + (System.currentTimeMillis() - startTime) + "ms");
}
public static void main(String[] args) {
System.out.println("主线程:" + Thread.currentThread().getName());
Main main = new Main();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
}
}
volatile例子:
package com.company;
public class Main extends Thread {
//剩余票数
private volatile int ticket = 1000;
//购买票数
private int buy = 0;
@Override
public void run() {
long startTime = System.currentTimeMillis();
while (ticket > 0) {
// synchronized (this) {
if (ticket > 0) {
ticket--;
buy++;
System.out.println("线程名" + Thread.currentThread().getName() + "余票:" + ticket + "," + "出售:" + buy);
}
// }
}
System.out.println("消耗:" + (System.currentTimeMillis() - startTime) + "ms");
}
public static void main(String[] args) {
System.out.println("主线程:" + Thread.currentThread().getName());
Main main = new Main();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
new Thread(main).start();
}
}