贪心算法入门(一)
1.什么是贪心算法?
贪心算法是一种解决问题的策略,它将复杂的问题分解为若干个步骤,并在每一步都选择当前最优的解决方案,最终希望能得到全局最优解。这种策略的核心在于“最优”二字,意味着我们追求的是以最少的时间和精力,快速获得正确的结果。
然而,“希望得到全局最优解”就表示贪心算法并不意味着一定能得到全局最优解。实际上,并不是所有问题都可以通过贪心策略解决。为了确保贪心策略的有效性,需要对其进行严格的证明。而且,不同的问题往往需要采用不同的贪心策略。
如果你觉得这一点仍然比较抽象,接下来我将通过五道具体的例题来详细说明贪心算法的应用及其背后的思路。
2.柠檬水找零
860. 柠檬水找零 - 力扣(LeetCode)


通过示例和问题描述,理解这个问题并不难,所以我就不过多阐述了。就是看能不能给bills数组里面的每个顾客正确的找零,并且一开始我们手上是没有零钱的,也就意味着如果bills[0] > 5, 那么我们是没有办法给顾客找零的,因为此时我们还没有任何零钱。以此类推,当检索完整个数组都能正确找零时返回true,否则返回false。
在理清问题后,不要急于去想出一个贪心策略,也不要急于去想用代码如何实现。而是首先在你脑子里面去想示例里面的答案是如何一步一步得到的。

