【贪心算法】贪心算法七
贪心算法七
- 1.整数替换
- 2.俄罗斯套娃信封问题
- 3.可被三整除的最大和
- 4.距离相等的条形码
- 5.重构字符串

点赞👍👍收藏🌟🌟关注💖💖
你的支持是对我最大的鼓励,我们一起努力吧!😃😃
1.整数替换
题目链接: 397. 整数替换
题目描述:

算法原理:
解法一:模拟(递归 + 记忆化搜索)
假设n = 18,我们要干的事情是把18变成1最小的步数。因为18是一个偶数只能除2变成9,拿到9这个数字,要干的其实也是一件相同的事情,要把9变成1最小的步数。
此时这里就出现了重复的子问题,大问题是18变成1的最小步数,18/2=9后就从了9变成1的最小步数的相同问题。因此我们可以把重复子问题拿到设计出函数头
int dfs(int n) 给一个整数n返回n变成1的最小步数。函数体 其实就是题目给的,如果n是偶数/2,如果n是奇数要么+1,要么-1我们求得是最小步数所以是 min(dfs(n-1),dfs(n+1)),递归出口 当 n == 1是之间返回0就行了。

在递归过程中发现大量重复,就可以用记忆化搜索,建一个数组,但是这道题的数据范围是1 <= n <= 2^31 - 1,我们要开这么大的空间肯定不行,因此搞一个hash<int,int> 第一个参数对应数字n,第二个参数对应这个数变成1的最小步数。

class Solution {unordered_map<int,int> hash;
public:int integerReplacement(int n) {return dfs(n);}// 递归int dfs(long long n) // 细节问题 数据范围1 <= n <= 2^31 - 1 加1会越界{if(n == 1){return 0;}if(n % 2 == 0) // 如果是偶数 {return 1 + dfs(n / 2);}else{return 1 + min(dfs(n - 1), dfs(n + 1));}}// 记忆化搜索int dfs(long long n){if(hash.count(n)){return hash[n];}if(n == 1){hash[1] = 0;return hash[1];}if(n % 2 == 0){hash[n] = 1 + dfs(n / 2);return hash[n];}else{hash[n] = 1 + min(dfs(n - 1), dfs(n + 1));return hash[n];}}
};
解法二:贪心
补充知识:二进制
- 偶数:二进制表示中最后一位是 0
- 奇数:二进制表示中最后一位是 1
- /2 操作:二进制表示中统一右移一位
我们这里研究的都是整数。
前两个可以自己举例看看。我们看最后一个

接下来想我们的贪心策略:
如果n是偶数没法贪,只能执行/2操作
是奇数就可以贪,要么执行+1,要么执行-1操作。
在模拟解法我们就是试试+1操作和-1操作看谁最小,但是如果在没有试之前就已经知道是+1好还是-1好,直接让奇数沿着较好的选择走,就可以舍去一个选择,那我们的时间复杂度会变得更优。
所以我们的贪心就是判断是+1好还是-1好。
如何判断?分情况讨论:
奇数的二进制最后一位是0,所以我们可以把奇数分为两大类
第一类:前面二进制位是 …,最后两个二进制位是 01
第二类:前面二进制位是 …,最后两个二进制位是 11
其中第一类我们默认 n > 1,也就是说 … 有1,如果没有1的话就是00…01了,直接返回即可。第二类默认 n > 3。

如果是 …01,接下来要么执行+1操作,要么执行-1操作。 +1操作会变成 …10,-1操作会变成 …00,那到底那个操作好呢? +1和-1操作都会变成偶数,偶数只能执行/2操作。假设…01是 …10001,执行+1操作会变成10010在执行/2操作会变成1001,执行-1操作会变成10000在执行/2操作会变成1000。这个时候就可以看出那个操作好了,肯定是-1操作好,因为1000我们可以一直/2操作尽快得到1,1001还需要在+1和-1操作在/2操作。
所以是奇数二进制最后两位是01,就执行-1操作,然后/2操作,会比较快得到1。

