LeetCode 面试题 16.03. 交点

Posted airfy

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了LeetCode 面试题 16.03. 交点相关的知识,希望对你有一定的参考价值。

题目描述:

给定两条线段(表示为起点start = {X1, Y1}和终点end = {X2, Y2}),如果它们有交点,请计算其交点,没有交点则返回空值。

要求浮点型误差不超过10^-6。若有多个交点(线段重叠)则返回 X 值最小的点,X 坐标相同则返回 Y 值最小的点。

 

就是要分类讨论一下,要注意的是在判断两条线段是否相交时,不要使用浮点运算,可能会造成判断错误。

class Solution {
public:
    vector<double> intersection(vector<int>& start1, vector<int>& end1, vector<int>& start2, vector<int>& end2) {
        //设置别称,方便写代码
        const int& x1 = start1[0];
        const int& y1 = start1[1];
        const int& x2 = end1[0];
        const int& y2 = end1[1];
        const int& x3 = start2[0];
        const int& y3 = start2[1];
        const int& x4 = end2[0];
        const int& y4 = end2[1];

        //计算两条直线的系数
        int A1, B1, C1, A2, B2, C2;
        A1 = y1 - y2;
        B1 = x2 - x1;
        C1 = y2 * x1 - y1 * x2;
        A2 = y3 - y4;
        B2 = x4 - x3;
        C2 = y4 * x3 - y3 * x4;

        vector<double> res;
        if ((y2-y1)*(x4-x3) != (x2-x1)*(y4-y3)) {//如果斜率不同,这种情况不可能重合
            //测试是否相交
            if (A1 * x3 + B1 * y3 + C1 == 0) { res.push_back(x3); res.push_back(y3); return res;}
            if (A1 * x4 + B1 * y4 + C1 == 0) { res.push_back(x4); res.push_back(y4); return res;}
            if (A2 * x1 + B2 * y1 + C2 == 0) { res.push_back(x1); res.push_back(y1); return res;}
            if (A2 * x2 + B2 * y2 + C2 == 0) { res.push_back(x2); res.push_back(y2); return res;}

            if ((A1 * x3 + B1 * y3 + C1) * (A1 * x4 + B1 * y4 + C1) < 0
                && (A2 * x1 + B2 * y1 + C2) * (A2 * x2 + B2 * y2 + C2) < 0) {
                res.push_back(B2*C1 - B1*C2);
                res.push_back(A1*C2 - A2*C1);
                res[0] /= A2 * B1 - A1 * B2;
                res[1] /= A2 * B1 - A1 * B2;
                return res;
            }
            else return res;
        }
        else {//斜率相同,这种时候有可能重合,也有可能平行
            if (A1 * x3 + B1 * y3 + C1 == 0) {//两条直线是重合的
                if (x1 != x2){//不垂直y轴
                    if (max(x1, x2) >= min(x3, x4) && min(x3,x4) >= min(x1,x2)) {
                        if (x3 < x4) return vector<double> {double(x3), double(y3)};
                        else return vector<double> {double(x4), double(y4)};
                    }
                    else if (max(x3, x4) >= min(x1, x2) && min(x3, x4) <= min(x1, x2)) {
                        if (x1 < x2) return vector<double> {double(x1), double(y1)};
                        else return vector<double> {double(x2), double(y2)};
                    }
                    else {
                        return res;
                    }
                }
                else {//垂直y轴
                    if (max(y1, y2) >= min(y3, y4) && min(y3, y4) >= min(y1, y2)) {
                        if (y3 < y4) return vector<double> {double(x3), double(y3)};
                        else return vector<double> {double(x4), double(y4)};
                    }
                    else if (max(y3, y4) >= min(y1, y2) && min(y3, y4) <= min(y1, y2)) {
                        if (y1 < y2) return vector<double> {double(x1), double(y1)};
                        else return vector<double> {double(x2), double(y2)};
                    }
                    else {
                        return res;
                    }
                }
                
            }
            else {//平行
                return res;
            }
        }
    }
};

技术图片

以上是关于LeetCode 面试题 16.03. 交点的主要内容,如果未能解决你的问题,请参考以下文章

链表相关面试题:返回两个链表的第一个交点,判断单链表是否有环,返回入环的第一个结点

Leetcode程序员面试金典面试题04.06.后继者

LeetCode:118.杨辉三角面试题 17.01. 不用加号的加法

数据结构二叉树相关面试题 Java版 LeetCode题 ------- 二叉树

200道大数据面试常考Leetcode算法题--数组篇02(python带代码解析)

每天一道面试题LeetCode 206 -- 反转链表