2-6基础算法-快速幂/倍增/构造

news2025/1/24 8:22:40

文章目录

  • 一.快速幂
  • 二.倍增
  • 三.构造

一.快速幂

快速幂算法是一种高效计算幂ab的方法,特别是当b非常大时。它基于幂运算的性质,将幂运算分解成一系列的平方操作,以此减少乘法的次数。算法的核心在于将指数b表示为二进制形式,并利用二进制位来决定何时将当前的底数的幂乘入结果中。

普通的快速幂算法

#include <iostream>
#include <algorithm>
using namespace std;
long long fastPower(long long base, long long power) {
	long long result = 1;
	while (power > 0) {
		if (power % 2 == 1) {
			result *= base;
		}
		base *= base;
		power /= 2;
	}
	return result;
}
int main() {
	long long base, power;
	cin >> base >> power;
	cout << fastPower(base, power);
}

为了适用于大数运算,提高效率,引入了位运算和模运算
位运算:比乘除法和模运算更快。
模运算:在每次乘法后进行,避免了大整数的处理和溢出的问题。

#include <iostream>
#include <algorithm>
using namespace std;
#define mod 998244353  //固定数
long long fastPower(long long base, long long power) {
	long long result = 1;
	while (power > 0) {
		if (power&1) { //power % 2 == 1
			result *= base% mod;
		}
		base *= base% mod;
		power >>= 1;//右移一位,相当于÷2,即power /= 2;
	}
	return result;
}
int main() {
	long long base, power;
	cin >> base >> power;
	cout << fastPower(base, power);
}

[例] 快速幂
在这里插入图片描述
评测系统

#include <iostream>
#include <algorithm>
using namespace std;
long long fastPower(long long base, long long power, long long k) {
	long long result = 1;
	while (power > 0) {
		if (power & 1) {
			result = result * base % k;
		}
		base = base * base % k;
		power >>= 1;
	}
	return result;
}
int main() {
	long long b, p, k;
	cin >> b >> p >> k;
	long long s = fastPower(b, p, k);
	cout << s;
}

二.倍增

倍增算法是一种用于解决各种问题的通用技术,例如在计算斐波那契数列、找到图中节点的祖先或者处理区间查询问题(如最小值或最大值)时。其基本思想是通过重复倍增一个数或者序列的长度,从而以对数时间解决问题。例如,在求解LCA(最近公共祖先)问题时,倍增算法可以在预处理阶段通过倍增跳跃来快速移动到任意节点的不同层级的祖先节点。

[例1] 最近公共祖先LCA查询

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
评测系统

分析:
(1)预处理
①使用DFS从根节点开始变量,计算每个结点的所有祖先
②使用数组parent[node][i]存储结点node的第2i个祖先,parent[node][0]就是结点node的父节点,parent[node][1]就是node的第2个祖先
③递推式parent[node][i]=parent[parent[node][i-1]][i-1] //两个i-1就是i
(2)查询
为了找到两个结点u和v的最低公共祖先,若u的深度≥v,我们将u的深度提升至与v相同,若提升后的u与v是同一结点,则该结点就是他们的最低公共祖先,否则同时提升二者的深度,直到他们有相同的第2i个祖先为止

例如查找第1248个祖先,1248二进制表示为10011100000
即1248=210+27+26+25
即从node向上跳210步,再跳27,再跳26,再跳25
即任意一个数都可以通过向上跳2i来表示

#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
const int MAXN = 1e5 + 5;//节点数量的上限
const int MAXLOG = 20;//最大深度,2^20>1e5
vector<int> tree[MAXN];//tree[i]存储了所有直接连接到节点i的子节点的列表,即树的邻接表表示
int parent[MAXN][MAXLOG];
int depth[MAXN];//每个节点在树中的深度
void dfs(int node, int prev) { //遍历树并填充 parent 和 depth 数组
	for (int i = 1; i < MAXLOG; i++) {
		parent[node][i] = parent[parent[node][i - 1]][i - 1];
	}
	for (int child : tree[node]) {  //递归处理子节点
		if (child != prev) {
			depth[child] = depth[node] + 1;
			parent[child][0] = node;
			dfs(child, node);
		}
	}
}
int lca(int u, int v) {
	if (depth[u] < depth[v]) swap(u, v);//确保u更深
	for (int i = MAXLOG - 1; i >= 0; i--) { //向上提升u
		if (depth[u] - (1 << i) >= depth[v]) { //1<<i相当于2的i次幂
			u = parent[u][i];
		}
	}
	if (u == v)
		return u;
	for (int i = MAXLOG - 1; i >= 0; i--) {
		if (parent[u][i] != parent[v][i]) {
			u = parent[u][i];
			v = parent[v][i];
		}
	}
	return parent[u][0];
}
int main() {
	int N;
	cin >> N;
	for (int i = 1; i < N; i++) {
		int u, v;
		cin >> u >> v;
		tree[u].push_back(v);
		tree[v].push_back(u);
	}
	dfs(1, -1);
	int Q;
	cin >> Q;
	while (Q--) {
		int a, b;
		cin >> a >> b;
		cout << lca(a, b) << endl;
	}
	return 0;
}