如果是 …11,接下来也是要么执行+1操作,要么执行-1操作,分析过程和上面一样。

但是n > 3这里有一个意外,当 n = 3的时候,我们需要特殊讨论,n = 3,二进制位前面都是0,后面虽然也是11。但是这里我们执行-1操作得到…10,然后在执行/2操作,直接就变成1了。这个和选择是不一样的。如果执行+1操作就会多一步/2操作。

我们这个贪心不用证明,分类讨论过程本身就是对这个贪心的证明。
那如何写代码呢?
如何判断二进制最后两位是01还是11呢?
拿n%4就可以了,因为n是奇数%4只能得到1和3,如果是1就是01情况,如果是3就是11情况。
class Solution {unordered_map<int,int> hash;
public:int integerReplacement(int n) {int ret = 0;while(n > 1){if(n % 2 == 0){n /= 2;++ret;}else{if(n == 3){ret += 2;n = 1;}else if(n % 4 == 1){n = n / 2;ret += 2;}else{n = n / 2 + 1;ret += 2;}}}return ret;}
};
2.俄罗斯套娃信封问题
题目链接: 354. 俄罗斯套娃信封问题
题目分析:

给一个二维数组,每一行表示信封的宽度和高度,当另一个信封的宽度和高度都比这个信封大的时候,这个信封就可以放进另一个信封里,如同俄罗斯套娃一样。最多能有多少个 信封能组成一组“俄罗斯套娃”信封(即可以把一个信封放到另一个信封里面)
算法原理:
解法一:常规解法(通用解法)-> 动态规划
先对数组排序,如果不排序的话,去找某一个信封能去套谁的时候,既要去它左边找找,也要去右边找找。说白了就是要变量数组一遍,才能去确定这个信封能去套谁。

但是如果我们把这个数组按照左端点排序后,此时在去确定一个信封能去套谁的时候,仅需去左边看看就行了。因为如果能套必须满足左大于左,右大于右,我们已经按左端点排好序,右边的左都比当前的左大,因此不用考虑右边。

此时我们的最长套娃序列特别像之前的最长递增子序列问题,最长递增子序列是在原始的数组中挑选一些数出来形成递增的序列,问最长的长度是多少。我们这里是在一些信封里面挑一些信封出来使它能形成一个套娃序列,问最长的套娃序列是多少。这个问题就和最长递增子序列一模一样。无非最长子序列是在一个个数中挑,我们这里是在一个个信封里面挑。

1.状态表示
dp[i] 表示:以 i 位置的信封为结尾的所有套娃序列中,最长套娃序列的长度
2.状态转移方程
根据最近一步,划分情况
i 位置本身就是套娃序列 ,长度是1
以 i 位置为结尾的最长套娃长度,那么就去 0 ~ i - 1 这段区间遍历一遍找到一个j位置,只要发现 i 信封 能套到 j 信封 外面,那此时用dp[j]在加上 i 这一个信封就是以 i 位置的信封为结尾套娃序列的长度。找到 0 ~ i -1 区间所有能套的 dp[j] + 1 的最大值,就是 i 位置的信封为结尾的所有套娃序列中,最长套娃序列的长度。
3初始化
数组初始为1,就可以不用考虑为的1情况

