本页目录

LeetCode 149.直线上最多的点数

暴力

遍历每一对点,一对点确定一条直线,第三层循环再次遍历所有点判断是否在此直线上。判断是否在的连线上,只需要判断斜率:

为避免除法精度问题,转为乘法判断:

时间复杂度为

JavaScript
/**
 * @param {number[][]} points
 * @return {number}
 */
var maxPoints = function(points) {
    const n=points.length;
    let ans=1;
    for(let i=0;i<n;i++)
    {
        for(let j=i+1;j<n;j++)
        {
            const [x1,y1]=points[i];
            const [x2,y2]=points[j];
            let cnt=0;
            for(let k=0;k<n;k++)
            {
                const [x3,y3]=points[k];
                if((y3-y1)*(x3-x2)===(y3-y2)*(x3-x1)) cnt++;
            }
            ans=Math.max(ans,cnt);
        }
    }
    return ans;
};

哈希

考虑对于每一个点,遍历其他点,以斜率做键,统计每个斜率对应的点数。这样对于每一个点都能得到“过该点的直线最多经过的点数”,最后的答案就是对每个点的结果再取最大值。为了保证斜率唯一,用约分后的两点的作为键值。

frac函数用于约分得到最简化分数:

JavaScript
const gcd = (x, y) => (y ? gcd(y, x % y) : x);
const frac = (x, y) => ([x / gcd(x, y), y / gcd(x, y)]);

代码如下,时间复杂度为

JavaScript
/**
 * @param {number[][]} points
 * @return {number}
 */

const gcd = (x, y) => (y ? gcd(y, x % y) : x);
const frac = (x, y) => ([x / gcd(x, y), y / gcd(x, y)]);

var maxPoints = function (points) {
    const n = points.length;
    let ans = 0;
    for (let i = 0; i < n; i++) {
        const mp = {};
        for (let j = i + 1; j < n; j++) {
            const [x1, y1] = points[i];
            const [x2, y2] = points[j];
            const key = frac(y2 - y1, x2 - x1);
            mp[key] = (mp[key] || 0) + 1;
        }
        ans = Math.max(...Object.values(mp), ans);
    }
    return ans + 1;
};