[例2] 如今仍是遥远的理想之城1

在这里插入图片描述
在这里插入图片描述
对于小数据

#include <iostream>
#include <algorithm>
using namespace std;
const int N = 2e5 + 5;
int main() {
	int n, k;
	cin >> n >> k;
	long long a[N];
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	long long position = 1;
	while(k--) {
		position = a[position];
	}
	cout << position;
}

对于大数据
需要移除重复的循环
例如
第 1 次传送:从 1 到 2
第 2 次传送:从 2 到 3
第 3 次传送:从 3 到 4
第 4 次传送:从 4 到 2
在第 4 次传送时,我们回到了传送阵 2,这里形成了一个循环:2 -> 3 -> 4 -> 2,这个循环会一直重复,循环长度为3
可以看出,当k足够大时,若pre=1说明一定没有节点能跳转到当前结点。若pre=2,当k足够大时一定有能跳转过来的,也就一定能出现循环

#include <iostream>
#include <algorithm>
using namespace std;
const int N = 2e5 + 5;
int main() {
	long long n, k;
	cin >> n >> k;
	long long a[N], pre[N] = { 0 };//pre[position]表示访问到position时总的传送次数【这里必须初始化为0】
	for (int i = 1; i <= n; i++) {
		cin >> a[i];
	}
	long long position = 1;
	long long num = 1;//num表示总传送次数
	while (k--) {
		position = a[position];
		if (pre[position])
			k = k % (num - pre[position]);//num-1-pre[position]表示这个循环的长度
		pre[position] = num++;

	}
	cout << position;
}

注:局部变量不会自动初始化为零,全局变量会初始化为0。在main函数内long long a[N], pre[N]时必须初始化为0,或者直接在全局定义。

[例3] 数的变换
在这里插入图片描述
在这里插入图片描述

分析:F[i][j]表示A初始为i时,经过2j次变换的值,其中20=1次变换表示进行一次变化,即输出A/B+C;21=2表示两次变换,即将A/B+C作为新的A,放入A/B+C中,输出结果

例如:
A = 10, B = 2, C = 1, D = 5
F[i][0]表示A=i时,即i/b+c的值,通过遍历i的所有可能,以此作为第0列的值,即初始化过程。由于A不断变为A/B+C,所以A的最大取值为max(1e5,A/B+C的大小)=max(1e5,2e5)=2e5,即F的最大行数
以下是第0列的初始化过程

for (int i = 0; i <= 200000; i++) {  //初始化
		F[i][0] = i / b + c;
}

然后我们按列更新
更新F[0][1]时,表示A=0经过21=2次变换的结果,具体的变化过程为A=0→A=1(一次变换)→A=1(两次变换)
在这里插入图片描述

更新F[3][1]同样

在这里插入图片描述

即对第一列所有的更新,我们都是把F[i][0]作为新的行,找到第0列对应的值(即一次变换)

for (int i = 0; i <= 200000; i++) {
	F[i][1] = F[F[i][0]],0];
}

更新F[18][2]时,从A=18起经过了4次变换,得到了3

在这里插入图片描述
显然,我们从F[18][2]的左侧F[18][1]=6开始,进行两次变换(标号3和4)更快得到了答案
更显然的,A=6进行两次变化,可以直接从表格中的F[6][1]得到(注意1是21
我们据此更新第二列,以当前行前一列的值(6)作为行,直接找到该行(6这一行)前一列(第1列)的值(3),即为结果

for (int i = 0; i <= 200000; i++) {
	F[i][2] = F[F[i][1]],1];
}

