在开发当中使用多线程的,经常会用到synchronized和volatitle。接下来就讲讲他们的使用场景。
synchronized
java关键字,方法用到这个关键字则对这个方法进行加锁。一次只能进入一个线程,其他线程只能等待或执行其他没有加锁的方法。等到这个线程完成后,下个线程才能进去。这样保证数据的原子性,同时性能也就下降。
volatile
java关键字,用来声明变量。
1.可见性
保证所有的线程的可见性,当一个线程修改了这个变量的值,volatile 保证了新值能立即同步到主内存,这样其他线程获取的是最新的值,保证了这个变量共享。
2.禁止指令重排序
指定重排序会涉及到编译器和处理器,单线程再跑程序的时候处理器会按照编译器的顺序执行,当多个线程在跑的时候,处理器就不按照编译器的顺序来,自己会重排序。(什么是指令重排序:是指CPU采用了允许将多条指令不按程序规定的顺序分开发送给各相应电路单元处理)
3.原子性
volatile不保证原子性,会存在线程安全问题。
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();
}
}