javaEE初阶————多线程初阶(3)
大家新年快乐呀,今天是第三期啦,大家前几期的内容掌握的怎么样啦?
1,线程死锁
1.1 构成死锁的场景
a)一个线程一把锁
这个在java中是不会发生的,因为我们之前讲的可重入机制,在其他语言中可能会发生的;
public static void main(String[] args) throws InterruptedException {Object locker = new Object();Thread t1 = new Thread(()->{synchronized (locker){synchronized (locker){System.out.println(1111);}}});t1.start();t1.join();System.out.println("main");}
按理来说,t1线程刚进synchronized就获取到了锁对象,就要保持,进入第二个synchronized就要请求第一个锁对象,第一个要保持不给你锁对象,它让第二个先给他,第二个synchronized说你先给我我才有锁对象给你呀,它俩就这么一直僵持着,但是java有可重入机制不会发生这样的死锁的;
b)两个线程两把锁
我们来模拟一个吃饺子的过程,小明小亮吃饺子,有酱油和醋对应两把锁,他们喜欢这两个东西一起加(我不喜欢),
public class Demo2 {public static void main(String[] args) throws InterruptedException {Object locker1 = new Object();//酱油Object locker2 = new Object();//醋Thread t1 = new Thread(()->{synchronized (locker1){System.out.println("获取到了酱油");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}synchronized (locker2){System.out.println("酱油和醋都是" + Thread.currentThread().getName() + "的啦");}}},"小明");Thread t2 = new Thread(()->{synchronized (locker2){System.out.println("获取到了醋");try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}synchronized (locker1){System.out.println("酱油和醋都是" + Thread.currentThread().getName() + "的啦");}}},"小亮");t1.start();t2.start();}
}
我们来看运行结果
没有人获得酱油和醋,并且程序也没有正常停止,
这俩线程都因为锁竞争阻塞了,这就构成了死锁,我们加那个sleep是为了保证小亮拿醋,小明拿酱油之后再竞争互相的,不然可能就小明太快了直接全拿走了,或者小亮全拿走了;
c)n个线程m把锁
一个很经典的模型,哲学家就餐问题:
1,哲学家可以放下筷子思考
2,哲学家拿筷子可以吃面条(没有洁癖)两根才能吃
但是哲学家都很固执,拿到了筷子是不会放手的,那么如果在当前这个图上每个人都想吃面条,每个人都到拿到了面前的筷子,要吃面条还需要一个筷子,他们就会想要别人的筷子,然而每个人都不会放开自己的筷子,你等我,我等你,最后大家都饿死,这就构成了死锁;但是按理来说这个模型出现这样的情况非常非常低那么中国10几亿人,这个概率就会无限放大,线程安全要做到完全没有危险的概率;
1.2 死锁的四个必要条件
a)互斥(基本特性)
一个线程获取到锁,其他线程再想获得这个锁就要阻塞等待;
b)不可剥夺(基本特性)
也可以叫不可抢占,如果线程1获取到了锁,线程2再想获取锁是不可以抢夺的,必须阻塞等待;
c)请求和保持
一个线程获取了锁1之后再不放弃锁1的前提下获取锁2;
d)循环等待
a等待b,b等待c,c等待d,d等待a;构成死锁循环;
1.3 如何避免死锁
我们刚才说的构成死锁的四种情况中,互斥和不可剥夺是锁的基本特性,我们是改变不了的,我们只能去改变(请求保持和循环等待);
a)打破请求和保持
请求和保持大概率是发生在嵌套中的,我们可以用并列来代替嵌套,但是通用性较低;
我们就拿刚才的吃饺子来举例子把;
public class Demo1 {public static void main(String[] args) {Object locker1 = new Object();//酱油Object locker2 = new Object();//醋Thread t1 = new Thread(()->{synchronized (locker1){System.out.println("小明拿到酱油");}try {Thread.sleep(1111);} catch (InterruptedException e) {throw new RuntimeException(e);}synchronized (locker2){System.out.println("小明拿到醋");}},"小明");Thread t2 = new Thread(()->{synchronized (locker1){System.out.println("小亮拿到酱油");}try {Thread.sleep(1111);} catch (InterruptedException e) {throw new RuntimeException(e);}synchronized (locker2){System.out.println("小亮拿到醋");}},"小亮");t1.start();t2.start();}
}
并列锁,虽然没有构成死锁,但是违背了我们的想法就是让小明和小亮获得两个锁,刚才说的通用性不强也是在这里;
b)打破循环等待
第二个方法,改变加锁的顺序,我们还有吃饺子的例子,但是这次要拿到两个锁:
public class Demo2 {public static void main(String[] args) {Object locker1 = new Object();//酱油Object locker2 = new Object();//醋Thread t1 = new Thread(()->{synchronized (locker1){System.out.println("小明拿到酱油啦");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (locker2){System.out.println("小明拿到醋和酱油啦");}}},"小明");Thread t2 = new Thread(()->{synchronized (locker1){System.out.println("小亮拿到酱油啦");try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}synchronized (locker2){System.out.println("小亮拿到醋和酱油啦");}}},"小亮");t1.start();t2.start();}
}
我们改变了加锁的顺序,
也是能避免死锁问题的;
———————————————————————————————————————————
2,内存可见性
这也是导致线程安全的问题之一
我们来写一个例子嗷:
import java.util.Scanner;public class Demo3 {static int i = 0;public static void main(String[] args) {Object obj = new Object();Thread t1 = new Thread(()->{while(i==0){}System.out.println("结束");});Thread t2 = new Thread(()->{try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}Scanner scanner = new Scanner(System.in);synchronized (obj){i = scanner.nextInt();}});t1.start();t2.start();}
}
我们来看这个代码,t1线程根据i的数值一直循环直到i的值被t2线程修改才停止,事实是这样的吗。我们来试试,
无法暂停,这是为啥:
这就是因为内存可见性问题,程序员的水平参差不齐,java大佬为了照顾我们这样的小卡拉米,就弄了个编译器优化,所以我们写的代码并不会直接执行,我们刚才写的while(i==0)这段代码,我们要等待t2线程来修改i,可能我们就用了几秒的时间但对于t1线程,这这边是沧海桑田,万物轮回,谁还记得什么t1呀它等于0就得了,再底层一点解释呢,就是有“工作内存” 和 “主内存”我们应该是从主内存中拿到数据,放到工作内存中,再从工作内存放回主内存,但是这么一直一直重复,去主内存的时间开销是工作内存的几千倍,编译器就不去主内存了直接去工作内存中拿数据,但是后期修改了主内存,然而此处代码已经完全忽略主内存了,就无法修改了;这就是内存可见性问题那么怎么避免呢?
———————————————————————————————————————————
3,volatile 关键字
我们可以使用volatile关键字避免内存可见性问题;
3.1 volatile 能保证内存可见性
import java.util.Scanner;public class Demo3 {volatile static int i = 0;public static void main(String[] args) {Object obj = new Object();Thread t1 = new Thread(()->{while(i==0){}System.out.println("结束");});Thread t2 = new Thread(()->{try {Thread.sleep(1000);} catch (InterruptedException e) {throw new RuntimeException(e);}Scanner scanner = new Scanner(System.in);synchronized (obj){i = scanner.nextInt();}});t1.start();t2.start();}
}
看,解决了吧,就加了一个volatile;
3.2 volatile 不能保证原子性
还记得原子性吗,就是这个操作在底层是不是原子的,是不是分几步,再多线程中会影响到这个操作,我们拿之前那个两个线程修改一个整形:
public class Demo4 {volatile static int count = 0;public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{for(int i=0;i<100000;i++){count++;}});Thread t2 = new Thread(()->{for(int i=0;i<100000;i++){count++;}});t1.start();t2.start();t1.join();t2.join();System.out.println("Final count: " + count);}
}
所有我们只有使用锁才行;
———————————————————————————————————————————
4,wait 和 notify
这是个什么玩意,线程不是随机调度,抢占式执行的吗,我们可以用这个玩意稍加限制,协调线程之间的逻辑顺序;
4.1 wait() 方法
这个东西跟锁和sleep不一样,都是等待,但是是有区别的,wait()是等的时候会释放锁,被唤醒再拿到锁而sleep这个byd它抱着锁睡,............锁的话就是阻塞等待嘛,
我们来试试wait()方法是搭配锁来使用的,最好还要加while循环
public class Demo5 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(()->{System.out.println("Thread 1");synchronized (locker){System.out.println(1000);try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println(2000);}});t1.start();}
}
我们来看看运行结果
死等,因为没有东西能够唤醒wait();
我们可以设置超时时间,也可以使用notify方法;
public class Demo5 {public static void main(String[] args) {Object locker = new Object();Thread t1 = new Thread(()->{System.out.println("Thread 1");synchronized (locker){System.out.println(1000);try {locker.wait(2000);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println(3000);}});t1.start();}
}
这样代码会在2秒后打印3000;
4.2 notify() 方法
用来唤醒wait()注意这些都是搭配锁对象来用的;
对于notify,如果存在多个使用同一个锁对象的wait,它没有规律,会随机唤醒一个wait
public class Demo6 {public static void main(String[] args) throws InterruptedException {Object locker = new Object();Thread t1 = new Thread(()->{System.out.println("Thread 1");synchronized (locker){System.out.println("线程1获得锁");try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("线程1释放锁");}});Thread t2 = new Thread(()->{System.out.println("Thread 2");synchronized (locker){System.out.println("线程2获得锁");try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("线程2释放锁");}});t1.start();t2.start();Thread.sleep(1000);synchronized (locker){locker.notify();}}
}
线程1成功释放了锁,说明notify唤醒了线程1的waite
我们再试试
4.3 notifyAll() 方法
这个就是全部释放
public class Demo7 {public static void main(String[] args) throws InterruptedException {Object locker = new Object();Thread t1 = new Thread(()->{System.out.println("Thread 1");synchronized (locker){System.out.println("线程1获得锁");try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("线程1释放锁");}});Thread t2 = new Thread(()->{System.out.println("Thread 2");synchronized (locker){System.out.println("线程2获得锁");try {locker.wait();} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("线程2释放锁");}});t1.start();t2.start();Thread.sleep(1000);synchronized (locker){locker.notifyAll();}}
}
完美
4.4 wait 和 sleep的对比
这个没啥好说的了,wait先加锁,到。wait()操作的时候释放锁,唤醒的时候再拿着锁,而sleep纯抱着锁睡,还会被interrupt唤醒,说实话抱着锁睡听不好的,可能会有很多线程都等着它很浪费时间的,sleep会释放Cpu的资源,不再占用了;就这样吧,大家加油,等我更新下一期;
相关文章:
javaEE初阶————多线程初阶(3)
大家新年快乐呀,今天是第三期啦,大家前几期的内容掌握的怎么样啦? 1,线程死锁 1.1 构成死锁的场景 a)一个线程一把锁 这个在java中是不会发生的,因为我们之前讲的可重入机制,在其他语言中可…...
Deep Sleep 96小时:一场没有硝烟的科技保卫战
2025年1月28日凌晨3点,当大多数人还沉浸在梦乡时,一场没有硝烟的战争悄然打响。代号“Deep Sleep”的服务器突遭海量数据洪流冲击,警报声响彻机房,一场针对中国关键信息基础设施的网络攻击来势汹汹! 面对美国发起的这场…...
【AI应用】免费的文本转语音工具:微软 Edge TTS 和 开源版 ChatTTS 对比
【AI论文解读】【AI知识点】【AI小项目】【AI战略思考】【AI日记】【读书与思考】【AI应用】 我试用了下Edge TTS,感觉还不错,不过它不支持克隆声音(比如自己的声音) 微软 Edge TTS 和 开源版 ChatTTS 都是免费的 文本转语音&…...
Deepseek-v3 / Dify api接入飞书机器人go程序
准备工作 开通了接收消息权限的飞书机器人,例如我希望用户跟飞书机器人私聊,就需要开通这个权限:读取用户发给机器人的单聊消息 im:message.p2p_msg:readonly准备好飞书机器人的API key 和Secretdeepseek-v3的api keysecret:http…...
流媒体技术原理
流媒体技术的原理主要涉及以下几个核心概念和技术: 1. 编码和压缩 编码:视频和音频原始数据通常非常庞大。为了传输和存储,首先需要通过编码将这些数据转换成更小、更易处理的格式。常见视频编码标准包括H.264、H.265(HEVC&…...
matlab simulink 三级倒立摆LQR控制
1、内容简介 略 matlab simulink 122-三级倒立摆LQR控制 可以交流、咨询、答疑 2、内容说明 略 要求初始条件[0.01 0.01 0.01 0.01 0 0 0 0] 调节时间希望在3s内,超调量尽量的小,最大不能超过0.05; 用simulink的…...
使用令牌桶算法通过redis实现限流
令牌桶算法是一种常用的限流算法,它可以平滑地控制请求的处理速率。在 Java 中结合 Redis 实现令牌桶算法,可以利用 Redis 的原子操作来保证多节点环境下的限流效果。 一 实现思路 初始化令牌桶:在 Redis 中存储令牌桶的相关信息࿰…...
Tableau实用技巧 —— 提取Tableau文件中图片
需求背景 在日常开发过程中,我们时常会遇到两种图片提取需求:一是本地报告中的图片文件丢失需要找回,二是从网络论坛中发现有价值的报告图片希望保存使用。针对这些实际应用场景,以下将详细介绍有效的图片提取方法。 解决思路 …...
记一次golang环境的变化
前两天编译打包了了个文件,把env的 goos 搞坏了 导致运行项目一直报错 先是这样 go: unsupported GOOS/GOARCH pair windows/amd64再是这样 /amd64supported GOOS/GOARCH pair linux咱就说,咱也是知道环境配置的有问题 ( go env GOOS &…...
web直播弹幕抓取分析 signature
声明: 本文章中所有内容仅供学习交流使用,不用于其他任何目的,抓包内容、敏感网址、数据接口等均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关! 前言 最近遇到太多难点了卡了很久&am…...
CVE-2024-13025-Codezips 大学管理系统 faculty.php sql 注入分析及拓展
Codezips 里面有很多cms系统,其中的一个College Management System In PHP With Source Code存在sql注入漏洞。 复现 对源码进行下载登录。 里面有很多远程js加载不出来但是不影响接口使用。 对于/college-mgmt-php-master/Front-end/faculty.php接口进行测试。…...
第二个Qt开发实例:在Qt中利用GPIO子系统和sysfs伪文件系统实现按钮(Push Button)点击控制GPIO口(效果为LED2灯的灭和亮)
引言 本文承接博文 https://blog.csdn.net/wenhao_ir/article/details/145420998 里的代码,在那里面代码的基础上添加上利用sysfs伪文件系统实现按钮(Push Button)点击控制GPIO口的代码,进而实现LED2灯的灭和亮。 最终的效果是点击下面的LED按钮实现LED…...
【自动化测试】使用Python selenium类库模拟手人工操作网页
使用Python selenium类库模拟手人工操作网页 背景准备工作安装Python版本安装selenium类库下载selenium驱动配置本地环境变量 自动化脚本输出页面表单自动化填充相关代码 背景 待操作网页必须使用IE浏览器登录访问用户本地只有edge浏览器,通过edge浏览器IE模式访问…...
Elasticsearch:向量搜索的快速介绍
作者:来自 Elastic Valentin Crettaz 本文是三篇系列文章中的第一篇,将深入探讨向量搜索(也称为语义搜索)的复杂性,以及它在 Elasticsearch 中的实现方式。 本文是三篇系列文章中的第一篇,将深入探讨向量搜…...
低至3折,百度智能云千帆宣布全面支持DeepSeek-R1/V3调用
DeepSeek-R1和 DeepSeek-V3模型已在百度智能云千帆平台上架 。 出品|产业家 新年伊始,百度智能云又传来新动作 。 2月3日百度智能云宣布, DeepSeek-R1和 DeepSeek-V3模型已在百度智能云千帆平台上架,同步推出超低价格方案,并…...
VSCode中使用EmmyLua插件对Unity的tolua断点调试
一.VSCode中搜索安装EmmyLua插件 二.创建和编辑launch.json文件 初始的launch.json是这样的 手动编辑加上一段内容如下图所示: 三.启动调试模式,并选择附加的进程...
RabbitMQ 从入门到精通:从工作模式到集群部署实战(三)
文章目录 使用CLI管理RabbitMQrabbitmqctlrabbitmq-queuesrabbitmq-diagnosticsrabbitmq-pluginsrabbitmq-streamsrabbitmq-upgraderabbitmqadmin 使用CLI管理RabbitMQ RabbitMQ CLI 工具需要安装兼容的 Erlang/OTP版本。 这些工具假定系统区域设置为 UTF-8(例如en…...
2025软件授权与保护领域的新趋势
2024年对威步而言是一个重要的里程碑——公司成立35年以来,一直专注于软件安全及软件授权管理,以保护企业的数字资产与知识产权。展望2025年,数字资产盗窃、数据泄露与网络犯罪等威胁仍在持续增长,威步将在新的形势下继续推动技术…...
线段树(点修,区查,区修)
文章目录 什么是线段树?线段树能够解决什么样的问题?模板 什么是线段树? 线段树是一种二叉搜索树,而二叉搜索树,首先满足二叉树,即每个结点最多有两颗子树,并且是一颗搜索树,我们要知…...
深度学习 - 神经网络的原理
## 深度学习 - 神经网络的原理 深度学习是机器学习的一个分支,其核心是模拟人脑神经网络的结构和功能,构建多层的神经网络模型,从数据中学习特征并进行预测或分类。 **神经网络的基本原理:** 1. **神经元模型:** * 神经网…...
DeepSeek辅助段落扩写的能力怎么样?
DeepSeek-R1在学术写作的诸多细节层面展现出了显著的应用价值。接下来我们将通过一系列具体案例,深入探讨该工具如何在扩写、翻译、发表以及内容改进等关键环节为学术写作提供有力支持。在提问环节,DeepSeek-R1能够高效地简化提示词,并精准地…...
深入Linux系列之进程地址空间
深入Linux系列之进程地址空间 1.引入 那么在之前的学习中,我们知道我们创建一个子进程的话,我们可以在代码层面调用fork函数来创建我们的子进程,那么fork函数的返回值根据我们当前所处进程的上下文是返回不同的值,它在父进程中返…...
虚拟DOM与Diff算法:Vue如何高效更新UI?
虚拟DOM与Diff算法:Vue如何高效更新UI? 虚拟DOM与Diff算法:Vue如何高效更新UI?什么是虚拟DOM?定义虚拟DOM的优势 Diff算法:如何高效计算UI差异定义核心思想Diff算法的步骤示例代码 Vue中的虚拟DOM与Diff算法…...
Golang 并发机制-6:掌握优雅的错误处理艺术
并发编程可能是提高软件系统效率和响应能力的一种强有力的技术。它允许多个工作负载同时运行,充分利用现代多核cpu。然而,巨大的能力带来巨大的责任,良好的错误管理是并发编程的主要任务之一。 并发代码的复杂性 并发编程增加了顺序程序所不…...
【MySQL】第二弹---数据库基础全解析:从概念到实践的深度探索
✨个人主页: 熬夜学编程的小林 💗系列专栏: 【C语言详解】 【数据结构详解】【C详解】【Linux系统编程】【MySQL】 目录 1. 数据库基础 1.1 什么是数据库 1.2 主流数据库 1.3 基本使用 1.3.1 MySQL安装 1.3.2 连接服务器 1.3.3 服务器…...
c++计算机教程
目的 做出-*/%计算机 要求 做出可以计算-*/%的计算机 实现 完整代码 #include<bits/stdc.h> int main() {std::cout<<"加 减- 乘* 除/ 取余% \没有了|(因为可以算三位)"<<"\n"<<"提示:每打完一个符号或打完一个数,\…...
win32汇编环境,对话框程序中自定义工具栏的使用示例三
;运行效果 ;win32汇编环境,对话框程序中自定义工具栏的使用示例三 ;这次是竖着的,以下为生成48*48大小的自定义工具栏图标,自已设计图标样式,显得更专业点。 ;原理是,先生成工具栏控件,再生成图像列表,然后弄几个图标加入图像列表,再把图像列表与工具栏控件关联。需留意…...
集合类不安全问题
ArrayList不是线程安全类,在多线程同时写的情况下,会抛出java.util.ConcurrentModificationException异常 解决办法: 1.使用Vector(ArrayList所有方法加synchronized,太重) 2.使用Collections.synchronized…...
怎么使用Cursor以及升级Cursor pro会员
什么是cursor Cursor:结合AI技术的代码编辑器,助力开发者提升编码效率与质量。作为Visual Studio Code的一个衍生版本,Cursor继承了其用户熟知的界面和插件兼容性,并加入了革命性的AI特性。这款编辑器自2023年1月推出以来&#x…...
启用gui,启动图形化界面
1、停止服务 2、开启maxscale GUI ,修改主配置文件(增加框框内两行) 3、启动服务 注:如果出现以下启动不成功 考虑权限问题 4、访问http://ip:8989 用户名/密码:admin/mariadb...
03-移除元素
给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素。元素的顺序可能发生改变。然后返回 nums 中与 val 不同的元素的数量。 假设 nums 中不等于 val 的元素数量为 k,要通过此题,您需要执行以下操作: 更改…...
LSSVM最小二乘支持向量机多变量多步光伏功率预测(Matlab)
代码下载:LSSVM最小二乘支持向量机多变量多步光伏功率预测(Matlab) LSSVM最小二乘支持向量机多变量多步光伏功率预测 一、引言 1.1、研究背景与意义 随着全球能源危机和环境问题的日益严重,可再生能源的开发利用成为了世界各国…...
使用Vue开发可复用的Web Components:跨框架组件封装指南
使用Vue开发可复用的Web Components:跨框架组件封装指南 使用Vue开发可复用的Web Components:跨框架组件封装指南引言什么是Web Components?为什么选择Vue开发Web Components? 封装跨框架组件的步骤1. 创建基本的Vue组件2. 将Vue组…...
用AI写游戏1——js实现贪吃蛇
使用模型通义千问 提示词: 用js html css 做一个贪吃蛇的动画 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Snake Game</title><link rel"stylesheet" href"c…...
星闪开发入门级教程之安装编译器与小项目烧录
系列文章目录 星闪开发入门级教程 好久不见,已经好几年没有发文章了,星闪-作为中国原生的新一代近距离无线联接技术品牌。我想着写点东西。为了适合新手,绝对小白文。 文章目录 系列文章目录前言一、Hispark Studio1.安装Hispark Studio2.安…...
java求职学习day32
JavaScript 详解 课程目标: 1 、 JavaScript 介绍 2 、 HTML 和 JavaScript 结合方式 3 、 JavaScript 的使用 4 、 DOM 操作 5 、 BOM 操作 1. JavaScript介绍 (1)虽然是 java 作为前缀,但 java 和 javascript 的关系,就像老婆和老婆…...
【Markdown语法】锚点机制:跳转任意位置
最近写文章时,发现有一个需求:想要实现一种点击跳转到文档中任意位置的功能,这就是锚点机制,就像游戏中的传送锚点,于是写成文章记录一下使用方式。 本文将详细介绍如何使用Markdown创建文档内部跳转(即锚…...
Docker安装pypiserver私服
Docker安装pypiserver私服 1 简介 Python开源包管理工具有pypiserver、devpi和Nexus等,pypiserver安装部署比较简单,性能也不错。 搭建pypiserver私服,可以自己构建镜像,也可以使用官网的docker镜像。 # Github地址 https://g…...
基于微信小程序的居住证申报系统设计与实现(LW+源码+讲解)
专注于大学生项目实战开发,讲解,毕业答疑辅导,欢迎高校老师/同行前辈交流合作✌。 技术范围:SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:…...
网站改HTTPS方法
默认的网站建设好后打开的样子那看起来像是钓鱼网站,现在的浏览器特别只能,就是你新买来的电脑默认的浏览器同样也会出现这样“不安全”提示。 传输协议启动了向全球用户安全传输网页内容的流程。然而,随着HTTPS的推出,传输协议通…...
什么是三层交换技术?与二层有什么区别?
什么是三层交换技术?让你的网络飞起来! 一. 什么是三层交换技术?二. 工作原理三. 优点四. 应用场景五. 总结 前言 点个免费的赞和关注,有错误的地方请指出,看个人主页有惊喜。 作者:神的孩子都在歌唱 大家好…...
极客说|利用 Azure AI Agent Service 创建自定义 VS Code Chat participant
作者:卢建晖 - 微软高级云技术布道师 「极客说」 是一档专注 AI 时代开发者分享的专栏,我们邀请来自微软以及技术社区专家,带来最前沿的技术干货与实践经验。在这里,您将看到深度教程、最佳实践和创新解决方案。关注「极客说」&a…...
Rust语言进阶之标准输入: stdin用法实例(一百零五)
简介: CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布:《Android系统多媒体进阶实战》🚀 优质专栏: Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏: 多媒体系统工程师系列【…...
力扣刷题思路
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言递归70. 爬楼梯112. 路径总和509. 斐波那契数 分治169. 多数元素240.搜索二维矩阵 II --- 二分查找 单调栈 ---「找最近一个比当前值大/小」的问题739. 每日温度…...
MUX-VLAN实验
一、搭建实验拓扑图 二、基本设备配置 设备接口IP地址子网掩码网关PC1E0/0/110.0.1.1255.255.255.0N/APC2E0/0/110.0.1.2255.255.255.0N/APC3E0/0/110.0.1.3255.255.255.0N/APC4E0/0/110.0.1.4255.255.255.0N/APC5E0/0/110.0.1.6255.255.255.0N/Aserver-1E0/0/110.0.1.5255.2…...
音频进阶学习十二——Z变换
文章目录 前言一、Z变换1.Z变换的作用2.Z变换公式3.Z的状态表示1) r 1 r1 r12) 0 < r < 1 0<r<1 0<r<13) r > 1 r>1 r>1 4.关于Z的解释 二、收敛域1.收敛域的定义2.收敛域的表示方式3.ROC的分析1)当 …...
理解推理型大语言模型
构建和改进推理模型的方法与策略 本文描述了构建推理模型的四种主要方法,以及我们如何增强大型语言模型(LLM)的推理能力。我希望这能为你提供有价值的见解,并帮助你了解这一领域快速发展的文献和热潮。 在2024年,LLM…...
【Brinson 绩效归因模型】
Brinson 绩效归因模型 1、前言2、Brinson模型介绍2.1 归因方式2.1.1 BHB 超额收益分解方案2.1.2 BF 超额收益分解方案 2.2 多期 Brinson 模型 1、前言 如此之多的基金,收益率各有不同,即使同等收益率的基金也各有不同的成因。在这种形势下,广…...
如何轻松将Matlab生成的图表嵌入PowerPoint演示文稿
文章目录 Matlab将生成的图添加PPT中一、Matlab脚本1.添加图片函数2.使用示例 总结 Matlab将生成的图添加PPT中 在许多科学、工程和商业领域,Matlab作为一款强大的数值计算和可视化工具,被广泛应用于数据分析和模型构建。然而,当涉及到分享这…...
组合(力扣77)
从这道题开始,我们正式进入回溯算法的学习。之前在二叉树中只是接触到了一丢丢,而这里我们将使用回溯算法解决很多经典问题。 那么这道题是如何使用回溯算法的呢?在讲回溯之前,先说明一下此题是如何递归的。毕竟回溯递归不分家&a…...