对于第j列的更新,按照同样的逻辑,我们有

for (int i = 0; i <= 200000; i++) {
	F[i][j] = F[F[i][j-1]],j-1];
}

再遍历所有的j,我们有

for (int j = 1; j <= 30; j++) {
	for (int i = 0; i <= 200000; i++) {
		F[i][j] = F[F[i][j - 1]][j - 1];
	}
}

j表示进行2j次变换,230>1e9=maxQ(查询次数的最大值)

以上过程是倍增的重要思想

当A=3时,1左移3位就是1000,视为二进制,对应的十进制是8,即23,可视为F[i][j]的j=3
当A=8时,1左移8位就是100000000,视为二进制,对应的十进制是256,即28,可视为F[i][j]的j=8
可以看出,A的值=1左移位数=j的值
而j的范围很小,是从0到30,我们遍历j就相当于遍历了A
1左移j位得到的这个二进制数中只会有一个1,我们将查询次数Q也写为二进制,二者相与,若结果为1,说明两个二进制数一定完全相同,即此时Q的二进制中也只有一个1,而这个1所在的位置可以通过1<<j表示
因此,通过遍历j,当相与为1时,我们可以直接在F数组中找到F[A][j]

for (int j = 0; j <= 30; j++) {
	if ((1 << j) & q) {
		answer = F[A][j];
		break;
	}
}

因此,本题的求解代码为

#include <iostream>
#include <algorithm>
using namespace std;
const int N = 2e5 + 5, M = 30 + 5;
int main() {
	long long a, b, c, q;
	long long F[N][M];
	cin >> a >> b >> c >> q;
	if (b == 1) { //特殊情况
		cout << a + q * c;
		return 0;
	}
	for (int i = 0; i <= 200000; i++) {
		F[i][0] = i / b + c;
	}
	for (int j = 1; j <= 30; j++) {
		for (int i = 0; i <= 200000; i++) {
			F[i][j] = F[F[i][j - 1]][j - 1];
		}
	}
	for (int j = 0; j <= 30; j++) {
		if ((1 << j) & q) {
			a = F[a][j];
			break;
		}
	}
	cout << a;
}

三.构造

通过观察问题的结构和规律,找到一种通用的方法或模式,使得在问题规模增大时,依然能够高效地得到答案。

1.小浩的ABC
在这里插入图片描述
在这里插入图片描述
评测系统

分析:
A*B+C>=2,若X<2,无解
若X<=106+1,则B=C=1,A=X-1。此时字典序一定是最小的
若X>106+1,为了让B更小,应该A尽可能大,即A=106
若A、C都是106,那么X一定是106的整数倍,即有X%106=0
若X不是106的整数倍,说明C一定不是106,设X=1e7+2
A=1e6,B=9,C=1e6,不满足
A=1e6,B=10,C=2,满足,显然B=X/A,C=X%A

#include <iostream>
#include <algorithm>
using namespace std;
int main() {
	long long T, A, B, C;
	cin >> T;
	while (T--) {
		long long X;
		cin >> X;
		if (X < 2) {
			cout << "-1" << endl;
		}
		else {
			if (X <= 1e6 + 1) {
				B = 1, C = 1, A = X - 1;
			}
			else {
				A = 1e6;
				if (X % 1000000 == 0) {
					C = 1e6, B = (X - C) / A;
				}
				else {
					B = X / 1000000;
					C = X % 1000000;
				}
			}
			cout << A << " " << B << " " << C << endl;
		}
	}
}

2.小新的质数序列挑战
在这里插入图片描述
在这里插入图片描述
评测系统

分析:质数:2、3、5、7、11
A和B任一个小于2都失败
每一个偶数都能由多个2组成,每一个奇数都能由1个3和多个2组成
因此输出只能为0或1
若AB都是偶数,输出一定为0
若AB相等,输出一定为0
其他情况需要具体分析
如一奇一偶
27和24,由9个3和8个3组成,输出为0
3和5,由3和2 3组成,输出为1
所以可以查看AB(大于1的)公因数,若有,则输出为0,否则为1
此方法也涵盖了都是偶数和两数相等的情况

