开始补AtCoder的数学题了,练下思维

AB两道都很简单,看懂题就OK。

C,D稍微麻烦一些

Problem C: GCD On Blackboard

为了解决此问题,我们需要了解最大公约数(GCD)的某些属性。

第一点是,GCD是可交换的(GCD(a,b)= GCD(b,a))。

第二个是3个数字的GCD是关联的(GCD(a,GCD(b,c))= GCD(GCD(a,b),c))。

最后,在不失一般性的前提下,如果a和b是2个正数且a <= b,则GCD(a,b)<= a。 换句话说,2个数字的GCD最多等于较小的数字。

回到问题陈述,我们有一个数字列表,可以修改列表中的一个数字,然后用另一个数字替换它,以使列表的GCD最大化。 让我们将关于GCD的已知知识应用于此问题:

假设我们的列表 \(A = A_1,A_2,A_3,...,A_n\),我们希望修改索引 \(A_i\)。 根据关联规则,我们知道:GCD(A1,A2,A3,...,Ai,...,An)可以重新排序为:GCD(A1,A2,A3,...,An,Ai)。 换句话说,我们可以计算除 \(A_i\)以外的所有数字的GCD,然后使用\(A_i\)计算结果的GCD的结果。 令g为除\(A_i\)以外的所有数字的GCD。 现在我们希望最大化GCD(g,Ai)。 由于对\(A_i\)进行简化后我们可以获得的最大GCD为g,因此答案为g。

因此,我们知道在位置i处可获得的最大gcd是除\(A_i\)以外的所有数字的gcd。 如果我们在每个 \(i\) 处计算这些gcd值,则所有结果的最大值是修改列表中的数字后可能的最大GCD。

那么我们将如何实现该算法? 为方便起见,让我们定义\(R_i\)和\(L_i\)的两个功能,如下所示:

\[R_{n + 1} = 0\\R_i = GCD(R_{i + 1},A_i)\\L_0= 0\\L_i = GCD(L_{i - 1},A_i)
\]

\(L_i\)代表A1…Ai的gcd,\(R_i\)代表Ai…An的GCD。 如果我们希望找到除Ai以外的所有元素的GCD,则可以找到Mi = GCD(Li-1,Ri + 1)。 这表示除Ai以外的所有数字的GCD。

我们可以预先计算R和L。最后,我们可以找到Mi的每个值,然后计算出最大值,这就是我们的答案。 示例代码如下(复杂度为\(O(n)\)):

const int N = 100000 + 5;
int main() {
ios_base::sync_with_stdio(false), cin.tie(0);
// L_i = gcd(A_1, A_2, A_3, ... A_i)
// R_i = gcd(A_i, A_{i+1}, A_{i+1}, ..., A_n)
int n;
ll a[N], L[N], R[N];
cin >> n; for (int i = 1; i <= n; ++i) {
cin >> a[i];
L[i] = R[i] = a[i];
} // precomputing L[i]
for (int i = 1; i <= n; i++) {
L[i] = __gcd(L[i], L[i - 1]);
}
// precomputing R[i]
for (int i = n; i > 0; i--) {
R[i] = __gcd(R[i], R[i + 1]);
} // computing each m_i and the max value
int res = 0;
for (int i = 1; i <= n; i++) {
int m = __gcd(L[i - 1], R[i + 1]);
res = max(m, res);
} cout << res;
return 0;
}

Problem D: Flipping Signs

让我们考虑元素A1,A2,A3,…,An的列表。 我们被允许翻转任意两个连续元素Ai和Ai + 1的符号。 需要注意的一个有用观察结果是(除了最后一个元素),我们可以翻转任何Ai的符号而不会影响Ai-1。

但是我们如何利用这个事实呢? 好吧,我们知道我们可以使从i = 1到i = n-1的每个元素对于任何列表都是正的。 由于我们不在乎操作数量,因此这种贪婪的方法效果很好。

给定任何数字A列表,我们将能够使用此方法使除最后一个元素之外的所有Ai均为正。 最后一个元素将为负或正。 让我们分别处理这种情况。