示例1中,第1位顾客用5美元买柠檬水,不用找零,与此同时我们手上多了一张5元的零钱。以此类推,直到第4位顾客付了10美元,我们需要找零5美元,此时我们的零钱库中有3张5美元,可以正确找零,此时零钱库变为2张5美元,1张10美元。最后一位顾客,我们只需找1张10美元和1张5美元,故可以实现正确找零,返回true。
再把以上思路,变现为代码即可。但其实还有一个细节问题需要我们考虑,就是如果顾客支付20元,而此时我们的零钱库中有3张5美元和1张10美元,那么我们应该选择3张5美元还是1张10美元1张5美元进行找补呢?这时可能会想到用递归的方式将两种方式递归下去,只要有一种的结果为true那么就为true。但同时递归这种方式时间复杂度就会大大提升。此时能不能有一种贪心策略去简化呢?
3张5美元和1张10美元1张5美元,应该更优先选择1张10美元1张5美元进行找补。这就是我们贪心的地方,因为5美元的用处比10美元更大,当顾客支付10美元时,5美元可以找补,同时5美元也可以为20找补。所以我们应该留着用处更大的5美元,以备不时之需。
那么这种策略对不对呢,这个贪心策略得到的结果是不是正确的,需要证明。我们可以反推假设,假设在最优解中,我们在某一次找补20美元的选择中,并没有选择1张10美元和1张5美元,而是选择了3张5美元。而我们的贪心策略是要选择1张10美元和1张5美元的,我们把贪心的选择往假设的最优解中进行替换,看是否会影响后续结果。
代码实现
class Solution {public boolean lemonadeChange(int[] bills) {int five = 0, ten = 0; // five表示零钱库中5美元的张数,ten表示10美元的张数for(int i = 0; i < bills.length; i++){ if(bills[i] == 5) five++; // 更新零钱库else if(bills[i] == 10){if(five == 0) return false; // 如果零钱库没有5美元返回falsefive--;ten++; // 更新零钱库}else{if(ten > 0 && five > 0) {ten--;five--;} // 优先选择用10美元和5美元找补else if(five >= 3) five -= 3; // 次选3张5美元(在没有10美元的情况下)else return false; // 上述条件都不满足则没有合适的零钱找补返回false}}return true;}
}
3. 将数组和减半的最少操作次数
2208. 将数组和减半的最少操作次数 - 力扣(LeetCode)

这道题的贪心策略并不难想,要让操作次数最少,只需每次都挑出数组最大的数进行减半,但是需要注意的是,减半后的数也需要放回原数组,然后挑出最大值。所以不能直接将数组用sort方法进行排序,因为还需要将减半后的数重新放回数组进行比较。采取使用大根堆的数据结构即可。
代码实现
class Solution {public int halveArray(int[] nums) {PriorityQueue<Double> q = new PriorityQueue<>((a, b) -> Double.compare(b, a));int ret = 0;double sum = 0;for(int x : nums){sum += x;q.add((double) x);} // 原数组总和sum /= 2.0; // 计算总和的一半while(sum > 0){double t = q.poll() / 2.0; // 拿出堆中最大的数sum -= t;ret++;q.offer(t);// 减半后的数重新放回大根堆}return ret;}
}
4.最大数
179. 最大数 - 力扣(LeetCode)

以示例1为例,10和2我们很容易看出来,应该让2在高位,10放在2后面。所以实现的逻辑就是,把两个数从高位到低位的数字挨个比较,直至数字不同,并把数字大的放在前面。比如10和2的比较逻辑就是,10的第一位数字1和2比较,2比1大所以放前面。再比如23和22, 23的第一位是2,22的第一位也是2,所以继续向后比较就3和2比较,所以23和22组成的最大值应该是2322。
但还存在一种情况就是,2和20,这种情况应该把谁放在前面呢,肉眼观察最大值应该是220,所以应该把2放前面20放后面。但是这个结论我们应该用逻辑推敲出来而不是凭空想。难道遇到这种情况就是把位数少的放前面吗?那2和24就显然不是这种情况,因为224比242要小。但是从这两个例子就可以看出端倪了,如果多出的数字比首位大那么就是位数多的排前面,否则排后面。2和20中0比2小,所以20应该排在后面,而2和24中4比2大,所以24应该排前面。
所以我们只要按照这种方法对数组进行降序排序,再依次遍历数组就可以得到最终结果值。其实还有一种更简单的比较方法,就是将数a和数b都转换为字符串,然后比较 a + b(a在前,b在后)大还是(b + a)大即可。
代码实现
class Solution {public String largestNumber(int[] nums) {String[] s = new String[nums.length];for(int i = 0; i < nums.length; i++) s[i] = "" + nums[i]; // 将数转换为字符串存入字符数组Arrays.sort(s, (a, b) -> (b + a).compareTo(a + b)); // 按照比较方法对字符数组进行降序排序if(Objects.equals(s[0],"0")) return "0"; // 判断特殊情况,如果排序后的最大值是0就返回0StringBuilder ret = new StringBuilder();for(String t : s) ret.append(t); // 依次添加return ret.toString();}
}
5.摆动序列
376. 摆动序列 - 力扣(LeetCode)

摆动序列其实就是一种波浪型曲线,要你找的这个子序列必须是先升再降再升再...,或者先降再升再降...。而且子序列就表示中间可以不用是连续的,也就表明你可以在数组里面跳着挑满足摆动的数。最终找出最长的子序列即可。
我先说一下我一开始看到这道题的思路,就是用动态规划。但其实这道题用贪心的思路是更简单的,但是动态规划和贪心的时间复杂度是一样的。我在力扣上跑的结果所花的时间也是差不多的。只是贪心的逻辑会更简单,但是贪心的策略是比较扩散性思维才能想到的,但是也不难。这里想讲一下动态规划的思路是因为是比较容易想和常规的。如果想直接看贪心策略的可以直接跳到贪心的思路。
动态规划思路:
首先明确状态表示,最开始想的是dp[i]表示以i位置为结尾的摆动最长子序列长度,但是很快被否定了,因为这样我没有办法写状态转移方程,因为我不知道以i位置结尾的是上升趋势还是下降趋势,就没有办法知道后续那个数可以加在这个状态之上。所以用两个数组表示不同的状态。
int[] f = new int[n]; // 以i结尾并且为上升趋势的摆动序列最长长度int[] g = new int[n]; // 以i结尾并且为下降趋势的摆动序列最长长度
当nums[ j ] < nums[ i ] (j 表示在 i 之前的所有索引值),为上升趋势,所以应该连上为下降趋势的序列组成新的摆动序列,即更新 f[ i ] = g[ i ] + 1, 当nums[ j ] > nums[ i ] 时, 更新g[ i ] = f[ i ] + 1。需要注意的是在更新i位置的值时,需要把i之前的所有的f[ j ] 和 g[ j ]都遍历比较一遍,看接在哪个位置后才能组成最长的摆动子序列。还有就是初始值问题,每个数自己也能单独组成摆渡序列,所以f [ i ] 和 g [ i ] 的初始值都应该为1。
动规代码实现:
class Solution {public int wiggleMaxLength(int[] nums) {int n = nums.length;int[] f = new int[n]; // 以i结尾并且为上升趋势int[] g = new int[n]; // 以i结尾并且为下降趋势for(int i = 0; i < n; i++) f[i] = g[i] = 1;int ret = 1;for(int i = 1; i < n; i++){for(int j = 0; j < i; j++){if(nums[j] > nums[i]) g[i] = Math.max(g[i], f[j] + 1);if(nums[j] < nums[i]) f[i] = Math.max(f[i], g[j] + 1);}ret = Math.max(ret, Math.max(f[i], g[i]));}return ret;}
}
以上实现思路时间复杂度为n², 因为多了一个找出最合适j位置的循环。所以我的优化思路就是想把这个寻找j位置的循环省掉。如果我的f 和 g函数表示在0 - i区间内最长摆动子序列的值,我就不用去找j位置了,直接用这个值往上加即可。但是这样的话我的判断条件又会缺少值,因为不知道最长子序列的最后一位数是什么,就没有办法判断哪些数可以接在后面,所以又新开一个二维数组hash,hash[i][0]表示在0-i区间内最长摆动序列且结尾为下降趋势的最后一位数的值,hash[i][1]表示在0-i区间内最长摆动序列且结尾为上升趋势的最后一位数的值。
所以条件判断变成了,nums[ i ] 和 hash [ i - 1] [ 0 ] 、 hash [ i - 1] [ 1 ] 进行比较,如果nums[ i ] > hash [ i - 1] [ 0 ], 表示当前位置的值大于在0 - i - 1区间内结尾为下降趋势的最长摆动序列的最后一位数,满足条件则更新f[ i ] 的值,并且hash [ i ] [ 1 ] 也要更新为当前数。
f[i] = g[i - 1] + 1; hash[i][1] = nums[i];
如果不满足,f[ i ] 的值应该延续 f[ i - 1] 的值,并且hash [ i ] [ 1 ] 也应该延续hash [ i - 1] [ 1 ] 的值。
g[i]同理。
优化后的代码:
class Solution {public int wiggleMaxLength(int[] nums) {int n = nums.length;int[] f = new int[n]; // 在0-i区间内 结尾为上升趋势int[] g = new int[n]; // 在0-i区间内 结尾并且为下降趋势for(int i = 0; i < n; i++) f[i] = g[i] = 1;int[][] hash = new int[n][2];hash[0][0] = hash[0][1] = nums[0];for(int i = 1; i < n; i++){if(nums[i] > hash[i - 1][0]) { f[i] = g[i - 1] + 1; hash[i][1] = nums[i]; }else { f[i] = f[i - 1]; hash[i][1] = hash[i - 1][1];}if(nums[i] < hash[i - 1][1]) { g[i] = f[i - 1] + 1; hash[i][0] = nums[i]; }else { g[i] = g[i - 1]; hash[i][0] = hash[i - 1][0];}}return Math.max(f[n -1], g[n - 1]);}
}
贪心思路:
把数组用折线表示,下图是我随便画的一种情况,你可以随便用折线画你假设的数组,从图中可以看出我们要要找的摆动序列长度的最大值就是一共有多少个拐点,但是连续的相同拐点只能算一个。这些拐点也叫极点,判断是不是极点的方式:该点和左区间值的差值left和和右区间的差值right相乘是否小于0,小于0则为极点,大于0说明一直都是上升趋势或下降趋势,等于0则说明是连续相等的区间。
贪心代码实现
class Solution {public int wiggleMaxLength(int[] nums) {int n = nums.length;if (n < 2) return n;int ret = 0, left = 0, right = 0;for(int i = 0; i < n - 1; i++){ // 注意没有考虑最后一个right = nums[i + 1] - nums[i]; if(right == 0) continue; // 先将右区间更新到拐点位置if(right * left <= 0) ret++; // 条件中等于0是为了第一次循环左区间为0的特殊情况left = right;}return ret + 1; // 要加上最后一个位置,因为最后一个位置必为极点}
}
6.最长递增子序列
300. 最长递增子序列 - 力扣(LeetCode)

这道题依然会讲解动态规划和贪心两种方法思路,再在动态规划的基础上去优化得到贪心。
动态规划思路:
dp[ i ] 表示以i位置为结尾的最长递增子序列, 状态转移方程为j < i && nums[ j ] < nums[ i ] ,dp [ i ] = max( dp [ j ] + 1) 。
动态规划代码实现
class Solution {public int lengthOfLIS(int[] nums) {int n = nums.length;int[] dp = new int[n]; // 表示以dp[i]位置结尾的最长递增子序列长度Arrays.fill(dp,1);int ret = 1;for(int i = 1; i < n; i++){for(int j = 0; j < i; j++){if(nums[i] > nums[j]) dp[i] = Math.max(dp[i], dp[j] + 1); // 满足条件表示i位置的值可以跟在j位置后形成递增序列}ret = Math.max(ret, dp[i]); // 更新最大值}return ret;}
}
时间复杂度是n²。因为多了找最合适的j位置的循环,需要想办法把这个循环省掉。根据上一题给的思路,依然可以把状态表示dp[ i ] 重新定义为在0 - i区间的最长递增子序列长度。但是依然需要考虑判断条件是跟谁比较,由于这里定义为区间内并不知道结尾的数是多少,所以新开一个数组key[i]表示在0 - i区间内最长递增子序列的最后一个数。看似很合理但是实际上状态转移方程还是写不出来的。
key[0] = nums[0];for(int i = 1; i < n; i++){if(nums[i] > key[i - 1]) { dp[i] = dp[i - 1] + 1; key[i] = nums[i]; }else (nums[i] < key[i - 1]) { dp[i] = dp[i - 1]; key[i] = key[i - 1]; }}
nums[ i ] > key[i - 1] 这种情况是可以正确写出状态转移方程的, 只需在上一个区间的最长序列长度加1即可,并且更新结尾值为当前i位置的值。但如果是小于的情况,dp[ i ] = dp[ i - 1]继续延续上一个区间的状态没错,可是 key[ i ] 的值却不能简单的用延续处理。例如数组为【10, 1 ,3】这种情况,当i = 1 的时候如果简单的把key[ i ] 的值延续上一次的状态为10,那么最终肯定是不能得到正确答案长度为2的序列1,3的。那么key[ i ] 的值在这种情况下应该怎么更新呢?你会发现需要处理很多种情况,并且只记录递增子序列最后的数根本不能应对这些情况,因为有可能最长递增子序列需要以另外一个位置为起点开头,或者说在中间把一些点替换掉。例如【9,10,1,2,3】,【1,2,11,12,5,6,7】。
贪心思路
贪心策略就是在以上思路发散扩展得来的。以示例1为例:
从左至右扫描数组,10可以看成一个单独的递增序列,长度为1。再扫描到9, 9 比 10 小,此时也能单独看成长度为1的递增序列。这时我们把10这个单独递增序列剔除掉,保留9的序列。因为在长度同样的情况下,保留越小的数,能够加在该数后形成递增的情况越多。扫描到2同理,保留2的单独序列,扫描到5,比2大,直接让长度加1,扫描到3,比2大比5小,此时可以把最长递增序列的第二位换成3,最长长度依然为2。以此类推。

通过扫描数组不断地更新每个位置的数,和添加新位置,最终得到最长递增子序列长度。
添加位置的时机:只要比最后一个位置的数大就添加位置。
更新某个位置的时机:从第一个位置依次往后比较,找到一个小于等于该数的位置,在该位置更新,更新的值为当前扫描的数。

贪心代码实现
class Solution {public int lengthOfLIS(int[] nums) {List<Integer> list = new ArrayList<>();for(int x : nums){if(list.size() == 0) list.add(x); // 第一个数直接放入listelse{if(x > list.get(list.size() - 1)) list.add(x); // 如果大于最后一个位置,直接添加放到新位置else{ // 否则从第一个位置开始依次比较找到放入的位置for(int i = 0; i < list.size(); i++){if(x <= list.get(i)) { list.set(i,x); break;} // 找到第一个小于等于的位置更新该位置的值}}}}return list.size();// list的大小表示在过程中最多添加了多少位置}
}
优化:
在更新某个位置的数时,可以使用二分查找,快速找到该位置进行更新。需要注意的是,找出的位置是严格大于上个位置的数的,在更新left和right的时候需要注意这个细节。
class Solution {public int lengthOfLIS(int[] nums) {List<Integer> list = new ArrayList<>();for(int x : nums){if(list.size() == 0) list.add(x);else{if(x > list.get(list.size() - 1)) list.add(x);else{int left = 0, right = list.size() - 1, mid = (left + right) / 2;while(left < right){if(x > list.get(mid)) left = mid + 1;else if(x < list.get(mid)) right = mid;else break;mid = (left + right) / 2;}list.set(mid,x);}}}return list.size();}
}
结语
欢迎大家在评论区讨论留言,有不理解的地方可以私信或者评论。
相关文章:
贪心算法入门(一)
1.什么是贪心算法? 贪心算法是一种解决问题的策略,它将复杂的问题分解为若干个步骤,并在每一步都选择当前最优的解决方案,最终希望能得到全局最优解。这种策略的核心在于“最优”二字,意味着我们追求的是以最少的时间和…...
C# ref和out 有什么区别,分别用在那种场景
在C#中,ref和out都是用于按引用传递参数的关键字,但它们有一些细微的差别和使用场景。 ref 关键字 ref 关键字用于按引用传递参数。这意味着当你将一个变量作为参数传递给一个方法时,你不是传递变量的值,而是传递变量的引用。因…...
TikTok直播专线:提升直播效果和体验
作为当今全球最受欢迎的社交媒体平台之一,TikTok为商家提供了无限的商机和市场。然而,商家在使用TikTok时也面临着许多挑战,如网络延迟、直播中断以及账号被封等问题。TikTok直播专线旨在为商家提供高速稳定的网络连接,助力他们在…...
由浅入深逐步理解spring boot中如何实现websocket
实现websocket的方式 1.springboot中有两种方式实现websocket,一种是基于原生的基于注解的websocket,另一种是基于spring封装后的WebSocketHandler 基于原生注解实现websocket 1)先引入websocket的starter坐标 <dependency><grou…...
1-petalinux 问题记录-根文件系统分区问题
在MPSOC上使用SD第二分区配置根文件系统的时候,需要选择对应的bootargs,但是板子上有emmc和sd两个区域,至于配置哪一种mmcblk0就出现了问题,从vivado中的BlockDesign和MLK XCZU2CG原理图来看的话,我使用的SD卡应该属于…...
微信小程序的上拉刷新与下拉刷新
效果图如下: 上拉刷新 与 下拉刷新 代码如下: joked.wxml <scroll-view class"scroll" scroll-y refresher-enabled refresher-default-style"white" bindrefresherrefresh"onRefresh" refresher-triggered&qu…...
【大语言模型】ACL2024论文-05 GenTranslate: 大型语言模型是生成性多语种语音和机器翻译器
【大语言模型】ACL2024论文-05 GenTranslate: 大型语言模型是生成性多语种语音和机器翻译器 GenTranslate: 大型语言模型是生成性多语种语音和机器翻译器 目录 文章目录 【大语言模型】ACL2024论文-05 GenTranslate: 大型语言模型是生成性多语种语音和机器翻译器目录摘要研究背…...
KPRCB结构之ReadySummary和DispatcherReadyListHead
ReadySummary: Uint4B DispatcherReadyListHead : [32] _LIST_ENTRY 请参考 _KTHREAD *__fastcall KiSelectReadyThread(ULONG LowPriority, _KPRCB *Prcb)...
批处理之for语句从入门到精通--呕血整理
文章目录 一、前言二、for语句的基本用法三、文本解析显神威:for /f 用法详解四、翻箱倒柜遍历文件夹:for /r五、仅仅为了匹配第一层目录而存在:for /d六、计数循环:for /l后记 for语句从入门到精通 一、前言 在批处理中&#…...
pycharm小游戏贪吃蛇及pygame模块学习()
由于代码量大,会逐渐发布 一.pycharm学习 在PyCharm中使用Pygame插入音乐和图片时,有以下这些注意事项: 插入音乐: - 文件格式支持:Pygame常用的音乐格式如MP3、OGG等,但MP3可能需额外安装库…...
redis实战--黑马商城 记录
一、视频地址 黑马程序员Redis入门到实战教程,深度透析redis底层原理redis分布式锁企业解决方案黑马点评实战项目 二、笔记地址 Redis基础篇Redis实战篇...
机器人技术革新:人工智能的强力驱动
内容概要 在当今世界,机器人技术与人工智能的结合正如星星与大海,彼此辉映。随着科技的不断进步,人工智能不仅仅是为机器人赋予了“聪明的大脑”,更是推动了整个行业的快速发展。回顾机器人技术的发展历程,我们会发现…...
漫途焊机安全生产监管方案,提升安全生产管理水平!
随着智能制造时代的到来,企业安全生产管理的重要性日益凸显。特别是在现代工厂中,焊机的安全生产监管成为了一个不容忽视的重要环节。传统的焊机安全生产监管方式存在诸多不足,如人工巡检频率低、数据延迟、安全隐患发现不及时等问题。因此&a…...
动态规划之两个数组的 dp(上)
文章目录 最长公共子序列不相交的线不同的子序列通配符匹配 最长公共子序列 题目:最长公共子序列 思路 选取s1的[0, i]区间以及s2的[0, j]区间作为研究对象 状态表示:dp[i][j]表示,s1的[0, i]区间以及s2的[0, j]区间内…...
DC-9靶机通关
这是这个系列的最后一个靶机了!!!经过前面的锻炼和学习,这次我的目标是尽量不借助任何教程或者提示来拿下这个靶机!!!下面我们看能不能成功!!! 1.实验环境 攻…...
前端注释都应该怎么写?
以下是一些前端注释的例子,展示了如何应用前面提到的建议: 1. 使用清晰、简洁的语言 // 计算两个数的平均值 function calculateAverage(a, b) {return (a b) / 2; }2. 描述代码的目的和功能 // 将日期格式化为 "YYYY-MM-DD" 的字符串 fun…...
深入解析缓存模式下的数据一致性问题
今天,我们来聊聊常见的缓存模式和数据一致性问题。 常见的缓存模式有:Cache Aside、Read Through、Write Through、Write Back、Refresh Ahead、Singleflight。 缓存模式 Cache Aside 在 Cache Aside 模式中,是把缓存当做一个独立的数据源…...
嵌入式常用功能之通讯协议1--IIC
嵌入式常用功能之通讯协议1--串口 嵌入式常用功能之通讯协议1--IIC(本文) 嵌入式常用功能之通讯协议1--SPI 一、IIC总线协议介绍 Inter-Integrated Circuit(集成电路总线),是由 Philips 半导体公司(现在的 NXP 半导体…...
【Wi-Fi】Wi-Fi 7(802.11be) Vs Wi-Fi 8 (802.11bn)
介绍 WiFi 7 (802.11be) 是 WiFi-6 (802.11ax) 的继任者,旨在提高数据速率并减少拥挤环境中的延迟。 WiFi 8 (8021.1bn)是后续标准,专注于提高 WLAN 连接的可靠性, 提高…...
Ubuntu软件包管理机制
文章目录 🍊自我介绍🍊Ubuntu软件包管理机制🍊软件安装命令详解: 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞关注评论收藏(一键四连)哦~ 🍊自我介绍 Hello,大家好…...
网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...
JavaSec-RCE
简介 RCE(Remote Code Execution),可以分为:命令注入(Command Injection)、代码注入(Code Injection) 代码注入 1.漏洞场景:Groovy代码注入 Groovy是一种基于JVM的动态语言,语法简洁,支持闭包、动态类型和Java互操作性,…...
Flask RESTful 示例
目录 1. 环境准备2. 安装依赖3. 修改main.py4. 运行应用5. API使用示例获取所有任务获取单个任务创建新任务更新任务删除任务 中文乱码问题: 下面创建一个简单的Flask RESTful API示例。首先,我们需要创建环境,安装必要的依赖,然后…...
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
高频面试之3Zookeeper
高频面试之3Zookeeper 文章目录 高频面试之3Zookeeper3.1 常用命令3.2 选举机制3.3 Zookeeper符合法则中哪两个?3.4 Zookeeper脑裂3.5 Zookeeper用来干嘛了 3.1 常用命令 ls、get、create、delete、deleteall3.2 选举机制 半数机制(过半机制࿰…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...
用机器学习破解新能源领域的“弃风”难题
音乐发烧友深有体会,玩音乐的本质就是玩电网。火电声音偏暖,水电偏冷,风电偏空旷。至于太阳能发的电,则略显朦胧和单薄。 不知你是否有感觉,近两年家里的音响声音越来越冷,听起来越来越单薄? —…...
Windows安装Miniconda
一、下载 https://www.anaconda.com/download/success 二、安装 三、配置镜像源 Anaconda/Miniconda pip 配置清华镜像源_anaconda配置清华源-CSDN博客 四、常用操作命令 Anaconda/Miniconda 基本操作命令_miniconda创建环境命令-CSDN博客...
Web中间件--tomcat学习
Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...