4.填表顺序
从左往右
5.返回值
dp[i] 表示:以 i 位置的信封为结尾的所有套娃序列中,最长套娃序列的长度。我们要的是整个区间最长套娃序列的长度,所以返回dp表中的最大值。
class Solution {
public:int maxEnvelopes(vector<vector<int>>& e) {// 动态规划sort(e.begin(), e.end());int n = e.size();vector<int> dp(n, 1);int ret = 1;for(int i = 1; i < n; ++i){for(int j = 0 ; j < i; ++j){if(e[j][0] < e[i][0] && e[j][1] < e[i][1])dp[i] = max(dp[i], dp[j] + 1);}ret = max(ret, dp[i]);}return ret;}
};
因为这道题的数据量太大我们的动规会超时,但是动规是解决这类题的常规方法。这道题不行不代表相同类型的题不行,比如1263. 推箱子这道题用动规是可以通过的。
解法二:重写排序 + 贪心 + 二分
动态超时了,肯定得用贪心 + 二分了,但是为什么多一个重写排序呢?
如果我们仅仅只是按照左端点排序,接下来用贪心和二分,你会发现我们要分类讨论,原因就是之前研究的最长递增子序列只有一个限定条件 只在一堆数中去挑,然后贪心保留是长度为1,长度为2 … 的最后一个元素的最小值,比如长度为2:5,现在来了一个3,我们可以把5干掉,保留3,原因是能跟在5的后面更能跟在3的后面。但是我们这道题给的是一个个区间,有的时候我们并不能直接删除,有的时候是把之前结果保留而把新来的给删除。虽然能做但是需要分类讨论比较麻烦。所以我们先重写排序在贪心和二分。
下面我们给的是左端点都是不同的排完序后的样子,如果左端点不一样,我们其实可以把左端点删去。原因就是左端点都不一样,我们还按照左端点从小到大排好序了,那就相当于前面的是严格递增,所以不考虑前边信封左端点是多少。那不就变成了在 3、1、8、12、3中挑一个最长递增子序列了嘛。

但是我们这里是可能有重复的左端点的,假设有重复的话,我们排完序是下面这种情况,如果我们继续不看左端点,我们可能会挑出来4、6、7、9长度为4的序列,但是这并不符合,原因就是我们的左必须大于左才能套。

那如何避免这种情况呢?很简单当左端点相同的时候,我们就按照右端点从大到小排序。当继续不看左端点,我们在挑7的时候绝对不要9,因为当左边相同的时候右边是按照从大到小排的,同理挑4绝对不会考虑前三个。

重写排序后就完完全全变成只有一个限制的最长递增序列了。

class Solution {
public:int maxEnvelopes(vector<vector<int>>& e) {// 重写排序 + 贪心 + 二分sort(e.begin(), e.end(),[&](vector<int>& e1, vector<int>& e2){return e1[0] != e2[0] ? e1[0] < e2[0] : e1[1] > e2[1];});// 贪心 + 二分vector<int> ret;ret.push_back(e[0][1]);int n = e.size();for(int i = 1; i < n; ++i){if(e[i][1] > ret.back()){ret.push_back(e[i][1]);}else{int left = 0, right = ret.size() - 1;while(left < right){int mid = (left + right) >> 1;if(ret[mid] < e[i][1])left = mid + 1;elseright = mid;}ret[left] = e[i][1];}}return ret.size();}
};
3.可被三整除的最大和
题目链接: 1262. 可被三整除的最大和
题目分析:

这道题的意思是给一个数组,在这个数组中挑一些数使这些数的和能被3整除,并且这些挑的这些数的和是最大的。
算法原理:
解法一:正难则反 + 贪心 + 分类讨论
题目要在数组中挑一些数的和能被3整除并且和是最大的,我们可以直接把整个数组中的数全部挑出来,%3正好等于0,那我就不用考虑了,如果%3不等于,那我在全部挑选的基础上删一些数就可以了。
先把所有的数累加在一起,根据累加和,删除一些数
假设所有数的和是sum,接下来就分类讨论:
- sum % 3 == 0
直接返回sum

- sum % 3 == 1
我们定义一些数,x :标记 % 3 == 1 的数,y : 标记 % 3 == 2 的数
如果%3 == 1,必定有下面几种情况:
第一种情况:
-
存在一个%3 == 1 的数,剩下所有数的和%3 == 0
-
存在四个%3 == 1 的数,剩下所有数的和%3 == 0,其实可以把三个%3 == 1的数归到剩下所有数的和%3 == 0里面
-
存在七个%3 == 1 的数,剩下所有数的和%3 == 0,也可以把六个%3 == 1的数归到剩下所有数的和%3 == 0里面

剩下的也不用枚举了,我们只用考虑第一种情况就行了,原因就是不管这里有多少种情况,我们仅需删去第一种情况的x1就可以让剩下的数的和%3==0。

那么贪心的地方来了,要删除怎样的x1,肯定是最小的x1。因为我们想让sum的和最大。

第二种情况:
存在两个%3 == 2 的数 y1,y2 (2 + 2 = 4 % 3 = 1),或者和上面一样存在y1、y2、y3、y4…,但是和上面一样我们仅需第一种情况就行了,此时删y1,y2。此时贪的地方来了,为了使sum最大,删y1和y2,一个是最小的,一个是次小的。

因为sum %3 == 1 会分为这两种情况,因此我们取这两种情况的最大值。

- sum % 3 == 2
也是两种情况,要么存在一个y1 使sum % 3 == 2,要么存在一个x1,一个x2 使 sum % 3 == 2。我们依旧取两种情况的最大值。

如何求一堆数中的最小值以及次小值?
同理求一堆数中的最大值和次大值也是一样的求法。
第一种方法:sort排序 O(nlogn)
第二种方法:分类讨论
先定义两个数x1、x2,然后初始化为无穷大。然后从左到右遍历根据新来的x,分类讨论:
- x < x1

- x1 < x < x2

3. x > x2
并不影响最小和次小的。所以不用考虑
class Solution {
public:int maxSumDivThree(vector<int>& nums) {const int INF = 0x3f3f3f3f;int sum = 0, x1 = INF, x2 = INF, y1 = INF, y2 = INF;for(auto x: nums){sum += x;if(x % 3 == 1){if(x < x1) x2 = x1, x1 = x;else if(x < x2) x2 = x;}else if(x % 3 == 2){if(x < y1) y2 = y1, y1 = x;else if(x < y2) y2 = x;} }//分类讨论if(sum % 3 == 0) return sum;else if(sum % 3 == 1) return max(sum - x1, sum - y1 - y2);else return max(sum - y1, sum - x1 - x2);}
};
解法二:动态规划
从一堆数中选取一些数,使这些数的和能被3整除。其实这道题就是一个01背包问题。
1.状态表示
dp[i][j] 表示:从前 i 个数中选取一些数,这些数的和模3等于 j (0 <= j < 3) 时,最大值的和是多少
2.状态转移方程
根据最后一个位置,划分情况
不选 i 位置这个数,那就去 0 ~ i - 1 区间去选一些数的和模3等于j 时最大值的和,正好是 dp[i-1][j]

选 i 之后,还是去 0 ~ i - 1 区间去选一些数的和模3,但是此时就不是直接去找和模3等于 j 的了,因为 i 位置这个数 nums[i] % 3 会等于0、1、2 中的任意一个数,那么去 0 ~ i - 1区间去找和模3等于 的 j 也要随 nums[i] % 3 的改变而去改变。
以nums[i]%3 == 1为例,如果求的dp[i][0],那就要去 0 ~ i -1 区间去找和%3 但是 j 为 2 的情况,因为这个和 加上 nums[i] 才会有 和 % 3 等于 0,比如 nums[i] 是 1,去 0 ~ i - 1找的和是2,2 % 3 = 2, (2 + 1)% 3 == 0 。同理求dp[i][1],dp[i][2]也是一样。

我们要求的是最大和,因此取两种情况的最大值

3.初始化
- 多开一行,列开3个
- 里面的值要保证后序的填表是正确的
- 下标的映射关系
第一行为空表示没有数可以选,此时和%3等于0,不选就行了,直接就是0,第一格填0,没有数可以选还要和%3 == 1 和 2 是不可能存在的,可以给这两个位置得值位-1表示不存在得情况,但是我们下面写代码要判断一下 不等于-1 才能要这个状态,但是因为我们求得又是最大值,我们可以使它俩足够小就行了这样就可以不用去判断了,也不会影响填表。因此可以给-0x3f3f3f3f。
第一列除了第一格下面的不用初始化。直接放在填表里面就行了。

4.填表顺序
从上往下,从左往右
5.返回值
dp[i][j] 表示:从前 i 个数中选取一些数,这些数的和模3等于 j (0 <= j < 3) 时,最大值的和是多少,我们要得使从所有数种选一些数,这些数得和%3 == 0 最大值是多少,因此返回的是 dp[n][0]
class Solution {
public:int maxSumDivThree(vector<int>& nums) {// 动态规划const int INF = 0x3f3f3f3f;int n = nums.size();vector<vector<int>> dp(n + 1, vector<int>(3, -INF));dp[0][0] = 0;for(int i = 1; i <= n; ++i)for(int j = 0; j < 3; ++j)dp[i][j] = max(dp[i - 1][j], dp[i - 1][(j - nums[i -1] % 3 + 3) % 3] + nums[i - 1]);return dp[n][0];}
};
优化:利用滚动数组做优化
背包问题哪里我们使用的是一个数组充当滚动数组,但是这里我们要用两个数组充当滚动数组,因为%3可能会等于0、1、2,那在一个数组中更新下一行的值及其可能会覆盖j 为 0 、1、2的任何位置,比如 填 j = 0,会用到 j = 1, 但是填 j 也可能会用到 j = 0,但是前面已经把 j = 0 更新了找不到之前的值了,所以这里我们用两个数组充当滚动数组,就不担心这个问题了。并且如果是两个数组充当滚动数组,01背包优化填表顺序从左往右,从右往左都行。
class Solution {
public:int maxSumDivThree(vector<int>& nums) {// 利用滚动数组优化(二个数组)const int INF = 0x3f3f3f3f;int n = nums.size();vector<int> dp(3, -INF);dp[0] = 0;for(int i = 1; i <= n; ++i){vector<int> ndp(3);for(int j = 0; j < 3; ++j)ndp[j] = max(dp[j], dp[(j - nums[i -1] % 3 + 3) % 3] + nums[i - 1]);dp = move(ndp);}return dp[0];}
};
4.距离相等的条形码
题目链接: 1054. 距离相等的条形码
题目分析:

算法原理:
解法:贪心 + 模拟
我们这道题就是让我们把这些数重新排列一下,相邻的两个不相同。此时我们这样考虑问题,我们有9个格子,把给的数放到格子里让相邻的两个不相同就可以了。此时我们可以这样处理,把相同的数看出一批数,每次摆放一批数,摆放的时候仅需让这些相同的数不相邻就可以了。如何做到不相邻特别简单,每次摆的时候隔一个格子。这样绝对会让相同的数不相邻。

我们先在偶数位上摆,摆完后在摆奇数位。此时摆完后会发现相邻两个数是不相同的。
贪心策略:
- 每次处理一批相同的数
- 摆放的时候,每次隔一个格子
但是这个策略还有一个问题,可能会有把相同的数放在相邻的位置,这里我们还要多加一个限定条件。
- 先处理出现次数最多的那个数,剩下的数的处理顺序无所谓

证明:
题目一定有解,我们可以得到一个性质,假设有n个数,我们可以分成 (n+1)/2 个组。如果题目一定有解,我们可以证明的是:出现次数最多的那个数,不超过(n+1)/2个。
假设出现次数最多的那个数,超过(n+1)/2个。那此时去摆这些数的时候必定会有一组里面出现相同的数。但是题目一定有解,因此出现次数最多的那个数,不超过(n+1)/2个。

我们的策略是先处理出现次数最多的那个数,剩下的数的处理顺序无所谓。
第一种情况:出现次数最多的数,正好出现 (n + 1)/2。我们先处理最多的那个数,剩下的数无论怎么放都不会相邻。

第二种情况:出现次数最多的数,小于(n + 1)/ 2。此时我们也可以证明相同的数不相邻,因为如果相邻必定是后面的数x出现次数还要比o还要多,但是这种情况绝对不会存在。因为我们的前提就是出现次数最多的数,小于(n + 1)/ 2,那就是x就是出现次数最多的数。但是我们是优先处理出现次数最多的那次数。所以如果是先填o,x绝对不可能相邻。

class Solution {
public:vector<int> rearrangeBarcodes(vector<int>& b) {// 统计每个数出现的频次unordered_map<int,int> hash;int maxVal = 0, maxCount = 0;for(auto x : b){if(maxCount < ++hash[x]){maxCount = hash[x];maxVal = x;}}// 先处理出现次数最多的那个数int n = b.size();vector<int> ret(n);int index = 0;for(int i = 0; i < maxCount; ++i){ret[index] = maxVal;index += 2; }//处理剩下的数hash.erase(maxVal);for(auto& [x, y] : hash){for(int i = 0; i < y; ++i){if(index >= n) index = 1;ret[index] = x;index += 2;}}return ret;}
};
5.重构字符串
题目链接: 767. 重构字符串
题目分析:

算法原理:
解法:贪心 + 模拟
- 每次处理一批相同的字符
- 摆放的时候,隔一个位置放一个字符
- 优先处理出现次数最多的那一个字符
这道题并没有告诉我们一定会有排序,所以我们要先判断一下是否有排列,方法很简单,出现次数最多的字符个数不超过(n+1)/2就行了。
class Solution {
public:string reorganizeString(string s) {// 统计每个字符出现的频次int hash[26] = { 0 };char maxChar = ' '; int maxCount = 0;for(auto ch : s){if(maxCount < ++hash[ch - 'a']){maxCount = hash[ch - 'a'];maxChar = ch;}}// 先判断⼀下int n = s.size();if(maxCount > ((n + 1) / 2)) return "";// 先处理出现次数最多的那个字符string ret(n,' ');int index = 0;for(int i = 0; i < maxCount; ++i){ret[index] = maxChar;index += 2;}//处理剩下的数hash[maxChar - 'a'] = 0;for(int i = 0; i < 26; ++i){for(int j = 0; j < hash[i]; ++j){if(index >= n) index = 1;ret[index] = 'a' + i;index += 2;}} return ret;}
};
相关文章:
【贪心算法】贪心算法七
贪心算法七 1.整数替换2.俄罗斯套娃信封问题3.可被三整除的最大和4.距离相等的条形码5.重构字符串 点赞👍👍收藏🌟🌟关注💖💖 你的支持是对我最大的鼓励,我们一起努力吧!😃…...
LangChain教程 - 表达式语言 (LCEL) -构建智能链
系列文章索引 LangChain教程 - 系列文章 LangChain提供了一种灵活且强大的表达式语言 (LangChain Expression Language, LCEL),用于创建复杂的逻辑链。通过将不同的可运行对象组合起来,LCEL可以实现顺序链、嵌套链、并行链、路由以及动态构建等高级功能…...
使用Locust对Redis进行负载测试
1.安装环境 安装redis brew install redis 开启redis服务 brew services start redis 停止redis服务 brew services stop redis 安装Python库 pip install locust redis 2.编写脚本 loadTest.py # codingutf-8 import json import random import time import redis …...
HIVE数据仓库分层
1:为什么要分层 大多数情况下,我们完成的数据体系却是依赖复杂、层级混乱的。在不知不觉的情况下,我们可能会做出一套表依赖结构混乱,甚至出现循环依赖的数据体系。 我们需要一套行之有效的数据组织和管理方法来让我们的数据体系…...
数据结构与算法之动态规划: LeetCode 2407. 最长递增子序列 II (Ts版)
最长递增子序列 II https://leetcode.cn/problems/longest-increasing-subsequence-ii/description/ 描述 给你一个整数数组 nums 和一个整数 k找到 nums 中满足以下要求的最长子序列: 子序列 严格递增子序列中相邻元素的差值 不超过 k请你返回满足上述要求的 最…...
电子电气架构 --- 什么是自动驾驶技术中的域控制单元(DCU)?
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 所谓鸡汤,要么蛊惑你认命,要么怂恿你拼命,但都是回避问题的根源,以现象替代逻辑,以情绪代替思考,把消极接受现实的懦弱,伪装成乐观面对不幸的…...
html5css3
1.html5新增语义化标签 <header><nav><article><section><aside><footer> 2.新增多媒体标签 视频<video>格式:map4,webm,ogg <video controls"controls" autoplay"autoplay" muted"mute…...
FPGA多路红外相机视频拼接输出,提供2套工程源码和技术支持
目录 1、前言工程概述免责声明 2、相关方案推荐我已有的所有工程源码总目录----方便你快速找到自己喜欢的项目我这里已有的红外相机图像处理解决方案本博已有的已有的FPGA视频拼接叠加融合方案 3、工程详细设计方案工程设计原理框图红外相机FDMA多路视频拼接算法FDMA图像缓存视…...
python实战(十二)——如何进行新词发现?
一、概念 新词发现是NLP的一个重要任务,旨在从大量的文本数据中自动识别和提取出未在词典中出现的新词或短语,这对于信息检索、文本挖掘、机器翻译等应用具有重要意义,因为新词往往包含了最新的知识和信息。 随着互联网的不断发展,…...
动手做计算机网络仿真实验入门学习
打开软件 work1 添加串行接口模块,先关电源,添加之后再开电源 自动选择连接 所有传输介质 自动连接 串行线 绿色是通的,红色是不通的。 显示接口。se是serial串行的简写。 Fa是fast ethernet的简写。 为计算机配置ip地址: 为服…...
完整的 FFmpeg 命令使用教程
FFmpeg 是一个开源的跨平台音视频处理工具,它能够处理几乎所有的视频、音频格式,并提供了强大的功能如格式转换、视频剪辑、合并、提取音频等。FFmpeg 通过命令行界面(CLI)操作,尽管有一些图形界面的前端工具ÿ…...
Leetcode 3405. Count the Number of Arrays with K Matching Adjacent Elements
Leetcode 3405. Count the Number of Arrays with K Matching Adjacent Elements 1. 解题思路2. 代码实现 题目链接:3405. Count the Number of Arrays with K Matching Adjacent Elements 1. 解题思路 这一题虽然是一道hard的题目,但是委实是有点名不…...
Springboot(五十六)SpringBoot3集成SkyWalking
这里我们将skywalking集成到Springboot中。 关于docker部署skyWalking的相关问题,请移步《docker(二十八)docker-compose部署链路追踪SkyWalking》 一:下载java-agents 先放一下skyWalking的官网下载地址 Downloads | Apache SkyWalking 其他的版本的 APM 地址(这个我不需…...
有没有免费提取音频的软件?音频编辑软件介绍!
出于工作和生活娱乐等原因,有时候我们需要把音频单独提取出来(比如歌曲伴奏、人声清唱等、乐器独奏等)。要提取音频必须借助音频处理软件,那么有没有免费提取音频的软件呢?下面我们将为大家介绍几款免费软件࿰…...
Linux 中查看内存使用情况全攻略
Linux 中查看内存使用情况全攻略 在 Linux 系统运维与开发工作里,精准掌握内存使用状况对系统性能优化、故障排查起着举足轻重的作用。Linux 提供了多款实用工具来查看内存详情,下面我们就结合实际示例,深入了解这些工具的使用方法。 一、fr…...
【SQL Server】教材数据库(3)
接着教材数据库(1)的内容,完成下列查询。 1 查询订购高等教育出版社教材的学生姓名 2 查询比所有高等教育出版社的图书都贵的图书信息 3 列出每位学生姓名、订购教材书名、价格。 1、嵌套查询:use jiaocai select student.nam…...
使用 ECharts 与 Vue 构建数据可视化组件
在前端开发中,数据可视化是非常重要的一部分。ECharts 作为一个功能强大且易于使用的开源数据可视化库,被广泛应用于各种图表展示需求中。而 Vue.js 是当下流行的前端框架之一,它的数据驱动和组件化开发模式让我们能轻松地将 ECharts 集成到 …...
Yocto 项目 - 共享状态缓存 (Shared State Cache) 机制
引言 在嵌入式开发中,构建效率直接影响项目的开发进度和质量。Yocto 项目通过其核心工具 BitBake 提供了灵活而强大的构建能力。然而,OpenEmbedded 构建系统的传统设计是从头开始构建所有内容(Build from Scratch),这…...
Unity3D仿星露谷物语开发9之创建农场Scene
1、目标 绘制农场的场景。通过不同Sorting Layer控制物体的显示优先级,绘制Tilemap地图,添加Tilemap Collider碰撞器,同时添加Composite Collider碰撞器优化性能。 ps:绘制Tilemap的技巧:通过"Shift [" 可…...
STM32-笔记20-测量按键按下时间
1、按键按下的时间-思路 我们先检测下降沿信号,检测到以后,在回调函数里切换成检测上升沿信号,当两个信号都检测到的时候,这段时间就是按键按下的时间,如图所示:>N*(ARR1)CCRx的值 N是在这段时间内&…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
unix/linux,sudo,其发展历程详细时间线、由来、历史背景
sudo 的诞生和演化,本身就是一部 Unix/Linux 系统管理哲学变迁的微缩史。来,让我们拨开时间的迷雾,一同探寻 sudo 那波澜壮阔(也颇为实用主义)的发展历程。 历史背景:su的时代与困境 ( 20 世纪 70 年代 - 80 年代初) 在 sudo 出现之前,Unix 系统管理员和需要特权操作的…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
Mysql8 忘记密码重置,以及问题解决
1.使用免密登录 找到配置MySQL文件,我的文件路径是/etc/mysql/my.cnf,有的人的是/etc/mysql/mysql.cnf 在里最后加入 skip-grant-tables重启MySQL服务 service mysql restartShutting down MySQL… SUCCESS! Starting MySQL… SUCCESS! 重启成功 2.登…...
Python+ZeroMQ实战:智能车辆状态监控与模拟模式自动切换
目录 关键点 技术实现1 技术实现2 摘要: 本文将介绍如何利用Python和ZeroMQ消息队列构建一个智能车辆状态监控系统。系统能够根据时间策略自动切换驾驶模式(自动驾驶、人工驾驶、远程驾驶、主动安全),并通过实时消息推送更新车…...
django blank 与 null的区别
1.blank blank控制表单验证时是否允许字段为空 2.null null控制数据库层面是否为空 但是,要注意以下几点: Django的表单验证与null无关:null参数控制的是数据库层面字段是否可以为NULL,而blank参数控制的是Django表单验证时字…...
【Veristand】Veristand环境安装教程-Linux RT / Windows
首先声明,此教程是针对Simulink编译模型并导入Veristand中编写的,同时需要注意的是老用户编译可能用的是Veristand Model Framework,那个是历史版本,且NI不会再维护,新版本编译支持为VeriStand Model Generation Suppo…...
前端高频面试题2:浏览器/计算机网络
本专栏相关链接 前端高频面试题1:HTML/CSS 前端高频面试题2:浏览器/计算机网络 前端高频面试题3:JavaScript 1.什么是强缓存、协商缓存? 强缓存: 当浏览器请求资源时,首先检查本地缓存是否命中。如果命…...
[拓扑优化] 1.概述
常见的拓扑优化方法有:均匀化法、变密度法、渐进结构优化法、水平集法、移动可变形组件法等。 常见的数值计算方法有:有限元法、有限差分法、边界元法、离散元法、无网格法、扩展有限元法、等几何分析等。 将上述数值计算方法与拓扑优化方法结合&#…...