所以现在我们可以将最后一个元素设为正数或负数。 如果为正,则所有元素均为正,列表的总和就是我们的答案。 如果它是负数,那么我们需要检查一些变化以获得最大的总和结果。 首先,我们需要检查最后两个元素中哪个更大。 由于我们可以使最后一个元素为正,而使An-1为负。 实际上,我们可以使任何一个Aj为负(j <n),以使最后一个元素为正。 考虑三个连续的正a,b,c元素。 我们首先可以翻转a和b的符号,这将使它们变为负数。 在下一步中,我们可以翻转b的符号,这将使b变为正,而c变为负。 这将使a变为负数,我们可以再次翻转c的符号使其变为正数,依此类推。 我们可以链接此操作,该操作将只留下a为负数,其余元素保持正值(最后一个元素除外,该元素将从负数变为正数)。

因此,我们有了算法最后一步的配方:我们检查范围A1…An-1中的最小元素。 将此最小元素称为m。 如果m <An,则将m的符号翻转为负,然后使An为正。 否则,我们将m保持为正。 这个清单的总和就是我们的答案。

Bellow是此解决方案的C ++实现。 注意,该解决方案与列表的其余部分分开处理An-1,An的情况,但是它与上述算法相同。

using ll = long long;
const int N = 1e5 + 10;
int main() {
ios_base::sync_with_stdio(false), cin.tie(0);
int n, A[N];
// store the sum of the result in a long long pair
ll s = 0;
cin >> n;
for (int i = 1; i <= n; ++i) cin >> A[i]; for (int i = 1; i <= n - 2; ++i) {
if (A[i] < 0) A[i] *= -1, A[i + 1] *= -1;
s += A[i];
} // if n is 2, we only need to consider those
// Also , if the last 2 elements are either both positive
// or both negative then we simply flip their signs too
if ((n == 2) or (A[n] > 0 and A[n - 1] > 0) or (A[n] < 0 and A[n - 1] < 0))
s += max(A[n - 1] + A[n], -A[n - 1] - A[n]);
else {
// find the min element in the list (excluding last 2 elements)
int* mni = min_element(A + 1, A + n - 2);
int mnv = *mni; // if the minimum value in the remaining list is larger than
// the smaller value of the last 2 elements, then we should
// just keep it positive
if (mnv > min(abs(A[n - 1]), abs(A[n]))) {
s += max(A[n - 1] + A[n], -A[n - 1] - A[n]);
}
// otherwise, we can make that value negative, and include the
// last 2 elements as positive elements
else {
s -= mnv * 2;
s += abs(A[n - 1]);
s += abs(A[n]);
}
}
cout << s << "\n"; return 0;
}

AtCoder | ABC 125 Person Editorial的更多相关文章

  1. ATCODER ABC 099

    ATCODER ABC 099 记录一下自己第一场AK的比赛吧...虽然还是被各种踩... 只能说ABC确实是比较容易. A 题目大意 给你一个数(1~1999),让你判断它是不是大于999. Sol ...

  2. Atcoder ABC 141

    Atcoder ABC 141 A - Weather Prediction SB题啊,不讲. #include<iostream> #include<cstdio> #inc ...

  3. Atcoder ABC 139E

    Atcoder ABC 139E 题意: n支球队大循环赛,每支队伍一天只能打一场,求最少几天能打完. 解法: 考虑抽象图论模型,既然一天只能打一场,那么就把每一支球队和它需要交手的球队连边. 求出拓 ...

  4. Atcoder ABC 139D

    Atcoder ABC 139D 解法: 等差数列求和公式,记得开 $ long long $ CODE: #include<iostream> #include<cstdio> ...

  5. Atcoder ABC 139C

    Atcoder ABC 139C 题意: 有 $ n $ 个正方形,选择一个起始位置,使得从这个位置向右的小于等于这个正方形的高度的数量最多. 解法: 简单递推. CODE: #include< ...

  6. Atcoder ABC 139B

    Atcoder ABC 139B 题意: 一开始有1个插口,你的插排有 $ a $ 个插口,你需要 $ b $ 个插口,问你最少需要多少个插排. 解法: 暴力模拟. CODE: #include< ...

  7. Atcoder ABC 139A

    Atcoder ABC 139A 题意: 给你两个字符串,记录对应位置字符相同的个数 $ (n=3) $ 解法: 暴力枚举. CODE: #include<iostream> #inclu ...

  8. atcoder abc 244

    atcoder abc 244 D - swap hats 给定两个 R,G,B 的排列 进行刚好 \(10^{18}\) 次操作,每一次选择两个交换 问最后能否相同 刚好 \(10^{18}\) 次 ...

  9. AtCoder ABC 250 总结

    AtCoder ABC 250 总结 总体 连续若干次一样的结果:30min 切前 4 题,剩下卡在 T5 这几次卡在 T5 都是一次比一次接近, 什么 dp 前缀和打挂,精度被卡,能水过的题连水法都 ...

  10. AtCoder ABC 242 题解

    AtCoder ABC 242 题解 A T-shirt 排名前 \(A\) 可得 T-shirt 排名 \([A+1,B]\) 中随机选 \(C\) 个得 T-shirt 给出排名 \(X\) ,求 ...

