同步问题:
案例一:
//线程同步问题一 :线程不安全,买票问题
public class UnsafeBuyTicket implements Runnable {
//票数
private int ticketNums = 10;
//标志位
private boolean flag = true;
@Override
public void run() {
//买票
while (flag) {
buyTicket();
}
}
public void buyTicket() {
if (ticketNums <= 0) {
flag = false;
return;
}
//模拟网络延时
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "-->拿到了第" + ticketNums-- + "张票");
}
public static void main(String[] args) {
UnsafeBuyTicket station = new UnsafeBuyTicket();
new Thread(station,"苦逼的我").start();
new Thread(station,"牛逼的你们").start();
new Thread(station,"可恶的黄牛党").start();
}
}
输出结果:
牛逼的你们–>拿到了第10张票
可恶的黄牛党–>拿到了第9张票
苦逼的我–>拿到了第8张票
苦逼的我–>拿到了第7张票
可恶的黄牛党–>拿到了第6张票
牛逼的你们–>拿到了第5张票
可恶的黄牛党–>拿到了第4张票
牛逼的你们–>拿到了第3张票
苦逼的我–>拿到了第2张票
牛逼的你们–>拿到了第1张票
可恶的黄牛党–>拿到了第0张票
苦逼的我–>拿到了第-1张票
注意我们看到了有0张票和-1张票的,说明了不安全。
案例二:
//银行(存钱:存了多少,取钱:去了多少) , 两个人 , 账户
//并发问题,线程不安全
public class UnsafeBank {
public static void main(String[] args) {
Account account = new Account(100,"招商卡");
Bank you = new Bank("痛苦的你",account,50);
Bank wife = new Bank("开心的媳妇",account,100);
you.start();
wife.start();
}
}
//账户
class Account{
int money;//余额
String name; //卡名
public Account(int money, String name) {
this.money = money;
this.name = name;
}
}
//银行
class Bank extends Thread{
//存钱:存了多少,取钱:取了多少
Account account; //账户
int drawingMoney; //取了多少钱
int nowMoney; //手里有多少钱
public Bank(String name,Account account,int drawingMoney){
super(name);
this.account = account;
this.drawingMoney = drawingMoney;
}
@Override
public void run() {
//判断能否取钱
if (account.money-drawingMoney<0){
return;
}
//为了放大问题发生性,我们加个延时.
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
//余额 = 余额 - 你去走的钱
account.money = account.money - drawingMoney;
//你的钱 = 你的钱 + 你取的钱
nowMoney = drawingMoney + nowMoney;
System.out.println(this.account.name+"账户余额:"+account.money);
System.out.println(this.getName()+"手里的钱:"+nowMoney);
}
}
输出结果:
招商卡账户余额:-50
招商卡账户余额:-50
痛苦的你手里的钱:50
开心的媳妇手里的钱:100
注意*:输出结果中银行余额出现负值,不安全;
案例三:
//线程不安全问题3,集合操作
import java.util.ArrayList;
import java.util.List;
//思考?怎么让这些问题变安全.
public class UnSafeList {
public static void main(String[] args) throws InterruptedException {
List list = new ArrayList();
for (int i = 0; i < 200000; i++) {
new Thread(()->{
list.add(Thread.currentThread().getName());
}).start();
}
for (int i = 5;i>0;i--){
Thread.sleep(1000);
System.out.println("倒计时"+i);
}
System.out.println(list.size());
}
}
输出结果:
倒计时5
倒计时4
倒计时3
倒计时2
倒计时1
199998
注意输出结果中没有到200000,不安全
同步方法 : public synchronized void method(int args) {}
synchronized方法控制对 “对象” 的访问 , 每个对象对应一把锁 , 每个synchronized方法都必须获得调用该方法的对象的锁才能执行 , 否则线程会阻塞 ;
方法一旦执行 , 就独占该锁 , 直到该方法返回才释放锁 , 后面被阻塞的线程才能获得这个锁 , 继续执行;
缺陷 : 若将一个大的方法申明为synchronized 将会影响效率
同步块:
同步块 : synchronized (Obj ) { }
Obj 称之为 同步监视器
Obj 可以是任何对象 , 但是推荐使用共享资源作为同步监视器
同步方法中无需指定同步监视器 , 因为同步方法的同步监视器就是this , 就是这个对象本身 , 或者是 class [ 反射中讲解 ]
案例一:
public class SafeBuyTicket implements Runnable {
//票数
private int ticketNums = 10;
//标志位
private boolean flag = true;
@Override
public void run() {
//买票
while (flag) {
buyTicket();
}
}
//同步方法,关键词synchroinzed.
//关键字是锁
//实现的机制是队列
//还能所反射的那个class
public synchronized void buyTicket() {
if (ticketNums <= 0) {
flag = false;
return;
}
//模拟网络延时
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "-->拿到了第" + ticketNums-- + "张票");
}
public static void main(String[] args) {
SafeBuyTicket station = new SafeBuyTicket();
new Thread(station,"苦逼的我").start();
new Thread(station,"牛逼的你们").start();
new Thread(station,"可恶的黄牛党").start();
}
}
输出结果:
苦逼的我–>拿到了第10张票
苦逼的我–>拿到了第9张票
苦逼的我–>拿到了第8张票
苦逼的我–>拿到了第7张票
可恶的黄牛党–>拿到了第6张票
可恶的黄牛党–>拿到了第5张票
可恶的黄牛党–>拿到了第4张票
牛逼的你们–>拿到了第3张票
牛逼的你们–>拿到了第2张票
牛逼的你们–>拿到了第1张票
案例二:
public class SafeBank {
public static void main(String[] args) {
Account2 account = new Account2(100,"招商卡");
Bank2 you = new Bank2("痛苦的你",account,50);
Bank2 wife = new Bank2("开心的媳妇",account,100);
you.start();
wife.start();
}
}
//账户
//实体类
class Account2{
int money;//余额
String name; //卡名
public Account2(int money, String name) {
this.money = money;
this.name = name;
}
}
//银行
class Bank2 extends Thread{
//存钱:存了多少,取钱:取了多少
Account2 account; //账户
int drawingMoney; //取了多少钱
int nowMoney; //手里有多少钱
public Bank2(String name,Account2 account,int drawingMoney){
super(name);
this.account = account;
this.drawingMoney = drawingMoney;
}
@Override
public void run() {
drwaing();
}
//synchronized本身锁的是this.就是这个对象本身
public void drwaing(){
//提高性能的代码
if (account.money<=0){
return;
}
//如何判断锁的对象
// 谁需要实现增删改就去锁定他
synchronized (account){
//判断能否取钱
if (account.money-drawingMoney<0){
System.out.println(Thread.currentThread().getName()+"活该,没取到钱");
return;
}
//为了放大问题发生性,我们加个延时.
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
//余额 = 余额 - 你去走的钱
account.money = account.money - drawingMoney;
//你的钱 = 你的钱 + 你取的钱
nowMoney = drawingMoney + nowMoney;
System.out.println(this.account.name+"账户余额:"+account.money);
System.out.println(this.getName()+"手里的钱:"+nowMoney);
}
}
}
输出结果:
招商卡账户余额:50
痛苦的你手里的钱:50
开心的媳妇活该,没取到钱
注意银行卡没有出现负数,安全;
案例三:
import java.util.ArrayList;
import java.util.List;
//JUC并发编程 , 保证线程安全的一些类.
public class SafeList {
public static void main(String[] args) throws InterruptedException {
List list = new ArrayList();
for (int i = 0; i < 10000; i++) {
new Thread(()->{
synchronized (list){
list.add(Thread.currentThread().getName());
}
}).start();
}
for (int i = 5;i>0;i--){
Thread.sleep(1000);
System.out.println("倒计时"+i);
}
System.out.println(list.size());
}
}
输出结果:
倒计时5
倒计时4
倒计时3
倒计时2
倒计时1
10000
注意结果安全
案例三(2):
import java.util.concurrent.CopyOnWriteArrayList;
public class SafeJUCList {
public static void main(String[] args) throws InterruptedException {
//保证线程安全的list , ArrayList
CopyOnWriteArrayList list = new CopyOnWriteArrayList();
for (int i = 0; i < 10000; i++) {
new Thread(()->{
list.add(Thread.currentThread().getName());
}).start();
}
for (int i = 5;i>0;i--){
Thread.sleep(1000);
System.out.println("倒计时"+i);
}
System.out.println(list.size());
}
}
输出结果:
倒计时5
倒计时4
倒计时3
倒计时2
倒计时1
10000
注意输出结果安全
class A{
private final ReentrantLock lock = new ReenTrantLock();
public void m(){
lock.lock();
try{
//保证线程安全的代码;
}
finally{
lock.unlock();
//如果同步代码有异常,要将unlock()写入finally语句块
}
}
案例:
Lock锁
import java.util.concurrent.locks.ReentrantLock;
public class TestLock {
public static void main(String[] args) {
HelloWorld helloWorld = new HelloWorld();
new Thread(helloWorld).start();
new Thread(helloWorld).start();
}
}
class HelloWorld implements Runnable{
int ticketNums = 100;
//可重入锁
private final ReentrantLock lock = new ReentrantLock();
@Override
public void run() {
while (true){
try {
lock.lock(); //加锁
//判断是否有票
if (ticketNums>0){
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(ticketNums--);
}else {
break;
}
} finally {
lock.unlock();//解锁
}
}
}
}
死锁避免方法:
案例:
//死锁问题
//两个线程抱着自己的锁 , 然后想要对方的锁 .
// 于是产生一个问题 —> 死锁
public class DeadLocked {
public static void main(String[] args) {
Makeup g1 = new Makeup(0,"白雪公主");
Makeup g2 = new Makeup(1,"灰姑凉");
new Thread(g1).start();
new Thread(g2).start();
}
}
//化妆
class Makeup implements Runnable{
//选择
int choice;
//谁进来了
String girlName;
//两个对象
static LipStick lipStick = new LipStick();
static Mirror mirror = new Mirror();
public Makeup(int choice,String girlName){
this.choice = choice;
this.girlName = girlName;
}
@Override
public void run() {
//化妆
try {
makeup();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
//化妆的方法
public void makeup() throws InterruptedException {
if (choice==0){ //先拿口红,再拿镜子
synchronized (lipStick){
System.out.println("拿到口红");
Thread.sleep(1000);
//等待拿镜子的人释放锁
synchronized (mirror){
System.out.println("拿到镜子");
}
}
}else { //先拿镜子 , 再拿口红
synchronized (mirror){
System.out.println("拿到镜子");
Thread.sleep(2000);
//等待拿口红的人释放锁
synchronized (lipStick){
System.out.println("拿到口红");
}
}
}
}
}
//口红
class LipStick{
}
//镜子
class Mirror{
}
结果输出:
程序死了