2024年最全多线程学习笔记04线程同步,字节跳动 前端面试题

最后

由于篇幅限制,pdf文档的详解资料太全面,细节内容实在太多啦,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

this.name = name;

}

}

//银行,模拟取款

class Drawing extends Thread {

Account account;//账户

//取了多少钱

int drawingMoney;

//手中现有多少钱

int nowMoney;

public Drawing(Account account, int drawingMoney, String name) {

super(name);

this.account = account;

this.drawingMoney = drawingMoney;

}

//取钱

@Override

public void run() {

//判断有没有钱

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 = nowMoney + drawingMoney;

System.out.println(account.name + “余额为:” + account.money);

//这里this.getName()等价于Thread.currentThread().getName()

System.out.println(this.getName() + “手里的钱” + nowMoney);

}

}

运行结果:

在这里插入图片描述

原本卡内只有100万,两人都成功取钱,卡中余额为-50,线程不安全!

案例三,不安全的集合:

import java.util.ArrayList;

import java.util.List;

//线程不安全的集合,ArrayList就是一个

public class UnsafeList {

public static void main(String[] args) {

List list = new ArrayList();

for (int i = 0; i < 10000; i++) {

new Thread(() -> {

list.add(Thread.currentThread().getName());

}).start();

}

System.out.println(list.size());

}

}

在这里插入图片描述

不安全的原因是有两个线程在同一瞬间,将两个数组添加到同一个位置,从而产生了数据覆盖。

同步方法

  • 由于我们可以通过private关键字来保证数据对象只能被方法访问(get、set方法),所以我们只需要针对方法提出一套机制,这套机制就是synchronized 关键字,它包括两种用法:synchronized方法和synchronized块

同步方法: public synchronized void method(int args){}

  • synchronized方法控制对“对象”的访问,每个对象对应一把锁,每个synchronized方法都必须获得调用该方法的对象的锁才能执行,否则线程会阻塞,方法一旦执行,就独占该锁,直到该方法返回才释放锁,后面被阻塞的线程才能获得这个锁,继续执行

缺陷:若将一个大的方法申明为synchronized 将会影响效率

同步方法弊端
  • 方法里面需要修改的内容才需要锁,锁的太多,浪费资源

将购票案例优化:

public class UnsafeBuyTicket {

public static void main(String[] args) {

BuyTicket station = new BuyTicket();

//三个人买票

new Thread(station, “我”).start();

new Thread(station, “你”).start();

new Thread(station, “黄牛党”).start();

}

}

class BuyTicket implements Runnable {

//票

private int ticketNums = 10;

boolean flag = true;//外部停止方式

@Override

public void run() {

//买票

while (flag) {

buy();

}

}

//synchronized同步方法,锁的是this

private synchronized void buy() {

//判断是否有票

if (ticketNums <= 0) {

flag = false;

return;

}

//模拟延时

try {

Thread.sleep(80);

} catch (InterruptedException e) {

e.printStackTrace();

}

//买票

System.out.println(Thread.currentThread().getName() + “拿到了” + ticketNums–);

}

}

运行结果:

在这里插入图片描述

  • 同步块:synchronized (Obj ){ }

  • Obj称之为同步监视器

Obj可以是任何对象,但是推荐使用共享资源作为同步监视器

同步方法中无需指定同步监视器,因为同步方法的同步监视器就是this ,就是这个对象本身,或者是 class[反射中讲解]

  • 同步监视器的执行过程

1、第一个线程访问,锁定同步监视器,执行其中代码;

2、第二个线程访问,发现同步监视器被锁定,无法访问;

3、第一个线程访问完毕,解锁同步监视器;

4、第二个线程访问,发现同步监视器没有锁,然后锁定并访问

优化取钱案例:

public class UnsafeBank {

public static void main(String[] args) {

//账户

Account account = new Account(100, “结婚基金”);

Drawing you = new Drawing(account, 50, “你”);//你要取50万

Drawing girlfriend = new Drawing(account, 100, “girlfriend”);//你你女朋友要全部取走

//你们两个都要取钱

you.start();

girlfriend.start();

}

}

//账户

class Account {

int money;//余额

String name;//卡名

public Account(int money, String name) {

this.money = money;

this.name = name;

}

}

//银行,模拟取款

class Drawing extends Thread {

Account account;//账户

//取了多少钱

int drawingMoney;

//手中现有多少钱

int nowMoney;

public Drawing(Account account, int drawingMoney, String name) {

super(name);

this.account = account;

this.drawingMoney = drawingMoney;

}

//取钱

@Override

public void run() {

//锁的对象就是变化的量,需要增删改的对象

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 = nowMoney + drawingMoney;

System.out.println(account.name + “余额为:” + account.money);

//这里this.getName()等价于Thread.currentThread().getName()

System.out.println(this.getName() + “手里的钱” + nowMoney);

}

}

}