#include <iostream>
#include <algorithm>
using namespace std;
long long f(long long a, long long b) { //求最大公因数
	while (b) {
		long long temp = b;
		b = a % b;
		a = temp;
	}
	return a;
}
int main() {
	int T;
	cin >> T;
	long long A, B;
	while (T--) {
		cin >> A >> B;
		if (A < 2 || B < 2) {
			cout << "-1" << endl;
		}
		else {
			if (f(A, B) > 1) {
				cout << "0" << endl;
			}
			else
				cout << "1" << endl;
		}
	}
}

3.小蓝找答案

在这里插入图片描述
在这里插入图片描述
评测系统

【本题分析与注释仅供参考】

分析:
①假设3个字符串的长度分别为1,2,3
s1:a
s2长度为2,aa满足
s3长度为3,aaa满足
因此长度|si|<长度|si+1|,显然没有产生进位,那么si就是si+1的前缀。所以si+1是由si向后补字符集中最小的字母,直到长度等于si+1

②假设3个字符串的长度分别为3,2,1
s1:aaa
s2长度为2,将s1的长度变为2(即截尾),得到aa,显然不满足,最后一位+1得到ab,满足
s3长度为1,将s2的长度变为1(截尾),得到a,显然不满足,最后一位+1得到b,满足
因此长度|si|≥长度|si+1|,一定产生了进位,具体来说是si截尾后的最后一位加1得到的。

③假设3个字符串的长度分别为3,2,2(这里k=2)
s1:aaa
s2长度为2,将s1的长度变为2(即截尾),得到aa,显然不满足,最后一位+1得到ab,满足
s3长度为2,将s2的长度变为2(不变),得到ab,显然不满足,最后一位+1得到了ac,这时c>字符集中最小的字母+1,需要再次进位(即进位到前一位),ac变为ba

④假设s2为abc(这里k=3)
s3长度为3,将s2的长度变为3(不变),得到abc,显然不满足,最后一位+1得到abd,此时d>字符集中最小的字母+2,需要再次进位(即进位到前一位),abd变为aca
因此,当某一位>字符集中最小的字母+k-1时,就产生了进位
或者更通用的,当某一位+1后>字符集中最小的字母+k时,就产生了进位

⑤假设4个字符串的长度分别为3,1,2,2
s1:aaa
s2长度为1,将s1的长度变为1(截尾),得到a,显然不满足,最后一位+1得到b,满足
s3长度为2,将s2作为前缀,后面补字符集中最小的字母至与s3等长,得到bb,满足
s4长度为2,将s2的长度变为2(不变),得到bb,显然不满足,最后一位+1得到了bc;这时c>字符集中最小的字母+1,需要再次进位(即进位到前一位),bc变为ca;这时c>字符集中最小的字母+1,需要再次进位(即进位到前一位),ca变为aaa,超出长度。即第一位产生进位就是不合法。

我们假设字符集的大小为k,使用二分去找最小的合法字符集的大小,并验证二分出来的k是否合法

最终代码

#include <iostream>
#include <algorithm>
using namespace std;
const int N = 2e5 + 5;
int s[N];
int n;
struct node {
    int pos;//字符串长度
    int num;//长度为pos的字符串产生的进位次数
}stk[N];//栈
int top;//栈顶指针
void insert(int v, int mid) {//处理进位
    while (top > 1 && stk[top].pos > v)//截尾。或者说当前长度的进位信息我们当前不需要考虑,可以移除
        top--;
    if (stk[top].pos == v)//如果二者长度相等
        stk[top].num++;//直接复制过来,最后一位+1。即产生进位次数+1
    else//如果长度不相等,新信息压入栈
        stk[++top] = { v,1 };//进位次数初始化为1
    if (top > 1 && stk[top].num == mid)//若某个长度的字符串的进位次数≥mid,需要向前进位。如aa/ab两次进位==mid=2,此时需要向前进位变成ba
        insert(v - 1, mid);
}
int check(int mid) {//验证给定的字符集大小mid是否足够用来构造出满足条件的字符串序列
    top = 1;
    stk[top] = { 0,0 };
    for (int i = 1; i < n; i++) {
        if (s[i + 1] <= s[i]) //一定有进位
            insert(s[i + 1], mid);
    }
    return stk[1].num;//返回第一个位置上的进位,为0表示没有进位,即合法
}
int main() {
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> s[i];
    }
    int num = 0;
    for (int i = 1; i < n; i++) {
        if (s[i + 1] > s[i])
            num++;
    }
    if (num == n - 1) { //如果从第二个起每一个都能通过"补齐"实现,则字符集中只有a
        cout << "1";
        return 0;
    }
    //补不了,使用二分找k
    int left = 2, right = 2e5 + 5;
    int ans = 0;
    while (left + 1 != right) {
        int mid = (left + right) / 2;
        if (check(mid) == 0)
            right = mid;
        else
            left = mid;
    }
    if (check(left) == 0)
        cout << left;
    else
        cout << right;
}

