QOJ.ac
QOJ
ID | 题目 | 提交者 | 结果 | 用时 | 内存 | 语言 | 文件大小 | 提交时间 | 测评时间 |
---|---|---|---|---|---|---|---|---|---|
#186102 | #4886. Best Sun | nhuang685 | WA | 1ms | 3896kb | C++20 | 17.5kb | 2023-09-23 07:06:14 | 2023-09-23 07:06:14 |
Judging History
answer
/**
* @file cf104491i-1.cpp
* @author n685
* @brief
* @date 2023-09-21
*
*
*/
#include <bits/stdc++.h>
#ifdef LOCAL
std::ifstream cin;
std::ofstream cout;
using std::cerr;
#else
using std::cin;
using std::cout;
#define cerr \
if (false) \
std::cerr
#endif
#ifdef LOCAL
#include "dd/debug.h"
#define dbg(...) lineInfo(__LINE__, #__VA_ARGS__), dbg1(__VA_ARGS__)
#define dbgR(...) lineInfo(__LINE__, #__VA_ARGS__), dbg2(__VA_ARGS__)
#define dbgP(p, ...) \
lineInfo(__LINE__, #__VA_ARGS__ " proj"), dbg3(p, __VA_ARGS__)
#define dbgRP(p, ...) \
lineInfo(__LINE__, #__VA_ARGS__ " proj"), dbg4(p, __VA_ARGS__)
void nline() { cerr << '\n'; }
#else
#define dbg(...) 42
#define dbgR(...) 4242
#define dbgP(...) 420
#define dbgRP(...) 420420
void nline() {}
#endif
using db = long double; // change it to double as needed
const db PI = std::acos(static_cast<db>(-1.0));
constexpr db EPS = 1e-9;
template <class T, class U> constexpr bool eq(const T &a, const U &b) {
if constexpr (std::is_floating_point<
typename std::common_type<T, U>::type>::value) {
return (std::abs((a - b) / std::max((db)1, (db)b)) < EPS);
} else {
return (a == b);
}
}
#define SIMPLE
template <class T> int sign(T v) { return (v > 0) - (v < 0); }
namespace G {
// based on https://codeforces.com/blog/entry/48122
namespace {
template <class T> struct Point {
T x, y;
Point() : x(0), y(0) {}
Point(T _x, T _y) : x(_x), y(_y) {}
// arithmetic
template <class U> Point &operator+=(const Point<U> &b) {
x += static_cast<T>(b.x), y += static_cast<T>(b.y);
return (*this);
}
template <class U> Point &operator-=(const Point<U> &b) {
x -= static_cast<T>(b.x), y -= static_cast<T>(b.y);
return (*this);
}
template <class U> Point &operator*=(const U &b) {
x *= static_cast<T>(b), y *= static_cast<T>(b);
return (*this);
}
template <class U> Point &operator/=(const U &b) {
x /= static_cast<T>(b), y /= static_cast<T>(b);
return (*this);
}
// other
template <class U> operator Point<U>() const {
return Point<U>(static_cast<U>(x), static_cast<U>(y));
}
friend std::istream &operator>>(std::istream &in, Point &p) {
in >> p.x >> p.y;
return in;
}
friend std::ostream &operator<<(std::ostream &out, const Point &p) {
#ifdef SIMPLE
out << p.x << ' ' << p.y;
#else
out << '(' << p.x << ", " << p.y << ')';
#endif
return out;
}
};
template <class T1, class T2> auto makePoint(const T1 &x, const T2 &y) {
using R = typename std::common_type<T1, T2>::type;
return Point<R>(static_cast<R>(x), static_cast<R>(y));
}
// basic operations
template <class T1, class T2> auto operator*(const Point<T1> &a, const T2 &b) {
return makePoint(a.x * b, a.y * b);
}
template <class T1, class T2> auto operator*(const T1 &b, const Point<T2> &a) {
return makePoint(a.x * b, a.y * b);
}
template <class T1, class T2> auto operator/(const Point<T1> &a, const T2 &b) {
return makePoint(a.x / b, a.y / b);
}
template <class T> auto operator-(const Point<T> &a) {
// return makePoint(-a.x, -a.y);
return Point<T>(-a.x, -a.y);
}
template <class T1, class T2>
auto operator+(const Point<T1> &a, const Point<T2> &b) {
return makePoint(a.x + b.x, a.y + b.y);
}
template <class T1, class T2>
auto operator-(const Point<T1> &a, const Point<T2> &b) {
return makePoint(a.x - b.x, a.y - b.y);
}
template <class T1, class T2>
auto operator*(const Point<T1> &a, const Point<T2> &b) {
return a.x * b.x + a.y * b.y;
}
template <class T1, class T2>
auto operator^(const Point<T1> &a, const Point<T2> &b) {
return a.x * b.y - a.y * b.x;
}
// comparison - y takes priority
template <class T1, class T2>
bool operator==(const Point<T1> &a, const Point<T2> &b) {
return (eq(a.x, b.x) && eq(a.y, b.y));
}
template <class T1, class T2>
bool operator!=(const Point<T1> &a, const Point<T2> &b) {
return !(a == b);
}
template <class T1, class T2>
bool operator<(const Point<T1> &a, const Point<T2> &b) {
return eq(a.y, b.y) ? (a.x < b.x) : (a.y < b.y);
}
template <class T1, class T2>
bool operator>(const Point<T1> &a, const Point<T2> &b) {
return eq(a.y, b.y) ? (a.x > b.x) : (a.y > b.y);
}
template <class T1, class T2>
bool operator<=(const Point<T1> &a, const Point<T2> &b) {
return eq(a.y, b.y) ? (a.x <= b.x) : (a.y <= b.y);
}
template <class T1, class T2>
bool operator>=(const Point<T1> &a, const Point<T2> &b) {
return eq(a.y, b.y) ? (a.x >= b.x) : (a.y >= b.y);
}
// comparison - x takes priority
/* template <class T1, class T2>
bool operator==(const Point<T1> &a, const Point<T2> &b) {
return (eq(a.x, b.x) && eq(a.y, b.y));
}
template <class T1, class T2>
bool operator!=(const Point<T1> &a, const Point<T2> &b) {
return !(a == b);
}
template <class T1, class T2>
bool operator<(const Point<T1> &a, const Point<T2> &b) {
return eq(a.x, b.x) ? (a.y < b.y) : (a.x < b.x);
}
template <class T1, class T2>
bool operator>(const Point<T1> &a, const Point<T2> &b) {
return eq(a.x, b.x) ? (a.y > b.y) : (a.x > b.x);
}
template <class T1, class T2>
bool operator<=(const Point<T1> &a, const Point<T2> &b) {
return eq(a.x, b.x) ? (a.y <= b.y) : (a.x <= b.x);
}
template <class T1, class T2>
bool operator>=(const Point<T1> &a, const Point<T2> &b) {
return eq(a.x, b.x) ? (a.y >= b.y) : (a.x >= b.x);
} */
// angles
/**
* @brief determines if a comes before b in the counter-clockwise ordering
*
* @param a lhs
* @param b rhs
* @return -1 if a comes before b, 0 if a == b, and 1 if a comes after b
*/
template <class T1, class T2> int ccw(const Point<T1> &a, const Point<T2> &b) {
return sign(b ^ a);
}
template <class T1, class T2, class T3>
int ccw(const Point<T1> &a, const Point<T2> &b, const Point<T3> &o) {
return ccw(a - o, b - o);
}
template <class T> struct AngleCompare {
const Point<T> o;
AngleCompare(const Point<T> &origin = Point<T>()) : o(origin) {}
bool operator()(const Point<T> &a, const Point<T> &b) {
int v = ccw(a, b, o);
if (v == 0) {
return dist2(a, o) < dist2(b, o);
} else {
return v < 0;
}
}
};
template <class T1> auto angle(const Point<T1> &a) {
db d = std::atan2(static_cast<db>(a.y), static_cast<db>(a.x));
if (d < 0) {
d += 2 * PI;
}
return d;
}
template <class T1, class T2>
auto angle(const Point<T1> &a, const Point<T2> &b) {
return std::atan2(static_cast<db>(a ^ b), static_cast<db>(a * b));
}
template <class T1, class T2, class T3>
auto angle(const Point<T1> &a, const Point<T2> &b, const Point<T3> &o) {
return angle(a - o, b - o);
}
template <class T> auto rotate(const Point<T> &a, db sa, db ca) {
return makePoint(ca * a.x - sa * a.y, sa * a.x + ca * a.y);
}
template <class T> auto rotate(const Point<T> &a, db angle) {
return rotate(a, std::sin(angle), std::cos(angle));
}
template <class T1, class T2>
auto rotate(const Point<T1> &a, const Point<T2> &o, db angle) {
return rotate(a - o, angle) + o;
}
template <class T> auto perp(const Point<T> &a) { return makePoint(-a.y, a.x); }
template <class T1, class T2>
auto perp(const Point<T1> &a, const Point<T2> &o) {
return perp(a - o) + o;
}
// distances
template <class T> auto norm2(const Point<T> &a) { return a * a; }
template <class T> db norm(const Point<T> &a) {
return std::sqrt(static_cast<db>(norm2(a)));
}
template <class T> auto abs(const Point<T> &a) {
return std::sqrt(static_cast<db>(norm2(a)));
}
template <class T1, class T2>
auto dist2(const Point<T1> &a, const Point<T2> &b) {
return norm2(a - b);
}
template <class T1, class T2> db dist(const Point<T1> &a, const Point<T2> &b) {
return norm(a - b);
}
template <class T1, class T2>
Point<db> bisector(const Point<T1> &a, const Point<T2> &b) {
return a * norm(b) + b * norm(a);
// return a / norm(a) + b / norm(b);
}
template <class T1, class T2, class T3>
Point<db> bisector(const Point<T1> &a, const Point<T2> &b, const Point<T3> &o) {
return bisector(a - o, b - o) + o;
}
} // namespace
namespace {
template <class T> struct Line {
Point<T> a, ab;
Line() : a(), ab() {}
template <class T1, class T2>
Line(const Point<T1> &_a, const Point<T2> &_b, bool two_points = true) {
a = _a;
if (two_points) {
ab = _b - _a;
} else {
ab = _b;
}
}
Point<T> b() const { return (a + ab); }
friend std::ostream &operator<<(std::ostream &out, Line l) {
#ifdef SIMPLE
out << l.a << ' ' << l.b();
#else
out << l.a << " -> " << l.b();
#endif
return out;
}
// commenting this line out because it allowed for operations on bool like xor
// operator bool() const { return (ab != Point<T>()); }
bool line() const { return (ab != Point<T>()); }
Line operator-() const { return Line(b(), a); }
};
// a -> b or a -> (a + b)
template <class T1, class T2>
auto makeLine(const Point<T1> &a, const Point<T2> &b, bool two_points = true) {
using R = typename std::common_type<T1, T2>::type;
return Line<R>(static_cast<Point<R>>(a), static_cast<Point<R>>(b),
two_points);
}
// (x1, y1) -> (x2, y2)
template <class T1, class T2, class T3, class T4>
auto makeLine(const T1 &x1, const T2 &y1, const T3 &x2, const T4 &y2) {
return makeLine(makePoint(x1, y1), makePoint(x2, y2));
}
// Ax + By + C = 0
template <class T1, class T2, class T3>
auto makeLine(const T1 &A, const T2 &B, const T3 &C) {
assert(A != 0 || B != 0);
if (B == 0) {
return makeLine(makePoint(-static_cast<db>(C) / A, 0), makePoint(B, -A),
false);
} else {
return makeLine(makePoint(0, -static_cast<db>(C) / B), makePoint(B, -A),
false);
}
}
template <class T> std::array<T, 3> standard(const Line<T> &l) {
/* if (l.ab.y < 0)
return {-l.ab.y, l.ab.x, -l.ab.x * l.a.y + l.ab.y * l.a.x};
else */
return {l.ab.y, -l.ab.x, l.ab.x * l.a.y - l.ab.y * l.a.x};
}
template <class T> auto len2(const Line<T> &l) { return norm2(l.ab); }
template <class T> auto len(const Line<T> &l) { return norm(l.ab); }
/**
* @brief specifies the type of line; can take the three forms line, ray, and
* seg
*
*/
enum LineType { line, ray, seg };
template <LineType t, class T1, class T2>
bool on(const Point<T1> &p, const Line<T2> &l) {
if (!l.line()) {
return (p == l.a);
} else if (t >= 1 && ((p - l.a) * l.ab) <= 0) {
return (p == l.a);
} else if (t == 2 && ((p - l.b()) * l.ab) >= 0) {
return (p == l.b());
} else {
return eq((p - l.a) ^ l.ab, 0);
}
}
template <LineType t, class T1, class T2>
auto dist(const Point<T1> &p, const Line<T2> &l) {
if (!l.line()) {
return dist(p, l.a);
} else if (t >= 1 && ((p - l.a) * l.ab) <= 0) {
return dist(p, l.a);
} else if (t == 2 && ((p - l.b()) * l.ab) >= 0) {
return dist(p, l.b());
} else {
return std::abs((p - l.a) ^ l.ab) / norm(l.ab);
}
}
template <LineType t = line, class T1, class T2>
Point<db> closest(const Point<T1> &p, const Line<T2> &l) {
if (!l.line()) {
return l.a;
} else if (t >= 1 && ((p - l.a) * l.ab) <= 0) {
return l.a;
} else if (t == 2 && ((p - l.b()) * l.ab) >= 0) {
return l.b();
} else {
return l.a + l.ab * static_cast<db>((p - l.a) * l.ab) / norm2(l.ab);
}
}
template <LineType t = line, class T1, class T2>
Point<db> reflect(const Point<T1> &p, const Line<T2> &l) {
auto proj = closest<t>(p, l);
return 2 * proj - p;
}
/**
* @brief Determines and computes the intersection of two lines
*
* @tparam T1 type of first line
* @tparam T2 type of second line
* @tparam t1 line type of first line
* @tparam t2 line type of second line
* @param l1 first line
* @param l2 second line
* @return Whether the two lines interesct, as well as the point of
* intersection if and only if the lines intersect and are not parallel
*/
template <LineType t1, LineType t2, class T1, class T2>
std::pair<bool, Point<db>> intersect(const Line<T1> &l1, const Line<T2> &l2) {
if (!l1.line()) {
return {on<t2>(l1.a, l2), l1.a};
} else if (!l2.line()) {
return {on<t1>(l2.a, l1), l2.a};
}
if (eq(l1.ab ^ l2.ab, 0)) {
if (!eq(l1.ab ^ (l2.a - l1.a), 0)) {
return {false, Point<db>()};
}
if (t1 == line || t2 == line) {
return {true, Point<db>()};
}
if (t1 == seg) {
if (on<t2>(l1.a, l2) || on<t2>(l1.b(), l2)) {
return {true, Point<db>()};
}
}
if (t2 == seg) {
if (on<t1>(l2.a, l1) || on<t1>(l2.b(), l1)) {
return {true, Point<db>()};
}
}
if (t1 == ray && t2 == ray) {
if ((l1.ab * l2.ab) > 0 || on<t2>(l1.a, l2) || on<t1>(l2.a, l1)) {
return {true, Point<db>()};
}
}
return {false, Point<db>()};
}
// ((l1.a + i * l1.ab - l2.a) ^ l2.ab) == 0
auto i = (l2.a - l1.a) ^ l2.ab;
auto j = (l1.a - l2.a) ^ l1.ab;
auto d = l1.ab ^ l2.ab;
if (d < 0 && ((t1 == 2 && i < d) || (t1 >= 1 && i > 0) ||
(t2 == 2 && j > -d) || (t2 >= 1 && j < 0))) {
return {false, Point<db>()};
} else if (d > 0 && ((t1 == 2 && i > d) || (t1 >= 1 && i < 0) ||
(t2 == 2 && j < -d) || (t2 >= 1 && j > 0))) {
return {false, Point<db>()};
} else {
return {true, l1.a + static_cast<db>(i) / d * l1.ab};
}
}
template <LineType t1, LineType t2, class T1, class T2>
db dist(const Line<T1> &l1, const Line<T2> &l2) {
if (!l1.line()) {
return dist<t2>(l1.a, l2);
} else if (!l2.line()) {
return dist<t1>(l2.a, l1);
}
if (intersect<t1, t2>(l1, l2).first) {
return 0;
}
return std::min({dist<t2>(l1.a, l2), dist<t2>(l1.b(), l2), dist<t1>(l2.a, l1),
dist<t1>(l2.b(), l1)});
}
} // namespace
} // namespace G
using namespace G;
struct Event {
int type;
int i, j;
};
std::mt19937 rng(std::chrono::steady_clock::now().time_since_epoch().count());
constexpr db EPS2 = 1e-7;
db area(G::Point<int64_t> a, G::Point<int64_t> b, G::Point<int64_t> c) {
return std::abs((db)((a ^ b) + (b ^ c) + (c ^ a)) / 2);
}
void solve() {
int n;
cin >> n;
std::vector<G::Point<int64_t>> p(n);
for (int i = 0; i < n; ++i) {
cin >> p[i];
}
#ifndef LOCAL
std::shuffle(p.begin(), p.end(), rng);
#endif
std::vector<Event> v;
for (int i = 0; i < n; ++i) {
for (int j = i + 1; j < n; ++j) {
v.push_back(Event{0, i, j});
v.push_back(Event{0, j, i});
v.push_back(Event{1, i, j});
v.push_back(Event{1, j, i});
}
}
std::ranges::sort(v, {}, [&](auto a) {
db d = G::angle(p[a.i] - p[a.j]) + ((a.type) ? PI / 2 : (db)0);
if (d >= 2 * PI) {
d -= 2 * PI;
}
return std::pair{d, -a.type};
});
std::vector<std::vector<db>> off(n, std::vector<db>(n));
for (int i = 0; i < n; ++i) {
for (int j = 0; j < n; ++j) {
if (i == j) {
continue;
}
G::Line<int64_t> l = G::makeLine(p[i], p[j]);
for (int k = 0; k < n; ++k) {
if (k == i || k == j) {
continue;
}
if (G::ccw(p[k], p[j], p[i]) == -1 &&
G::on<G::seg>(G::closest<G::line>(p[k], l), l)) {
off[i][j] += std::min(G::dist(p[k], p[i]), G::dist(p[k], p[j]));
}
}
}
}
db ans = 0;
for (int s = 0; s < n; ++s) {
std::vector<std::vector<bool>> g(n, std::vector<bool>(n));
auto good = [&](db x) {
std::vector<db> dp(n, -4e18), sub(n);
dp[s] = 0;
for (auto [type, i, j] : v) {
if (type == 0) {
if (!g[i][j] && i != s && j != s) {
continue;
}
dp[i] = std::max(dp[i], dp[j] + area(p[s], p[i], p[j]) -
x * (G::dist(p[i], p[j]) + off[j][i]));
} else {
dp[j] -= x * G::dist(p[i], p[j]);
sub[j] += G::dist(p[i], p[j]);
}
}
return (dp[s] >= 0);
};
std::vector<int> lp;
for (int i = 0; i < n; ++i) {
if (s != i && p[s].y <= p[i].y) {
lp.push_back(i);
}
}
std::ranges::sort(lp, G::AngleCompare<int64_t>(p[s]),
[&](auto a) { return p[a]; });
int m = (int)lp.size();
for (int ii = 0; ii < m; ++ii) {
int i = lp[ii];
if (s == i || p[s].y > p[i].y) {
continue;
}
G::Point<int64_t> mi = G::makePoint<int64_t>(1, 0) + p[i];
for (int jj = ii + 1; jj < m; ++jj) {
int j = lp[jj];
if (s == j || p[s].y > p[j].y) {
continue;
}
G::Point<int64_t> d = p[j] - p[i];
if (G::angle(d - p[i]) >= G::angle(mi - p[i])) {
g[i][j] = true;
g[j][i] = true;
mi = d;
}
}
}
if (good(ans)) {
db l = ans, r = 4e18;
while (std::abs(r - l) >= EPS2) {
db mid = (l + r) / 2;
if (good(mid)) {
l = mid;
} else {
r = mid;
}
}
ans = l;
}
}
cout << std::fixed << std::setprecision(10) << ans << '\n';
}
int main() {
#ifdef LOCAL
cin.open("input.txt");
cout.rdbuf()->pubsetbuf(0, 0);
cout.open("output.txt");
#else
cin.tie(nullptr)->sync_with_stdio(false);
#endif
int t;
cin >> t;
while (t--) {
solve();
}
}
详细
Test #1:
score: 0
Wrong Answer
time: 1ms
memory: 3896kb
input:
4 3 -1 -1 1 -1 0 1 4 0 0 10 0 0 10 8 1 5 2 0 -2 0 1 1 -1 1 0 3 8 4 4 -4 4 4 -4 -4 -4 5 6 -6 5 -5 -6 6 -5
output:
0.3090169442 1.2368614048 0.2711375005 0.7624711004
result:
wrong answer 4th numbers differ - expected: '1.5631002', found: '0.7624711', error = '0.5122059'