WHUT第五周训练整理
写在前面的话:我的能力也有限,错误是在所难免的!因此如发现错误还请指出一同学习!
索引
(难度由题目自身难度与本周做题情况进行分类,仅供新生参考!)
一、easy:02、05、07、14、15、17
二、medium:04、06、08、09、10、12、13、16
三、hard:01、11、18
四、乱入的二分图:03
本题解报告大部分使用的是C++语言,在必要的地方使用C语言解释。
声明:本周题目的难度较之间几周的题目有所提升,这里我只给出我自己的解法,肯定还有其他可能更优的解法,有兴趣的同学可以多了解了解。
一、easy
1002:Hanoi Tower Troubles Again!(打表)
题意:有 根柱子,有无数个小球(编号从 开始)可以套在柱子上,对于一个空的柱子可以放任意的小球,对于一个非空的柱子,如果当前的小球的编号 + 柱子顶部小球的编号 = 平方数,那么可以放在上面,否则只能套在新的空柱子上。现在问 根柱子最多可以套多少个小球?
分析:发现 很小,通过打表发现 的 最多只能处理 个小球,因此我这里就对于每个 都简单遍历 模拟一下 根柱子能处理的最多的小球就可以了。
Code:
const int MAXN = 50 + 10;
int arr[MAXN];
int main()
{
int T, n;
cin >> T;
while (T--)
{
cin >> n;
int cnt = 0;
for (int i = 1; i <= 1300; i++) // 简单遍历
{
int j;
for (j = 0; j < cnt; j++) // 在已有的柱子中尝试套上当前的小球
{
int now = arr[j] + i;
int tmp = sqrt(now);
if (tmp * tmp == now) // 是平方数则套上
{
arr[j] = i;
break;
}
}
if (j == cnt) // 所有柱子都套不上则另起一个柱子
{
arr[cnt++] = i;
}
if (cnt == n + 1) // 如果用的柱子超过了n则输出答案
{
cout << i - 1 << endl;
break;
}
}
}
return 0;
}
1005:Can you solve this equation?(数学+二分)
题意:给定 ,求 在 中的解。
分析:设 ,则 。
当 时,满足 ,即在 中 单调递增,使用二分法求解即可。
Notice:注意精度问题!不论是二分还是三分,尽量输出 值而不是 或者 。
Code:
const double eps = 1e-5;
double f(double x)
{
return 8 * pow(x, 4) + 7 * pow(x, 3) + 2 * pow(x, 2) + 3 * x + 6;
}
double solve(double y)
{
double l = 0, r = 100;
while (fabs((f(l) - y) - (f(r) - y)) >= eps)
{
double mid = (l + r) / 2;
if (f(mid) < y)
{
l = mid;
}
else
{
r = mid;
}
}
return l;
}
int main()
{
int T;
cin >> T;
while (T--)
{
double y;
cin >> y;
if (f(0) > y || f(100) < y) // 如果低于最小值或者大于最大值,则无解
{
cout << "No solution!" << endl;
}
else
{
cout << fixed << setprecision(4) << solve(y) << endl;
}
}
return 0;
}
1007:Strange fuction(数学+三分)
题意:给定 ,求函数 在 上的最小值。
分析:求导
再求导 得 当 。
故 在 上单调递增,又因 ,因此 的值由负到正。
因此函数 先减后增,是在 上的凹函数,上三分即可。
Code:
const double eps = 1e-6;
double y;
double f(double x) // 函数值
{
return 6 * pow(x, 7) + 8 * pow(x, 6) + 7 * pow(x, 3) + 5 * x * x - y * x;
}
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> y;
double l = 0, r = 100;
while (r-l >= eps) // 三分确定答案
{
double lm = (r + l) / 2;
double rm = (lm + r) / 2;
if (f(lm) < f(rm)) // 左边<右边,在凹函数中舍弃右边
{
r = rm;
}
else // 否则舍弃左边
{
l = lm;
}
}
cout << fixed << setprecision(4) << f(l) << endl;
}
return 0;
}
1014:Starship Hakodate-maru(暴力枚举)
题意:有 吨的货物需要装入容器,现在有两种容器,第一种只能装进 吨的货物,第二种只能装进 吨的货物,现在求利用这两种容器最多能装入多少货物?注意不能超过 。
分析: 最多是 ,直接双重循环枚举 更新答案即可。详见代码。
Code:
int main()
{
int n;
while (cin >> n, n)
{
int ans = 0;
for (int i = 0; i * i * i <= n; i++) // 枚举i
{
for (int j = 0; j * (j + 1) * (j + 2) / 6 <= n; j++) // 枚举j
{
int sum = i * i * i + j * (j + 1) * (j + 2) / 6; // 当前容器装入的货物吨数
if (sum <= n) // 保证不超过n
{
ans = max(ans, sum); // 更新答案
}
}
}
cout << ans << endl; // 输出
}
return 0;
}
1015:Pseudoprime numbers(素数判断+快速幂)
题意:给定 以及 ,现在问 是否是基于 的伪素数。
分析:首先 本身不能是素数,其次要满足 。比较基础比较套路,数据范围比较大, ,首先判断素数可以在 内完成,而求 模 的结果可以用快速幂 完成。详见代码。
Code:
long long poww(int a, int n, int mod) // 快速幂,取模
{
long long ans = 1;
long long pingfang = a;
while (n != 0)
{
if (n % 2 == 1)
{
ans = ans % mod * pingfang % mod; // 遇到二进制中的1时 说明需要取
}
pingfang = pingfang % mod * pingfang % mod; // pingfang不断往上翻
n /= 2;
}
return ans % mod;
}
int isPrime(int x) // 判断素数
{
for (int i = 2; i <= sqrt(x); i++)
{
if (x % i == 0)
return 0;
}
return 1;
}
int main()
{
int p, a;
while (cin >> p >> a, p + a)
{
if (isPrime(p) || poww(a, p, p) != a) // 两个条件
{
cout << "no" << endl;
}
else
{
cout << "yes" << endl;
}
}
return 0;
}
1017:Shopping(暴力+map)
题意:给 家店,一开始每家店的出售价格都是 ,之后 天中每天都给出这 家店的涨价数值(给出的顺序不固定),问 天中每天涨价后名为 的店的价格排名是多少。第 家店价格排名定义为如果有 家店的价格比 店贵,那么第 家店的排名就是 。
分析: 。看这数据范围可以直接直接进行暴力就可以了,每天更新每家店的价格,然后遍历找到比 店价格高的店数量即可,因为给出的顺序不固定,所以使用 来确定索引。
Code:
const int MAXN = 1e4 + 10;
int n, m;
map<string, int> id;
int arr[MAXN];
int main()
{
while (cin >> n)
{
memset(arr, 0, sizeof(arr)); // 清空店铺价格
id.clear(); // 清空map
int index = 0; // 用于确定店铺的索引
for (int i = 0; i < n; i++)
{
string str;
cin >> str;
id[str] = index++; // 给每家店分配索引
}
cin >> m;
for (int i = 0; i < m; i++)
{
for (int j = 0; j < n; j++)
{
int s;
string str;
cin >> s >> str;
arr[id[str]] += s; // 给每家店涨价
}
int now = arr[id["memory"]];
int rk = 1;
for (int j = 0; j < n; j++) // 遍历确定排名
{
if (arr[j] > now)
rk++;
}
cout << rk << endl;
}
}
return 0;
}
二、medium
1004:Turn the corner(计算几何+三分)
题意:给定车长 、车宽 、所处街道宽度 、目标街道宽度 ,问这辆车是否能够转弯成功。
分析:将图片按照 轴翻转,如图建立坐标系。
设车身与 轴形成的角度为 ,则 的取值范围为 。
设题目给的街道宽度分别为 , ,则我们需要突出关注 这个点。
把靠近内侧的车身延长形成直线 。
我们需要知道在所有 的取值情况下,在 这个点上是否 。
现在需要求 的取值为多少时这条直线才能在 上得到最大的 。
易知在车的转弯过程中车身距离内侧墙角的距离先变小后变大,是一个凹函数,因此可以利用三分法求出距离墙角的极小值。(本周是二分练习,三分也是要学习的,还不会的赶紧去百度!)
求出目标角度 之后在直线方程中代入 ,若 则通过,否则不能通过。详见代码。
Code:
const int MAXN = 5010;
const double eps = 1e-5; // 控制精度
const double PI = acos(-1.0); // 小技巧,π的取值
double x, y, l, w;
double f(double a) // 车身直线
{
return -tan(a) * y + w / cos(a) + l * sin(a);
}
double check(double l, double r) // 三分法
{
double lm, rm;
while (r - l > eps)
{
lm = (l + r) / 2;
rm = (lm + r) / 2;
if (f(lm) > f(rm))
{
r = rm;
}
else
{
l = lm;
}
}
return l;
}
int main()
{
while (cin >> x >> y >> l >> w)
{
// 至少要满足两侧街道宽度不少于车的宽度
if (w > x || w > y || f(check(0, PI / 2)) > x)
{
cout << "no" << endl;
}
else
{
cout << "yes" << endl;
}
}
return 0;
}
1006:Cup(数学+二分)
题意:给一个杯子的下圆半径 、上圆半径 、高度 以及杯中水的体积 ,求杯中水的高度。
分析:杯子的形状可能是一个圆柱体,也可能是一个圆台,设杯中水的高度为 ,水面半径为 。
① 圆柱体:用体积 除以底面积 即可,即 。
② 圆台:将圆台补成圆锥,设此时圆锥的高度为 如图所示,易得 ,即 。
且 ,即 , 这样只要确定了 的值那么 的值也就可以确定了。
又因为圆台体积公式 , 随着 的增大而增大,因此就可以上二分确定水的高度 ,检查体积是否为 。详见代码。
Code:
const double PI = acos(-1.0);
double f(double h, double R, double r) // 圆台体积公式
{
return PI * h * (R * R + r * r + R * r) / 3;
}
int main()
{
int T;
cin >> T;
while (T--)
{
double r, R, H, V;
cin >> r >> R >> H >> V;
if (r == R) // 如果是圆柱体
{
cout << fixed << setprecision(6) << V / (PI * r * r) << endl;
continue;
}
double x = H * r / (R - r); // 得到x
double left = 0, right = H;
while (right - left > eps) // 二分确定答案
{
double mid = (right + left) / 2; // 确定高度
double rr = R * (mid + x) / (H + x); // 确定半径
if (f(mid, rr, r) < V) // 根据体积缩小范围
{
left = mid;
}
else
{
right = mid;
}
}
cout << fixed << setprecision(6) << left << endl;
}
return 0;
}
1008:Can you find it?(合并排序+二分)
题意:给三个数组 ,在给一系列的值 ,问是否能在三个数组中各取一个值满足 。
分析: 最大可以是 ,询问最多有 条。
如果对于每条询问都执行三重循环枚举的话 必然超时!
而如果选择事先对数组 进行排序,之后执行两重循环枚举 和 ,在 中二分查找需要的值 ,这样 ,依然会超时!
现在考虑先双重循环枚举 ,将它们的和保存到新数组 中,并且排序,之后一重循环枚举 ,然后在 中二分查询需要的值 ,这样 ,这个就不会超时了。
Notice:这题有点莫名其妙,我的数组开小了给我提示 ,我把数组再开大就 了。
Code:
const int MAXN = 500 + 10;
int A[MAXN], B[MAXN], C[MAXN], D[MAXN*MAXN]; // 注意数组D的大小
int main()
{
int L, N, M;
int kase = 1;
while (cin >> L >> N >> M)
{
cout << "Case " << kase++ << ":" << endl;
for (int i = 0; i < L; i++)
{
cin >> A[i];
}
for (int i = 0; i < N; i++)
{
cin >> B[i];
}
for (int i = 0; i < M; i++)
{
cin >> C[i];
}
int index = 0;
for (int i = 0; i < N; i++) // 双重循环处理数组D
{
for (int j = 0; j < M; j++)
{
D[index++] = B[i] + C[j];
}
}
sort(D, D + index); // 必须要排序,二分查找的前提就是有序
int k;
cin >> k;
for (int i = 0; i < k; i++)
{
int target;
cin >> target;
int ok = 0;
for (int j = 0; j < L; j++)
{
int pos = lower_bound(D, D + index, target - A[j]) - D; // 在D中二分查找目标值
if (pos >= index) // 没找到,继续
continue;
if (D[pos] + A[j] == target) // 找到了,输出答案
{
ok = 1;
break;
}
}
if (ok)
cout << "YES" << endl;
else
cout << "NO" << endl;
}
}
return 0;
}
1009:Toxophily(物理/三分+二分)
题意:规定人站在起点 ,且人的高度 进行射箭,给定箭的初速度 ,现在要求最小的角度 ,让这支箭能够射到位于 的苹果,箭会受到重力的影响, 。
分析:设苹果的位置为 ,我们首先需要知道这支箭在所有的角度下射出去到达 时的最大高度 是否满足 ,只要满足这个条件,那么就一定有解。
根据我们的生活常识,随着角度的变大,落在 上的 值会先增后减,即使一个凸函数,那么我们就可以使用三分来确定极值点对应的角度 ,当然也可以使用物理的方式进行分析计算得到 ,我这里采取后者。
首先我们假设这支箭是可以射到 处的,我们计算在这个角度 下到达 的高度 。
如果这支箭不能到达 ,那么计算出来的 值是一个负数,可能不能满足要求。
设箭到达 的时间 ,
那么根据匀变速运动公式
求导
令 得
这样我们就通过了计算的方式而不是三分得到了目标 的值。
那么我们就知道了 在 的范围中单调递增,因此我们进行二分,查找 时对应的角度 。
Code:
const double eps = 1e-8;
const double g = 9.8;
double x, y, v;
double f(double a) // 角度a下在x=x1时的y
{
return x * tan(a) - g * x * x / (2 * v * v * cos(a) * cos(a));
}
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> x >> y >> v;
double left = 0, right = atan(v * v / (g * x)); // 计算目标角度a
if (f(right) < y) // 如果最大值还达不到苹果的高度,无解
{
cout << -1 << endl;
continue;
}
left = 0;
while (right - left > eps) // 否则进行二分
{
double mid = (left + right) / 2;
if (f(mid) < y)
{
left = mid;
}
else
{
right = mid;
}
}
cout << fixed << setprecision(6) << right << endl;
}
return 0;
}
1010:Pie(二分答案)
题意:给 个圆的半径,现在需要在这些圆中割出大小相同的 个小圆,问小圆的最大 。
分析:典型的二分答案,我们先二分确定小圆的 ,之后贪心得在这 个圆中割出小圆,检查这样割出的小圆数量是否 。
Notice:这题精度有点毒瘤,直接二分半径可能不够,应该要二分半径的平方,因为算出半径之后还需要算 ,因此可以对 进行二分。除此以外还要注意在 的时候要向下取整!
Code:
const int MAXN = 1e4 + 10;
const double eps = 1e-6;
const double PI = acos(-1.0);
int n, f;
double arr[MAXN];
int check(double mid) // 贪心检查是否满足要求
{
int cnt = 0;
for (int i = 0; i < n; i++)
{
cnt += (int)(arr[i] / mid); // 注意向下取整
}
return cnt >= f;
}
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> n >> f;
f++;
double maxV = 0;
for (int i = 0; i < n; i++)
{
cin >> arr[i];
arr[i] = arr[i] * arr[i]; // 先变成r^2
maxV = max(maxV, arr[i]); // 确定右边界
}
double l = 0, r = maxV;
double mid;
while (r - l > eps) // 二分确定最大r^2
{
mid = (r + l) / 2;
if (check(mid))
{
l = mid;
}
else
{
r = mid;
}
}
cout << fixed << setprecision(4) << mid * PI << endl; // 输出πr^2
}
return 0;
}
1012:Shell Pyramid(数学+二分)
题意:有一个金字塔,如下图从左到右代表金字塔从上到下的横截面,第一层
个,第二层比第一层多
个,第三层比第二层多
个 … 以此类推。现在要求整个金字塔中第
个所在的层数、行数以及列数。例如第
个是在第
层第
行第
个,第
个是在第
层第第
行第
个。
分析:我们可以先确定第
个是在第几层。
设 表示前 层的元素数量,那么
那么 满足的典型的三角形金字塔数
因为数据最大为 ,那么 值需要到 就可以了,因此我们可以先预处理出这个数组 ,然后进行二分找到第一个大于等于 的 ,这样就得到了对应的层数 。
那么接下来我们只需要知道 在当前层的哪一行就可以知道对应的列。
设 表示第 层的元素数量,那么
那么 其实就是等差数列,满足
同样我们预处理前 层的 数组,二分找到第一个大于等于 的 ,这样就得到了对应的行数 ,那么得到对应的列数 。详见代码。
Code:
const int MAXN = 1e6 + 10;
int num[MAXN], sum[MAXN]; // 分别代表数组B和A
int main()
{
for (int i = 1; i < MAXN; i++) // 预处理B数组
{
num[i] = num[i - 1] + i;
}
for (int i = 1; i < MAXN; i++) // 预处理A数组
{
sum[i] = sum[i - 1] + num[i];
}
int T;
cin >> T;
while (T--)
{
int s;
cin >> s;
int pos1 = lower_bound(sum, sum + MAXN, s) - sum; // 调库二分找到对应的层数
s -= sum[pos1 - 1];
int pos2 = lower_bound(num, num + MAXN, s) - num; // 调库二分找到对应的行数
s -= num[pos2 - 1];
cout << pos1 << " " << pos2 << " " << s << endl; // 输出层、行、列
}
return 0;
}
1013:Cable master(二分答案)
题意:给 根电缆的长度,需要切割出长度相等的 根电缆,问最大的长度。
分析:与 类似,二分答案,然后贪心地检查就可以了。详见代码。
Code:
// 在[0, maxV]中二分切割的长度x,遍历n条现有的电缆,电缆长度/x = 最多可以切割的数量,将数量加起来判断是否 >= k
const int MAXN = 1e4 + 10;
const double eps = 1e-9;
int n, k;
double arr[MAXN];
int check(double mid)
{
int cnt = 0;
for (int i = 0; i < n; i++)
{
cnt += (int)(arr[i] / mid); // 注意向下取整
}
return cnt >= k;
}
int main()
{
while (cin >> n >> k, n + k)
{
double l = 0, r = 0;
for (int i = 0; i < n; i++)
{
cin >> arr[i];
r = max(r, arr[i]); // 确定右边界
}
while (r - l > eps)
{
double mid = (l + r) / 2;
if (check(mid))
{
l = mid;
}
else
{
r = mid;
}
}
cout << fixed << setprecision(2) << l << endl;
}
return 0;
}
1016:Assemble(二分答案+分桶优化)
题意:需要组装电脑,现在给 个配件的信息以及预算 ,给的配件中会出现类型相同的不同型号配件,需要在所有类型的配件中选择其中一件来组装电脑,而电脑的性能满足木桶短板效应,取决于其中质量最差的配件。现在要问不超出预算的情况下组装的电脑能得到的最大性能。
分析:也是典型的二分答案,我们先二分确定电脑的性能,之后贪心的在所有的配件中进行选择,选择的配件质量不能小于当前二分的答案,如果最后选择了所有需要的配件并且没有超出预算就算满足要求。
如果直接这么写的话就 了。。。我们考虑优化,不需要在每次检查的时候对 个配件都进行检查,可以先把同类的配件按照价格升序放在一起,这样我们检查答案的时候就可以在每一类配件中选择最便宜的满足答案的配件后及时退出,继续到下一类中进行寻找!详见代码。
Code:
const int MAXN = 1000 + 10;
int n, money, cnt;
struct Good // 货物
{
string type, name; // 类别,名称
int price, quality; // 价格,质量
bool operator<(Good other) const // 自定义排序规则
{
return price < other.price; // 注意是按照价格排序
}
} goods[MAXN];
map<string, int> id; // 为每一类配件分配索引
vector<Good> vec[MAXN]; // 有很多个桶,里面保存同类货物
int check(int mid) // 检查答案
{
int cost = 0; // 保存费用
for (int i = 0; i < cnt; i++) // 遍历所有类别的配件
{
int ok = 0; // 用来确定是否成功选择了配件
for (auto v : vec[i]) // 枚举每个类别中的配件,C++11中的用法,用普通的for循环也一样
{
if (v.quality >= mid) // 因为已经按照价格排序,如果质量达到要求就选择,保证最优性
{
cost += v.price; // 记下已经花的钱
ok = 1; // 成功选择
break;
}
}
if (!ok || cost > money) // 如果没有成功选上或者超出预算了就说明这个答案不可行
return 0;
}
return 1; // 所有类别配件都选择了并且没有超出预算,答案可行
}
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> n >> money;
id.clear(); // 需要清空map
for (int i = 0; i < n; i++)
vec[i].clear(); // 清空所有的桶
cnt = 0; // 用于分配索引
int l = 0, r = 0;
for (int i = 0; i < n; i++)
{
cin >> goods[i].type >> goods[i].name >> goods[i].price >> goods[i].quality;
r = max(r, goods[i].quality); // 确定右边界
if (!id.count(goods[i].type)) // 这个类别的配件没有出现过则分配新的索引
{
id[goods[i].type] = cnt++;
}
vec[id[goods[i].type]].push_back(goods[i]); // 把这个配件放到同类的桶中
}
for (int i = 0; i < cnt; i++) // 对所有的桶按照价格从低到高进行排序
{
sort(vec[i].begin(), vec[i].end());
}
while (r > l) // 二分确定答案
{
int mid = (l + r + 1) / 2;
if (check(mid))
{
l = mid;
}
else
{
r = mid - 1;
}
}
cout << l << endl;
}
return 0;
}
三、hard
1001:Fibonacci Numbers(通项公式+矩阵快速幂)
题意:求斐波那契数列第 项的值, ,由于 非常大,最多只需要输出 位的答案,如果超过了 位则输出前 位以及后 位,中间加上 “…”。
分析:我吐了,又是数学题,需要用到斐波那契的通项公式 以及矩阵快速幂。
① 时,该数列的位数不超过 位,因此这个部分直接预处理打表就可以了。
② ,分成高四位与低四位。
低四位可以用矩阵快速幂计算:
然后直接套上矩阵快速幂就可以啦。
高四位需要用到斐波那契的通项公式:
此时 ,因为此时 的影响已经非常小可以忽略不计了。由于我们只需要前四位,所以进行如下处理:
设 为 的长度
我们知道 ,例如
得到 ,
那么 ,
而 就是我们需要求的前四位,只需要把 的小数部分去除就可以了。
( )
对于这道题, ,则 ,详见代码。
参考:
https://www.cnblogs.com/martinue/p/5490535.html
https://www.cnblogs.com/heyujun/p/10298953.html
Code:
#include <bits/stdc++.h>
using namespace std;
const int MAXN = 40 + 10; // 只需要打表前40项就足够了
const int Mod = 1e4; // 在进行快速幂的时候用到,保留后四位
int N;
struct Matrix // 矩阵
{
int m[2][2];
Matrix() { memset(m, 0, sizeof(m)); }
void init() // 初始化
{
for (int i = 0; i < 2; i++)
m[i][i] = 1;
}
int *operator[](int id) { return m[id]; }
Matrix operator*(const Matrix &b) // 矩阵乘法
{
Matrix res;
for (int i = 0; i < 2; i++)
for (int j = 0; j < 2; j++)
for (int k = 0; k < 2; k++)
res[i][j] = (res[i][j] + m[i][k] * b.m[k][j] % Mod) % Mod;
return res;
}
};
int fibo[MAXN]; // 保存斐波那契打表的结果
int solve1() // 处理前四位
{
double s = log10(sqrt(5.0) / 5.0) + 1.0 * N * log10((1.0 + sqrt(5.0)) / 2.0); // 这里直接把s变成log(s)
s = s - (int)s; // 这一步和下一步一起相当于 10^(log(s)-len+4)
double ans = 1000 * pow(10.0, s);
return ans;
}
int solve2() // 处理后四位,矩阵快速幂
{
Matrix S, T, ans;
int n = N;
ans.init();
S[0][0] = 1;
T[0][0] = 1, T[0][1] = 1;
T[1][0] = 1, T[1][1] = 0;
while (n)
{
if (n & 1)
ans = ans * T;
n >>= 1;
T = T * T;
}
S = ans * S;
return ans[1][0];
}
int main()
{
fibo[0] = 0, fibo[1] = 1; // 打出前40的表
for (int i = 2; i <= 40; i++)
fibo[i] = fibo[i - 1] + fibo[i - 2];
while (~scanf("%d", &N))
{
if (N < 40) // 如果是前40项直接输出
printf("%d\n", fibo[N]);
else // 否则处理前四位和后四位
{
printf("%04d...%04d\n", solve1(), solve2());
}
}
return 0;
}
1011:Line belt(三分套三分)
题意:给两条线段 以及 ,在线段 上的速度为 ,在线段 上的速度为 ,在其他区域的速度为 。现在要从 到 ,求最短时间。
分析:因为需要从 走到 ,那么一定是从 上离开,从 上进行,那么我们设离开点为 ,进入点为 。
那么如何确定这两个点呢?先简化一下问题,假若给定了 ,那么怎么确定 呢?
如图,因为 上的速度与空白区域的速度是不同的,因此:
若空白区域的速度 大于 上的速度 ,那么从 直接到 即可;
否则先到 的中部 点再到 点。
不论哪种情况,随着 点从 到 的移动过程中, 到 的时间先减后增,或者单调递减,三分都可以应付,因此如果确定了 点,我们就可以使用三分法来求得 到 的最短时间。
那么现在的问题就是如何确定点 呢?
类似于刚刚的分析,随着 点从 到 移动过程中, 到 的时间先减后增,或者单调递增,因此我们还是可以使用三分法来确定 点,这样一来,我们需要求的 和 都求出来了,计算答案即可。
Notice:可以学习一下用结构体 来进行二分的方法。
Code:
const double eps = 1e-9;
struct Point // 点
{
double x, y; // 横纵坐标
Point(double _x, double _y) // 构造函数,初始化
{
x = _x;
y = _y;
}
};
double distance(Point x, Point y) // 求出两点之间的距离
{
return sqrt(pow(x.x - y.x, 2) + pow(x.y - y.y, 2));
}
double ax, ay, bx, by, cx, cy, dx, dy, p, q, r; // ABCD的坐标以及各个位置的速度
double f(Point a, Point b) // 确定P和Q点之后求从P到D的时间
{
double t = distance(a, b) / r + distance(a, Point(dx, dy)) / q;
return t;
}
double calc(Point a) // 确定P点之后求从P到D的时间
{
Point l(cx, cy), r(dx, dy);
while (distance(l, r) > eps) // 三分计算
{
Point lm((r.x + l.x) / 2, (r.y + l.y) / 2);
Point rm((lm.x + r.x) / 2, (lm.y + r.y) / 2);
if (f(lm, a) < f(rm, a)) // 凹函数,舍弃右边
{
r = rm;
}
else
{
l = lm;
}
}
return f(l, a) + distance(Point(ax, ay), a) / p; // P到D的时间+A到P的时间
}
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> ax >> ay >> bx >> by >> cx >> cy >> dx >> dy;
cin >> p >> q >> r;
Point l(ax, ay), r(bx, by);
while (distance(l, r) > eps) // 三分计算
{
Point lm((r.x + l.x) / 2, (r.y + l.y) / 2); // 学习一下,结构体来二分
Point rm((lm.x + r.x) / 2, (lm.y + r.y) / 2);
if (calc(lm) < calc(rm)) // 凹函数,舍弃右边
{
r = rm;
}
else
{
l = lm;
}
}
cout << fixed << setprecision(2) << calc(l) << endl;
}
return 0;
}
1018:考研路茫茫——早起看书(三分)
题意:给定 以及包含 个点的折线图 ,保证 是不降的,现在问最小的 的值是多少。
分析:我们首先需要明确最小 对应的横坐标 可能是在折线图 两点之间的,而两点 之间的连线方程为 , 为斜率。
设 ,
则 ,那么我们知道 可能是单调递减或者先减后增的,可以用三分来求得 两点线段中的极小值。
但是因为每段线段的斜率是不同的, 所以我们需要求所有的线段都求出斜率 再分别进行三分求极小值更新答案。详见代码。
Notice:这题好像精度有点问题…我最后输出三分 对应的值就对了。
Code:
const int MAXN = 1e4 + 10;
const double eps = 1e-9;
int m, n;
struct Point // 点
{
double x, y; // 横纵坐标
} points[MAXN];
double f(double a, double k, int i) // 求Q
{
return n / (a * a) + k * (a - points[i - 1].x) + points[i - 1].y;
}
int main()
{
while (cin >> m >> n)
{
for (int i = 0; i < m; i++)
{
cin >> points[i].x >> points[i].y;
}
double ans = 1e18; // 这里需要设置一个很大的值
for (int i = 1; i < m; i++) // 枚举每条线段
{
double k = (points[i].y - points[i - 1].y) / (points[i].x - points[i - 1].x); // 求出斜率k
double l = points[i - 1].x, r = points[i].x;
double lm, rm;
while (r - l > eps) // 二分求出线段中的极小值
{
lm = (l + r) / 2;
rm = (lm + r) / 2;
if (f(lm, k, i) < f(rm, k, i))
{
r = rm;
}
else
{
l = lm;
}
}
ans = min(ans, f(lm, k, i)); // 更新答案
}
cout << fixed << setprecision(3) << ans << endl;
}
return 0;
}
四、乱入的二分图
1003:Girls and Boys(二分图最大点独立集)
题意:有一堆的男男女女,现在给出每个人的若干暧昧对象,问最多能抓出多少个互相没有暧昧关系的人。
分析: 典型的二分图背景,不过这道题考的是二分图最大点独立集,即其中任意一对点之间没有连线的最大集合。这里有挺多图论的概念,还是等以后图论的时候再细说。这里提一下二分图的概念,在无向图中如果可以把所有的点分成两个不相交的集合,并且图中所有边的两个端点分别位于这两个集合中,这么这个图就成为二分图,如图所示。
图片引用自百度百科——“二分图”
说到二分图就不得不提到最大匹配,常用的算法有匈牙利、网络流Dinic等,初学者只要会用匈牙利算法就可以了,不会的自行百度学习即可。
最大匹配,就是图中的最大边集,满足任意两条边都没有公共的顶点。每个点最多只有一个匹配点。
而对于最大点独立集,现在我们只需要知道一个性质,在二分图中最大点独立集 = n - 最大匹配。
因此我们只需要套上匈牙利模板然后用点的总数 减去最大匹配就可以了。
Notice:注意要使用邻接表实现的匈牙利模板,邻接矩阵实现的匈牙利在本题会超时!
Code:
const int MAXN = 5010; // 点数的最大值
const int MAXM = 50010; // 边数的最大值
int n;
struct Edge // 边
{
int to, next; // to为边指向的点,next为下一条边
} edge[MAXM];
int head[MAXN], tot; // head为邻接表头,每个head对应一个点,tot表示head的总数
void init() // 初始化
{
tot = 0;
memset(head, -1, sizeof(head));
return;
}
void addEdge(int u, int v) // 加边
{
edge[tot].to = v;
edge[tot].next = head[u];
head[u] = tot++;
return;
}
int linker[MAXN]; // 保存匹配到的点
bool used[MAXN]; // 记录点是否被使用
int uN; // 记录点的数量
bool dfs(int u) // 寻找增广路
{
for (int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].to;
if (!used[v])
{
used[v] = true;
if (linker[v] == -1 || dfs(linker[v]))
{
linker[v] = u;
return true;
}
}
}
return false;
}
int hungary() // 匈牙利算法
{
int res = 0;
memset(linker, -1, sizeof(linker));
for (int u = 0; u < uN; u++) // 点的编号0~uN-1
{
memset(used, false, sizeof(used));
if (dfs(u))
{
res++;
}
}
return res;
}
int main()
{
while (scanf("%d", &n) == 1)
{
init(); // 多case,注意初始化
uN = n;
for (int i = 0; i < n; i++)
{
int num;
scanf("%d: (%d)", &i, &num); // 使用scanf好格式化输入
for (int j = 0; j < num; j++)
{
int v;
scanf("%d", &v);
addEdge(i, v);
}
}
printf("%d\n", n - hungary() / 2); // 因为这里是同一个集合内部进行匹配,所有匹配数量会乘以2,我们手动除以2
}
return 0;
}
【END】感谢观看!