题目描述
小蓝最近正在玩一款 RPG 游戏。
他的角色一共有 N 个可以加攻击力的技能。
其中第 i 个技能首次升级可以提升 Ai 点攻击力,以后每次升级增加的点数都会减少 Bi。
⌈Ai / Bi⌉(上取整)次之后,再升级该技能将不会改变攻击力。
现在小蓝可以总计升级 M 次技能,他可以任意选择升级的技能和次数。
请你计算小蓝最多可以提高多少点攻击力?
输入格式
输入第一行包含两个整数 N 和 M。
以下 N 行每行包含两个整数 Ai 和 Bi。
输出格式
输出一行包含一个整数表示答案。
数据范围
对于 40% 的评测用例,1 ≤ N,M ≤ 1000;
对于 60% 的评测用例,1 ≤ N ≤ 104,1 ≤ M ≤ 107;
对于所有评测用例,1 ≤ N ≤ 105,1 ≤ M ≤ 2×109,1 ≤ Ai,Bi ≤ 106。
输入样例
3 6
10 5
9 2
8 1
输出样例
47
具体实现
#include
using namespace std;
typedef long long LL;
const int N = 100010;
int n, m;
int a[N], b[N];
//a[N]表示每个技能首次可以提高的攻击力
//b[N]表示升级后减少的攻击力
//判断结果是否合法
bool check(int mid)
{
LL res = 0;
for (int i = 0; i < n; i ++ )
{
//前提条件
if (a[i] >= mid)
{
//求和公式
res += (a[i] - mid) / b[i] + 1;
}
}
//判断是否合法
return res >= m;
}
int main()
{
//输入数据
cin >> n >> m;
for (int i = 0; i < n; i ++ )
{
cin >> a[i] >> b[i];
}
//数组的边界是0到1e6,因为对于所有测评用例升级技能所得攻击力最大为1e6
int l = 0, r = 1e6;
//二分查找
while (l < r)
{
int mid = l + r + 1 >> 1;
if (check(mid))
{
l = mid;
}
else
{
r = mid - 1;
}
}
LL res = 0, cnt = 0;
//res表示总和
//cnt表示一共加多少项
for (int i = 0; i < n; i ++ )
{
if (a[i] >= r)
{
int c = (a[i] - r) / b[i] + 1; //首项
int end = a[i] - (c - 1) * b[i]; //末项
cnt += c;
res += (LL)(a[i] + end) * c / 2;
}
}
cout << res - (cnt - m) * r << endl; //果有重复的话,会多出来,既然已经限制了次数那么就要减去
return 0;
}
题目描述
一个未经初始化的变量,里面存储的值可能是任意的。
因此直接使用未初始化的变量,比如将其赋值给另一个变量,并不符合一般的编程逻辑。
代码中出现这种情况,往往是因为遗漏了初始化语句、或是打错了变量名。
对代码中使用了未初始化变量的语句进行检查,可以方便地排查出代码中的一些隐秘 Bug。
考虑一段包含 k 条赋值语句的简单代码。
该段代码最多使用到 n 个变量,分别记作 a1,a2,⋯,an;该段代码使用的常量均记作 a0。
第 i 条(1 ≤ i ≤ k)赋值语句为 axi = ayi,满足 1 ≤ xi ≤ n、0 ≤ yi ≤ n,表示将 ayi 的值赋给变量 axi。
其中 axi 被称为该赋值语句的左值,一定是个变量;ayi 被称为右值,可以是一个常量或变量。
对于任意一条赋值语句 axi = ayi,如果右值 ayi 是一个变量,则其应该在此之前被初始化过。
具体来说,如果变量 ayi 在前 i−1 条赋值语句中做为左值出现过,即存在 j < i 满足 xj = yi(这里无需考虑第 j 条赋值语句本身是否也有右值未初始化的问题),我们就认为在第 i 条赋值语句中 ayi 已被初始化;否则,我们认为该条语句存在右值未初始化的问题。
按照上述规则,试统计给定的代码中,有多少条赋值语句右值未被初始化。
输入格式
输入的第一行包含空格分隔的两个正整数 n、k,分别表示变量的数量和赋值语句的条数。
接下来输入 k 行,其中第 i 行(1 ≤ i ≤ k)包含空格分隔的两个正整数 xi、yi,表示第 i 条赋值语句。
纠错:其实 yi 可以等于 0,所以 yi 不一定是正整数(可参考样例),此为官网题面描述存在问题,特此指出。
输出格式
输出一个整数,表示有右值未被初始化问题的赋值语句条数。
数据范围
50% 的测试数据满足 0 < n,k ≤ 1000;
全部的测试数据满足 0 < n,k ≤ 105,1 ≤ xi ≤ n,0 ≤ yi ≤ n。
输入样例
10 7
1 2
3 3
3 0
3 3
6 2
2 1
8 2
输出样例
3
样例解释
其中第一、二、五条赋值语句右值未被初始化。
具体实现
#include
using namespace std;
const int N = 100010;
int n, m;
bool st[N];
int main()
{
cin >> n >> m;
st[0] = true;
int res = 0;
while (m -- )
{
int x, y;
cin >> x >> y;
if (!st[y])
{
res ++ ;
}
st[x] = true;
}
cout << res << endl;
return 0;
}
题目描述
输入格式
第一行包含一个整数 n,表示待处理的整数个数。
第二行包含空格分隔的 n 个整数,依次表示 a1,a2,⋯,an。
输出格式
输出共 n 行,每行一个浮点数,依次表示按上述方法归一化处理后的数据 f ( a 1 ) f(a_{1}) f(a1), f ( a 2 ) f(a_{2}) f(a2),⋯, f ( a n ) f(a_{n}) f(an)。
如果你输出的每个浮点数与参考结果相比,均满足绝对误差不大于 10−4,则该测试点满分,否则不得分。
数据范围
全部的测试数据保证 n,|ai| ≤ 1000,其中 |ai| 表示 ai 的绝对值。
且输入的 n 个整数 a1,a2,⋯,an 满足:方差 D ( a ) D(a) D(a) ≥ 1。
输入样例
7
-4 293 0 -22 12 654 1000
输出样例
-0.7485510379073613
0.04504284674812264
-0.7378629047806881
-0.7966476369773906
-0.7057985054006686
1.0096468614303775
1.9341703768876082
样例解释
平均值: a ˉ \begin{aligned}\bar{a}\end{aligned} aˉ ≈ 276.14285714285717
方差: D ( a ) \begin{aligned}D(a)\end{aligned} D(a) ≈ 140060.69387755104
标准差: D ( a ) \begin{aligned}\sqrt{D(a)}\end{aligned} D(a) ≈ 374.24683549437134
具体实现
#include
using namespace std;
const int N = 1010;
int n;
int w[N];
int main()
{
cin >> n;
int sum = 0;
for (int i = 0; i < n; i ++ )
{
cin >> w[i];
sum += w[i];
}
//强制类型转换
//求这组数据的平均值
double avg = (double)sum / n;
double d = 0;
for (int i = 0; i < n; i ++ )
{
//pow()函数用来求x的y次幂
//求这组数据的方差
d += pow(w[i] - avg, 2) / n;
}
//求这组数据的标准差
d = sqrt(d);
for (int i = 0; i < n; i ++ )
{
cout << (w[i] - avg) / d << endl;
}
return 0;
}
题目描述
输入格式
输入共两行。
第一行包含用空格分隔的两个整数 n 和 m,分别表示题目数量和顿顿老师的神秘数字。
第二行包含用空格分隔的 n 个整数 a1,a2,⋯,an,依次表示每道选择题的选项数目。
输出格式
输出一行包含一个整数表示答案。输出仅一行,包含用空格分隔的 n 个整数 b1,b2,⋯,bn,依次表示每道选择题的正确选项。
数据范围
50% 的测试数据满足:ai 全部等于 2,即每道题均只有两个选项,此时 ci = 2i;
全部的测试数据满足:1 ≤ n ≤ 20,ai ≥ 2 且 cn ≤ 109(根据题目描述中的定义 cn 表示全部 ai 的乘积)。
输入样例 1
15 32767
2 2 2 2 2 2 2 2 2 2 2 2 2 2 2
输出样例 1
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
输入样例 2
4 0
2 3 2 5
输出样例 2
0 0 0 0
输入样例 3
7 23333
3 5 20 10 4 3 10
输出样例 3
2 2 15 7 3 1 0
样例 3 解释
i | 1 | 2 | 3 | 4 | 5 | 6 | 7 |
---|---|---|---|---|---|---|---|
ai | 3 | 5 | 20 | 10 | 4 | 3 | 10 |
bi | 2 | 2 | 15 | 7 | 3 | 1 | 0 |
ci-1 | 1 | 3 | 15 | 300 | 3000 | 12000 | 36000 |
提示
具体实现
#include
using namespace std;
const int N = 25;
int n, m;
int a[N];
int main()
{
cin >> n >> m;
for (int i = 1; i <= n; i ++ )
{
cin >> a[i];
}
for (int i = 1; i <= n; i ++ )
{
cout << m % a[i] << ' ';
m /= a[i];
}
return 0;
}
题目描述
新学期伊始,适逢顿顿书城有购书满 x 元包邮的活动,小 P 同学欣然前往准备买些参考书。
一番浏览后,小 P 初步筛选出 n 本书加入购物车中,其中第 i 本(1 ≤ i ≤ n)的价格为 ai 元。
考虑到预算有限,在最终付款前小 P 决定再从购物车中删去几本书(也可以不删),使得剩余图书的价格总和 m 在满足包邮条件(m ≥ x)的前提下最小。
试帮助小 P 计算,最终选购哪些书可以在凑够 x 元包邮的前提下花费最小?
输入格式
输入的第一行包含空格分隔的两个正整数 n 和 x,分别表示购物车中图书数量和包邮条件。
接下来输入 n 行,其中第 i 行(1 ≤ i ≤ n)仅包含一个正整数 ai,表示购物车中第 i 本书的价格。
输入数据保证 n 本书的价格总和不小于 x。
输出格式
仅输出一个正整数,表示在满足包邮条件下的最小花费。
数据范围
70% 的测试数据满足:n ≤ 15;
全部的测试数据满足:n ≤ 30,每本书的价格 ai ≤ 104 且 x ≤ a1+a2+⋯+an。
输入样例 1
4 100
20
90
60
60
输出样例 1
110
样例 1 解释
购买前两本书 (20+90) 即可包邮且花费最小。
输入样例 2
3 30
15
40
30
输出样例 2
30
样例 2 解释
仅购买第三本书恰好可以满足包邮条件。
输入样例 3
2 90
50
50
输出样例 3
100
样例 3 解释
必须全部购买才能包邮。
具体实现
#include
using namespace std;
const int N = 33, M = 300010;
int n, x;
int w[N], f[M];
//w[N]表示每本书的价钱
//f[N]表示背包问题的数组
int main()
{
cin >> n >> x;
int sum = 0;
for (int i = 0; i < n; i ++ )
{
cin >> w[i];
sum += w[i];
}
//背包的容量
int m = sum - x;
//第一重循环枚举每一个物品
//第二重循环倒序枚举体积
for (int i = 0; i < n; i ++ )
{
for (int j = m; j >= w[i]; j -- )
{
f[j] = max(f[j], f[j - w[i]] + w[i]);
}
}
cout << sum - f[m] << endl;
return 0;
}