一些具体的分析:
①假设我们有一个字符串长度序列 [4, 3, 2, 1],并且我们正在按序处理这些字符串的长度。我们将关注栈中的变化,特别是在 top-- 过程中栈的状态。

while (top > 1 && stk[top].pos > v)//当前长度更小
	top--;

处理长度4:
假设栈初始为 [{0, 0}]。
长度4是序列中的第一个元素,因此不需要进位操作。栈状态不变。

处理长度3:
当我们开始处理长度3的字符串时,由于它小于4,我们需要在长度3上进行一次进位操作。
栈变为 [{0, 0}, {3, 1}]。

处理长度2:
现在我们处理长度2的字符串。在这里,长度为3的进位记录(即 {3, 1})不再相关,因为我们现在关注的是长度2。
执行 top–,栈中长度为3的记录被移除,栈变回 [{0, 0}]。
然后,我们在长度2上进行进位操作,栈变为 [{0, 0}, {2, 1}]。

处理长度1:
接下来是长度1的字符串。同样,长度为2的记录(即 {2, 1})现在不再相关。
执行 top–,移除长度为2的记录,栈变回 [{0, 0}]。
在长度1上进行
进位操作,栈变为 [{0, 0}, {1, 1}]。

在 top-- 的过程中,栈中存储的是之前处理的不同长度字符串的进位记录。每当我们转向处理一个更短的字符串时,所有比这个新长度长的字符串的进位记录都变得不再相关,因此需要被移除。这些进位记录反映了在之前长度上的字符集使用情况和进位情况。

②假设有一个字符串长度序列 [3, 2, 2, 1],我们考虑如何处理这些长度并记录进位信息:

if (stk[top].pos == v)//如果二者长度相等
	stk[top].num++;

处理长度3: 不需要进位,假设栈状态是 [{0, 0}]。
处理长度2: 第一次出现长度2,进行一次进位操作。栈状态变为 [{0, 0}, {2, 1}]。
再次处理长度2: 当再次遇到长度2时,栈顶元素已经是 {2, 1}。这时,这行代码 stk[top].num++ 会执行,表示在长度2上又发生了一次进位。栈状态更新为[{0, 0}, {2, 2}],意味着在长度为2的字符串上发生了两次进位操作。

③当字符串长度再次变为之前处理过的长度时,确保 num 是累加的
示例分析
考虑序列 [3, 2, 2, 1, 2, 2]:
处理长度 3,不涉及进位操作。
处理第一个长度 2,进行一次进位操作,栈变为 [{2, 1}]。
处理第二个长度 2,再次进位,栈更新为 [{2, 2}]。
处理长度 1,由于这比 2 小,栈清除长度 2 的记录,变回 [{0, 0}],然后为 1 添加进位记录,变为 [{1, 1}]。
再次处理长度 2,由于栈中没有长度 2 的记录,会添加一个新的进位记录,栈变为 [{1, 1}, {2, 1}]。
处理另一个长度 2,找到栈中的 {2, 1}记录,然后将其num增加,栈更新为[{1, 1}, {2, 2}]。
虽然top–的过程删除了部分数据,但num的更新过程是基于当前字符串长度更新的,也能实现累加的效果。

4.小蓝的无限集
在这里插入图片描述
在这里插入图片描述

分析:
若第一次计算1*a,得到的通式为n=ax+by (其中xy为任意满足要求的整数)
若第一次计算x+b,得到的通式为n=(1+b)* ax+by=b*(ax+y)+ax
我们可以令x=0,若b%(n-1)==0,则n一定在无限集中
上述条件不满足且a=1,则n一定不在无限集中
此外,根据通式有
n-ax=by
n-ax=b*(ax+y)
我们可以遍历x,计算n-ax,若b能够将其整除,则n一定在无限集中

