Leetcode 79 最佳观光组合

1 题目

1014. 最佳观光组合

给你一个正整数数组 values,其中 values[i] 表示第 i 个观光景点的评分,并且两个景点 ij 之间的 距离j - i

一对景点(i < j)组成的观光组合的得分为 values[i] + values[j] + i - j ,也就是景点的评分之和减去它们两者之间的距离。

返回一对观光景点能取得的最高分。

示例 1:

复制代码
输入:values = [8,1,5,2,6]
输出:11
解释:i = 0, j = 2, values[i] + values[j] + i - j = 8 + 5 + 0 - 2 = 11

示例 2:

复制代码
输入:values = [1,2]
输出:2

2 代码实现

cpp 复制代码
class Solution {
public:
    int maxScoreSightseeingPair(vector<int>& values) {
        int best = values[0]; 
        int ans = 0;

        for (int j = 1 ; j < values.size() ; j++){
            ans = max (ans, best + values[j] - j  );

            best = max(best , values[j] + j );
        }

        return ans;
    }
};

题解

思路一句话版

把公式

values[i] + values[j] + i - j

拆成

(values[i] + i) + (values[j] - j)

遍历 j 的时候,只需要维护 j 左边最大的 values[i] + i

C++ 代码(推荐写法)

cpp 复制代码
#include <vector>
#include <algorithm>
using namespace std;

class Solution {
public:
    int maxScoreSightseeingPair(vector<int>& values) {
        int best = values[0];  // 当前最大的 values[i] + i
        int ans = 0;

        for (int j = 1; j < values.size(); j++) {
            // 用当前 best 和 j 计算得分
            ans = max(ans, best + values[j] - j);

            // 更新 best,供后面的 j 使用
            best = max(best, values[j] + j);
        }

        return ans;
    }
};

为什么是 O(n)?

  • 每个元素只遍历一次

  • 只用了两个变量

  • 时间复杂度:O(n)

  • 空间复杂度:O(1)


再帮你对比一下暴力解法(为什么不行)

暴力写法(会超时)
cpp 复制代码
for (int i = 0; i < n; i++) {
    for (int j = i + 1; j < n; j++) {
        ans = max(ans, values[i] + values[j] + i - j);
    }
}
  • 时间复杂度:O(n²)

  • n 最大到 5 万,直接炸 💥


记忆口诀(刷题用)

i 在左,j 在右
i 的最大值提前存
j 来了直接算

详解这段代码中循环里计算得分和更新best的逻辑

首先,我们先明确这道题的核心公式:对于观光配对(i, j)(要求i < j),得分是values[i] + values[j] + i - j。我们可以把这个公式拆分成两部分:

  • 第一部分:values[i] + i(只和i有关)
  • 第二部分:values[j] - j(只和j有关)

所以得分就等价于**(values[i] + i) + (values[j] - j)** (i < j)。

代码逻辑逐行解释

cpp 复制代码
class Solution {
public:
    int maxScoreSightseeingPair(vector<int>& values) {
        int best = values[0];  // 初始化为 values[0] + 0(因为i=0时,values[i]+i就是values[0]+0)
        int ans = 0;

        for (int j = 1; j < values.size(); j++) {
            // 第一步:用当前找到的最优i(对应best=values[i]+i)和当前j计算得分
            // 得分 = (values[i]+i) + (values[j]-j) = best + values[j] - j
            ans = max(ans, best + values[j] - j);

            // 第二步:更新best,把当前j当作新的i候选(因为后面的j' > j,此时j就是i)
            // 新的候选值是 values[j] + j,和原来的best比,保留更大的那个
            best = max(best, values[j] + j);
        }

        return ans;
    }
};

举个例子帮你理解

假设values = [8,1,5,2,6],我们一步步走一遍:

  1. 初始化:best = 8 + 0 = 8ans = 0
  2. j=1 (values[j]=1):
    • 得分:8 + 1 - 1 = 8ans = 8
    • 更新 best:max(8, 1+1=2)best=8
  3. j=2 (values[j]=5):
    • 得分:8 + 5 - 2 = 11ans = 11
    • 更新 best:max(8, 5+2=7)best=8
  4. j=3 (values[j]=2):
    • 得分:8 + 2 - 3 = 7ans 还是 11
    • 更新 best:max(8, 2+3=5)best=8
  5. j=4 (values[j]=6):
    • 得分:8 + 6 - 4 = 10ans 还是 11
    • 更新 best:max(8, 6+4=10)best=10
  6. 最终返回ans=11(对应 i=0,j=2 的配对,得分 8+5+0-2=11,和计算结果一致)

总结

  1. 公式拆分是核心 :将原得分公式values[i]+values[j]+i-j拆分为(values[i]+i) + (values[j]-j),把双变量问题转化为单变量的遍历问题。
  2. 贪心维护最优值 :遍历j时,用best维护之前所有i < jvalues[i]+i的最大值,这样每一步只需要 O (1) 计算,整体时间复杂度是 O (n)。
  3. 遍历顺序 :因为要求i < j,所以从j=1开始遍历,先计算当前j的得分,再更新best(避免 j 自己和自己配对)。
相关推荐
lilihuigz14 小时前
Tutor LMS 4.0 Beta版全新上线:以学习者为中心的移动优先学习体验
学习·在线教育·lms
William Dawson14 小时前
2026软考中级系统集成项目管理工程师备考笔记
笔记·系统集成项目管理工程师
王老师青少年编程14 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮15 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说15 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove16 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung16 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了16 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL16 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰16 小时前
C++ 排列组合完整指南
开发语言·c++·算法