随机推荐

  1. MCU看门狗使用注意事项

    前言 最近因为项目产品硬件设计有问题,导致设计的一款产品把硬件电源开关以及硬件系统复位功能去掉了.更严重的是,这产品已经开始生产了,硬件已经无法修改,所以软件必须上看门狗,否则设备死机或是异常后就只能 ...

  2. C# 泛型编译特性对性能的影响

    C#作为一种强类型语言,具有丰富的泛型支持,允许开发者编写可以应对不同数据类型的通用代码.然而,在泛型编译时,针对结构和类作为泛型参数时,会对性能产生不同的影响. 泛型编译行为 在C#中,泛型编译行为 ...

  3. 0x02.加密和编码

    识别算法编码类型 看密文位数 看密文特征(数字.字母.大小写.符号等) 看当前密文存在的地方(web.数据库.操作系统等) 密码存储加密 md5:16位和32位由0-9和a-f组成的字符串 ,该加密方 ...

  4. 快速认识什么是:Docker

    Docker,一种可以将软件打包到容器中并在任何环境中可靠运行的工具.但什么是容器以及为什么需要容器呢?今天就来一起学快速入门一下Docker吧!希望本文对您有所帮助. 假设您使用 Cobol 构建了 ...

  5. Redis本地安装以及使用(详细教程)

    Redis 安装 Windows 下载安装 Redis默认端口:6379 整个过程如下: 1.下载连接 https://github.com/tporadowski/redis/releases Re ...

  6. WordPress简码实现的一些常用的效果

    首先要确保框架里已经安装好element pro插件,下面是使用简码,来实现效果,在element中找到简码,并且在WordPress后台主题编辑器中,找到function.php文件, 显示产品的分 ...

  7. Cloudeye对接Prometheus实现华为云全方位监控

    本文分享自华为云社区<Cloudeye对接Prometheus实现华为云全方位监控>,作者:可以交个朋友 . 一. 背景 云眼系统Cloudeye服务为我们提供了针对弹性云服务器.宽带等资 ...

  8. 一文带你深入理解K8s-Pod的意义和原理

    本文分享自华为云社区<深入理解K8s-Pod的意义和原理>,作者:breakDawn. 在Kubernetes概念中,有以下五种概念: 容器container:镜像管理的最小单位 生产任务 ...

  9. 记一次 .NET某收银软件 非托管泄露分析

    一:背景 1. 讲故事 在我的分析之旅中,遇到过很多程序的故障和杀毒软件扯上了关系,有杀毒软件导致的程序卡死,有杀毒软件导致的程序崩溃,这一篇又出现了一个杀毒软件导致的程序非托管内存泄露,真的是分析多 ...

  10. QRCoder1.4.3生成二维码,不依赖System.Drawing,解决"未能找到类型或命名空间名QRCode","及ImageFormatPng仅在windows上受支持"

    生成二维码1(简单) 包引用: <PackageReference Include="QRCoder" Version="1.4.3" /> usi ...