#include <iostream>
#include <algorithm>
using namespace std;
int main() {
    int t;
    cin >> t;
    int a, b, n;
    while (t--) {
        cin >> a >> b >> n;
        if ((n - 1) % b == 0) {
            cout << "Yes" << endl;
            continue;
        }
        else if (a == 1) {
            cout << "No" << endl;
            continue;
        }
        long long res = 1;
        bool flag = 0;
        while (res < n) {
            res *= a;
            if (res > n)
                break;
            else if (res == n) {
                cout << "Yes" << endl;
                flag = 1;
                break;
            }
            else if ((n - res) % b == 0) {
                cout << "Yes" << endl;
                flag = 1;
                break;
            }
        }
        if (flag == 1)
            continue;
        cout << "No" << endl;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1383677.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机毕业设计----Springboot+Vue调查问卷管理系统

基于Spring Boot的问卷调查系统 项目介绍 > * 本项目的在线问卷调查调查系统是基于SpringBoot开发的&#xff0c;采用了前后端分离模式来开发。 > * 前端开发使用了Vue、Element UI&#xff0c;后端的开发则是使用了SpringBoot、MyBatis技术。 项目配置 > * 下载…

docker部署开原博客系统halo

1.先看下效果 2.拉取镜像 docker pull halohub/halo:2.10 3.启动 mkdir -p /data/halo && cd /data/halodocker run -it -d --name halo -p 8090:8090 -v /data/halo/.halo2:/root/.halo2 halohub/halo:2.10 4.设置账户密码 设置后登陆 5.登陆 发布博客

【Docker】Dockerfile构建最小镜像

&#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Docker的相关操作吧 目录 &#x1f973;&#x1f973;Welcome 的Huihuis Code World ! !&#x1f973;&#x1f973; 前言 一.Dockerfile是什么 二.Dock…

【Qt】QThread moveTothread-多线程的两种实现方法

一、如何理解多线程 二、实现多线程的两种方式&#xff08;面向应用&#xff09; 2.1 继承 QThread 的类 2.2 (推荐这种方式)函数 moveTothread() 三、多线程的释放问题&#xff08;善后工作&#xff09; 多线程的两种实现方法 一、如何理解多线程二、实现多线程的两种方式&…

C++核心编程之类和对象---C++面向对象的三大特性--多态

目录 一、多态 1. 多态的概念 2.多态的分类&#xff1a; 1. 静态多态&#xff1a; 2. 动态多态&#xff1a; 3.静态多态和动态多态的区别&#xff1a; 4.动态多态需要满足的条件&#xff1a; 4.1重写的概念&#xff1a; 4.2动态多态的调用&#xff1a; 二、多态 三、多…

快速排序学习笔记

代码框架 // 在数组nums将下标从left到right中进行从小到大排序// 原理是先将一个元素排好序&#xff0c;然后将其他的元素排好序void sort(int[] nums, int left, int right) {if (left > right) {return;}// 对数组nums[left,right]进行切分&#xff0c;使得nums[left,p-1…

Pandoc:markdown转word

简介&#xff1a;Pandoc是由John MacFarlane开发的标记语言转换工具&#xff0c;可实现不同标记语言间的格式转换&#xff0c;堪称该领域中的“瑞士军刀”。Pandoc使用Haskell语言编写&#xff0c;以命令行形式实现与用户的交互&#xff0c;可支持多种操作系统&#xff1b;Pand…

A connection was successfully established with the server but then an error

在使用EFCore生成数据库的时候&#xff0c;报上面的错误&#xff01; 解决方法&#xff1a; 加&#xff08;EncryptTrue;TrustServerCertificateTrue;&#xff09;即可&#xff1a; "ConnectionStrings": { "DefaultConnection": "Data SourceLAP…

大厂咋做支付系统的核对?

核对是保障资金安全的重要机制。 时效角度&#xff0c;主要有&#xff1a; &#xff08;准&#xff09;实时核对 准确性不如离线核对&#xff0c;且需相应实时核对平台建设 离线核对&#xff08;如 T1 核对&#xff09; 主要问题是发现问题的时机较为后置&#xff0c;部分场景…

软件测试|教你使用Python下载图片

前言 我一直觉得Windows系统默认的桌面背景不好看&#xff0c;但是自己又没有好的资源可以进行替换&#xff0c;突然我一个朋友提醒了我&#xff0c;网络上的图片这么多&#xff0c;你甚至可以每天换很多个好看的背景&#xff0c;但是如果让我手动去设置的话&#xff0c;我觉得…

Consider defining a bean of type ‘XXXX‘ in your configuration.

今天学习尚硅谷的SpringCloud时&#xff0c;发现支付模块无法启动&#xff0c;控制台输出下面的错误: 看起来可能是dao层没有被注入。 然后根据我已有的知识&#xff0c;我检查了注解Mapper Mapper public interface PaymentDao {public int create(Payment payment);public…

Nacos下载与安装【Linux】

&#x1f95a;今日鸡汤&#x1f95a; 我不是天生的王者&#xff0c;但我骨子里流动着不让我低头的血液。 ——《海贼王》 目录 &#x1f32d;1.官网下载 &#x1f37f;2.根据需求下载Linux版本 &#x1f953;3.上传到Linux &#x1f9c2;4.解压Nacos &#x1f9c8; 5.…

C++力扣题目701--二叉搜索树中的插入操作

给定二叉搜索树&#xff08;BST&#xff09;的根节点 root 和要插入树中的值 value &#xff0c;将值插入二叉搜索树。 返回插入后二叉搜索树的根节点。 输入数据 保证 &#xff0c;新值和原始二叉搜索树中的任意节点值都不同。 注意&#xff0c;可能存在多种有效的插入方式&a…

温和去油去黑头,只需敷上一刻钟的泥膜就够了

冬季天气干燥&#xff0c;很多朋友脸部更容易出油&#xff0c;连带着黑头也变多了。这是因为干燥的环境会刺激皮脂腺分泌&#xff0c;导致皮肤油脂分泌过多&#xff0c;容易堵塞毛孔形成黑头。因此&#xff0c;在冬季特别需要注意控油去黑头的工作。 控油去黑头需要清洁毛孔&am…

Sentinel微服务保护

文章目录 Sentinel微服务保护1.初识Sentinel1.1.雪崩问题及解决方案1.1.1.雪崩问题1.1.2.解决方案1.1.3.总结 1.2.服务保护技术对比1.3.Sentinel介绍和安装1.3.1.初识Sentinel1.3.2.安装Sentinel 1.4.微服务整合Sentinel 2.流量控制2.1.簇点链路2.1.快速入门2.2.流控模式2.2.1.…

docker应用:vocechat

简介&#xff1a;VoceChat是一款超轻量级的Rust聊天应用程序、API和SDK&#xff0c;优先考虑私人托管。使用VoceChat建立您自己的聊天功能&#xff01;作为一款非常好用的通讯应用程序&#xff0c;它可以让你与朋友、家人和同事进行即时消息聊天&#xff0c;支持图片视频的分享…

Python相对导入和绝对导入

目录结构&#xff1a; 在 en_de_model_CDDD.py 文件有两种导入方式可以导入utils.py&#xff0c;分别是 相对导入&#xff1a; from ...public_utils.utils import canonicalize_smiles 绝对导入&#xff1a; from public_utils.utils import canonicalize_smiles 这里推荐使…

Fluent 动网格应用:2.5D 网格重构

1 概述 2.5D 网格重构是一种快速网格重构方法&#xff0c;主要应用于涡旋压缩机等存在复杂平面运动且无法简化为二维计算的问题。 涡旋压缩机工作原理&#xff08;视频源&#xff1a;维基百科&#xff09; 适用于 2.5D 动网格的问题特点&#xff1a; 计算域几何形状为柱体类形…

WorkPlus企业内部即时通信新选择,打造高效协作新格局

在企业内部&#xff0c;快速、高效的沟通与协作是推动工作进程的关键。而即时通信工具成为了企业内部沟通的重要工具。作为一款优秀的企业内部即时通信工具&#xff0c;WorkPlus通过其出色的性能和独特的功能&#xff0c;为企业打造高效协作的新格局。 为什么选择WorkPlus作为企…

算法训练营Day43(背包问题)

1049. 最后一块石头的重量 II 1049. 最后一块石头的重量 II - 力扣&#xff08;LeetCode&#xff09; class Solution {public int lastStoneWeightII(int[] stones) {int sum 0;for(int num:stones){sumnum;}int target sum/2;//1 dp数组&#xff1a;背包容量为j&#xf…