运行结果:

在这里插入图片描述

至于不安全的集合:

public class UnsafeList {

public static void main(String[] args) {

List list = new ArrayList();

for (int i = 0; i < 10000; i++) {

new Thread(() -> {

synchronized (list) {

list.add(Thread.currentThread().getName());

}

}).start();

}

try {

Thread.sleep(3000);

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println(list.size());

}

}

在这里插入图片描述

死锁

多个线程各自占有一些共享资源,并且互相等待其他线程占有的资源才能运行,而导致两个或者多个线程都在等待对方释放资源,都停止执行的情形。某一个同步块同时拥有“两个以上对象的锁”时,就可能会发生“死锁”的问题。

案例 化妆:

化妆需要两个资源,口红和镜子

//死锁:多个线程互相抱着对方需要的资源,然后形成僵持

public class DeadLock {

public static void main(String[] args) {

Makeup g1 = new Makeup(0, “灰姑娘”);

Makeup g2 = new Makeup(1, “白雪公主”);

g1.start();

g2.start();

}

}

//口红

class Lipstick {

}

//镜子

class Mirror {

}

class Makeup extends Thread {

//需要的资源只有一份,用static来保证

static Lipstick lipstick = new Lipstick();

static Mirror mirror = new Mirror();

int choice;//选择

String girlName;//使用化妆品的人

public Makeup(int choice, String girlName) {

this.choice = choice;

this.girlName = girlName;

}

@Override

public void run() {

try {

makeup();

} catch (InterruptedException e) {

e.printStackTrace();

}

}

//化妆方法,互相持有对方的锁,需要拿到对方的资源

private void makeup() throws InterruptedException {

if (choice == 0) {

synchronized (lipstick) {//获得口红的锁

System.out.println(this.girlName + “获得口红的锁”);

Thread.sleep(1000);

synchronized (mirror) {//一秒钟后想获得镜子

System.out.println(this.girlName + “获得镜子的锁”);

}

}

} else {

synchronized (mirror) {//获得镜子的锁

System.out.println(this.girlName + “获得镜子的锁”);

Thread.sleep(2000);

synchronized (lipstick) {//一秒钟后想获得镜子

System.out.println(this.girlName + “获得口红的锁”);

}

}

}

}

}

运行结果:

在这里插入图片描述

灰姑娘获得口红,下一步她想获得镜子,而白雪公主获得了镜子,下一步她想获得口红,所以程序僵持在这里无法继续。

解决办法:

private void makeup() throws InterruptedException {

if (choice == 0) {

synchronized (lipstick) {//获得口红的锁

System.out.println(this.girlName + “获得口红的锁”);

Thread.sleep(1000);

}

synchronized (mirror) {//一秒钟后想获得镜子

System.out.println(this.girlName + “获得镜子的锁”);

}

} else {

synchronized (mirror) {//获得镜子的锁

System.out.println(this.girlName + “获得镜子的锁”);

Thread.sleep(2000);

}

synchronized (lipstick) {//一秒钟后想获得镜子

System.out.println(this.girlName + “获得口红的锁”);

}

}

}

不要抱着对方的锁

运行结果:

在这里插入图片描述

小结

产生死锁的四个必要条件:

最后

一个好的心态和一个坚持的心很重要,很多冲着高薪的人想学习前端,但是能学到最后的没有几个,遇到困难就放弃了,这种人到处都是,就是因为有的东西难,所以他的回报才很大,我们评判一个前端开发者是什么水平,就是他解决问题的能力有多强。

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

分享一些前端面试题以及学习路线给大家

d.sleep(2000);

}

synchronized (lipstick) {//一秒钟后想获得镜子

System.out.println(this.girlName + “获得口红的锁”);

}

}

}

不要抱着对方的锁

运行结果:

在这里插入图片描述

小结

产生死锁的四个必要条件:

最后

一个好的心态和一个坚持的心很重要,很多冲着高薪的人想学习前端,但是能学到最后的没有几个,遇到困难就放弃了,这种人到处都是,就是因为有的东西难,所以他的回报才很大,我们评判一个前端开发者是什么水平,就是他解决问题的能力有多强。

开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】

分享一些前端面试题以及学习路线给大家

[外链图片转存中…(img-RuEhXbC7-1715679713837)]

[外链图片转存中…(img-Hsc1EZqP-1715679713837)]

  • 7
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值