【发布时间】:2011-08-14 12:15:41
【问题描述】:
给定 2 个三角形的边长。确定第二个三角形是否可以放入第一个三角形内?
有关更多详细信息,请阅读下面的完整问题说明:
http://acm.timus.ru/problem.aspx?space=1&num=1566&locale=en
我在下面的实现尝试了所有 (3!)^2 种可能的对齐三角形底边的组合。然后它会尝试在第一个三角形内移动第二个三角形,同时检查第二个三角形的底边是否不超过第一个三角形的底边。
但我一直得到错误的答案(WA)#16。
我给出的案例是第二张图片。很明显,如果您旋转 PQR 以对齐长度为 2.77 和 3.0 的边,第三个顶点将不在三角形 ABC 内。长度为 4.2 的边只能沿 len 5 的边对齐。因此这种情况仅在第二张图中显示的配置中满足。
你能帮我找出错误,建议一些我的算法失败的测试用例。也欢迎替代算法。
#include <cmath>
#include <iostream>
using namespace std;
const double PI = atan(1.0)* 4;
// Traingle ABC (Envelope)
double xa, ya, xb, yb, xc, yc;
// Traingle PQR (Postcard)
double xp, yp, xq, yq, xr, yr;
// Angle between sides AB and AC
double theta;
double signWrtLine(double x1, double y1, double x2, double y2, double x, double y)
{
double A = y2 - y1;
double B = x1 - x2;
double C = -(A * x1 + B * y1);
return (A * x + B * y + C);
}
bool fit()
{
if ((xr > xc) || (yq > yb)) return false;
if (signWrtLine(xa, ya, xb, yb, xq, yq) < 0) {
double d = (yq / tan(theta)) - xq;
return (xr + d <= xc);
}
return (signWrtLine(xa, ya, xb, yb, xq, yq) >= 0 &&
signWrtLine(xb, yb, xc, yc, xq, yq) >= 0 &&
signWrtLine(xc, yc, xa, ya, xq, yq) >= 0);
}
bool fit(double a[], double b[])
{
// generate the 3! permutations of the envelope
// loops i,k
for (int i = 0; i < 3; i++) {
double angle;
double u = a[i], v = a[(i + 1) % 3], w = a[(i + 2) % 3];
for (int k = 0; k < 2; k++) {
switch (k) {
case 0:
xa = 0, ya = 0;
angle = theta = acos((u * u + v * v - w * w) / (2 * u * v));
xb = v * cos(angle), yb = v * sin(angle);
xc = u, yc = 0;
break;
case 1:
// reflect envelope
swap(v, w);
angle = theta = acos((u * u + v * v - w * w) / (2 * u * v));
xb = v * cos(angle), yb = v * sin(angle);
break;
}
// generate the 3! permutations of the postcard
// loops j,k
for (int j = 0; j < 3; j++) {
double angle;
double u = b[j], v = b[(j + 1) % 3], w = b[(j + 2) % 3];
for (int k = 0; k < 2; k++) {
switch (k) {
case 0:
xp = 0, yp = 0;
angle = acos((u * u + v * v - w * w) / (2 * u * v));
xq = v * cos(angle), yq = v * sin(angle);
xr = u, yr = 0;
break;
case 1:
// reflect postcard
swap(v, w);
angle = acos((u * u + v * v - w * w) / (2 * u * v));
xq = v * cos(angle), yq = v * sin(angle);
break;
}
if (fit()) return true;
}
}
}
}
return false;
}
int main()
{
double a[3], b[3];
for (int i = 0; i < 3; i++) cin >> a[i];
for (int i = 0; i < 3; i++) cin >> b[i];
if(fit(a, b)) cout << "YES" << endl;
else cout << "NO" << endl;
return 0;
}
【问题讨论】:
-
对齐底座的方向可能不是让一个三角形适合另一个三角形的唯一方法。该链接上的图像暗示了不同的方向。
-
似乎是一种相当复杂的方法。为什么不使用一些几何图形并计算三角形的高度和各种角度。然后沿着第一个三角形的长边移动第二个最长的边,直到它的高度适合。比较角度并确保第一个三角形最长边有足够的空间容纳第二个(我可能已经解释得更好了)。
-
你能说明比较哪些角度吗?
-
您的算法对齐 2 个三角形的最长边,然后尝试根据高度找到合适的。但是对齐的碱基不必是适合退出的最大碱基。尝试以下测试用例:(3 3 2; 2 2 2)。如果我们将第一个三角形(3)的最长边与第二个三角形(2)的最长边对齐,则不合适。但是,如果对齐每个三角形的长度为 2 的基数,则显然存在拟合。
-
我认为该图像暗示了不适合的情况。如果图像表示有效拟合,则无法按照问题所述在 2 个边缘处密封信封。
标签: c++ computational-geometry