leetcode 1235

news2025/1/10 2:00:58

leetcode 1235

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

代码

class Solution {
public:
    int jobScheduling(vector<int>& startTime, vector<int>& endTime, vector<int>& profit) {
        int n = startTime.size();
        vector<vector<int>> jobs(n);

        for(int i=0; i<n; i++){
            jobs[i] = {startTime[i], endTime[i], profit[i]};
        }

        sort(jobs.begin(), jobs.end(), [](const vector<int>&job1, const vector<int>&job2){return job1[1] < job2[1]; });

        vector<int> dp(n+1);

        for(int i=1; i<=n; i++){
            //找到结束时间大于i-th job的开始时间的job id, 因为是有序的,所以id 可以转为对应的job数量, dp[2th job] 表示前两个job的最优解答,局部最优成为全局最优解答
            int k = upper_bound(jobs.begin(), jobs.begin()+i-1, jobs[i-1][0], [](int st, vector<int>& job){
                return st < job[1];
            }) - jobs.begin();
            
            dp[i] = max(dp[i-1], dp[k] + jobs[i-1][2]);
        }

        return dp[n];
    }
};

例子

在这里插入图片描述
遍历已经排序过的jobs, dp[0] =0;
1,3,20, -> dp[1] = 20
2,3,50 -> dp[2] = 50
4,6,70 -> dp[3] = dp[2] + 70 = 120
6,9,60 -> dp[4] = dp[3] + 60 = 180
3,10,100 -> dp[5] = max (dp[4], dp[2] +100) =180

upper_bound

std::upper_bound 函数在 C++ 标准库 <algorithm> 头文件中定义。以下是一个可能的实现:

template<class ForwardIt, class T, class Compare>
ForwardIt upper_bound(ForwardIt first, ForwardIt last, const T& value, Compare comp)
{
    ForwardIt it;
    typename std::iterator_traits<ForwardIt>::difference_type count, step;
    count = std::distance(first, last);

    while (count > 0) {
        it = first;
        step = count / 2;
        std::advance(it, step);
        if (!comp(value, *it)) {
            first = ++it;
            count -= step + 1;
        } else {
            count = step;
        }
    }
    return first;
}

这段代码展示了 std::upper_bound 的基本工作原理。它采用了二分查找的方法,在已排序的范围 [first, last) 中查找第一个大于 value 的元素。

参数解释:

  • first:范围的起始迭代器
  • last:范围的终止迭代器
  • value:要查找的值
  • comp:比较函数对象,用于定义元素的比较方式

该实现假设了输入的迭代器是随机访问迭代器,因此可以使用 std::advancestd::distance 来计算迭代器之间的距离和移动迭代器。实际的标准库实现可能会更加复杂,并且会根据不同的情况进行优化。

upper_bound(jobs.begin(), jobs.begin()+i-1, jobs[i-1][0], [](int st, vector<int>& job){
                return st < job[1];
            })

upper_bound(jobs.begin(), jobs.begin()+i-1, jobs[i-1][0], [](int st, vector<int>& job){
                return st > job[1];
            })

的区别:
这两个 upper_bound 函数调用的区别在于它们所使用的比较函数对象的不同。这些函数都是用来在已排序的序列中查找某个值的上界的。让我们分析一下这两个调用:

  1. 第一个调用使用的比较函数是 [](int st, vector<int>& job){ return st < job[1]; },它的作用是比较给定的时间 st 和任务的结束时间 job[1]。这个比较函数适用于在结束时间升序排序的情况下查找 st 的上界,即在 jobs 中找到第一个结束时间大于 st 的任务的位置。

  2. 第二个调用使用的比较函数是 [](int st, vector<int>& job){ return st > job[1]; },它的作用是比较给定的时间 st 和任务的结束时间 job[1]。这个比较函数适用于在结束时间降序排序的情况下查找 st 的上界,即在 jobs 中找到第一个结束时间小于 st 的任务的位置。

所以,这两个调用的区别在于它们所使用的比较函数导致了不同的排序顺序和查找逻辑。

函数对象介绍

函数对象是指可以像函数一样被调用的对象。在C++中,函数对象可以是函数指针、函数、Lambda 表达式或重载了函数调用运算符 operator() 的类对象。

函数对象可以作为参数传递给标准库中的算法,如 std::sortstd::find_if 等,用于指定算法的行为。这种方式使得算法更加灵活,可以根据不同的需求使用不同的比较方式或操作方式。

以下是一些关于函数对象的重要概念和用法:

  1. 函数调用运算符 operator():重载了这个运算符的类对象可以像函数一样被调用。当对象被调用时,operator() 会被执行。

  2. Lambda 表达式:Lambda 表达式是一种轻量级的匿名函数,可以用于创建临时的函数对象。它可以在需要函数对象的地方直接定义,使代码更加简洁。

  3. 标准库算法:标准库提供了许多算法,如排序、查找、变换等,这些算法通常都可以接受函数对象作为参数,用于指定算法的行为。

  4. 适配器:函数对象可以通过适配器来改变其行为,如 std::bind 可以用于绑定参数,std::not1 可以用于对谓词取反等。

函数对象的灵活性和可重用性使得它们在C++中被广泛应用于各种场景,包括算法、事件处理、回调函数等。

可调用对象

可调用对象是指可以像函数一样被调用的对象。在 C++ 中,可调用对象可以是函数、函数指针、成员函数指针、函数对象(仿函数)、Lambda 表达式等。它们都可以在调用时像函数一样被执行。

不同类型的可调用对象:

  1. 函数:最基本的可调用对象,就是传统的函数。
int add(int a, int b) {
    return a + b;
}
  1. 函数指针:指向函数的指针,可以像函数一样被调用。
int (*funcPtr)(int, int) = add;
int result = funcPtr(3, 4); // 调用函数指针
  1. 成员函数指针:指向类的成员函数的指针。
class MyClass {
public:
    int multiply(int a, int b) {
        return a * b;
    }
};

int (MyClass::*memFuncPtr)(int, int) = &MyClass::multiply;
MyClass obj;
int result = (obj.*memFuncPtr)(3, 4); // 调用成员函数指针
  1. 函数对象(仿函数):重载了函数调用运算符 operator() 的类对象,可以像函数一样被调用。
class Add {
public:
    int operator()(int a, int b) {
        return a + b;
    }
};

Add addObj;
int result = addObj(3, 4); // 调用函数对象
  1. Lambda 表达式:匿名函数,可以直接在需要的地方定义和使用,也可以像函数一样被调用。
auto lambda = [](int a, int b) { return a + b; };
int result = lambda(3, 4); // 调用 Lambda 表达式

在 C++ 中,可调用对象的灵活性和多样性使得它们可以适用于各种不同的场景,例如作为算法的参数、事件处理、回调函数等。

lower_bound 和 upper_bound 的区别?

lower_boundupper_bound 在功能上有所区别,尽管它们都是用于在有序序列中进行查找的算法:

  1. lower_bound

    • 返回的是第一个大于或等于给定值的元素的位置。
    • 如果在序列中找不到大于或等于给定值的元素,则返回指向序列末尾的迭代器。
    • 如果序列中存在多个等于给定值的元素,lower_bound 会返回它们中第一个元素的位置。
  2. upper_bound

    • 返回的是第一个大于给定值的元素的位置。
    • 如果在序列中找不到大于给定值的元素,则返回指向序列末尾的迭代器。
    • 如果序列中存在多个等于给定值的元素,upper_bound 会返回大于这些元素的第一个位置。

因此,lower_bound 返回的位置可能是等于给定值的元素的位置,而 upper_bound 则一定是大于给定值的元素的位置。这两个函数在处理有序序列时很有用,尤其是在需要进行范围查询或插入操作时。

sort 函数

std::sort 是 C++ 标准库中的一个算法,位于 <algorithm> 头文件中,用于对序列进行排序。它采用的是快速排序(Quicksort)或者堆排序(Heapsort)等效率较高的排序算法。

std::sort 函数的函数签名如下:

template< class RandomIt >
void sort( RandomIt first, RandomIt last );

其中:

  • firstlast 是表示要排序的序列的迭代器范围,即 [first, last),它们定义了要排序的区间。

std::sort 函数会按照默认的升序规则对指定的序列进行排序。

要按照降序规则排序,可以通过传递一个自定义的比较函数作为第三个参数。比较函数应当返回一个布尔值,指示其第一个参数是否应该排在第二个参数之前。如果第一个参数应排在第二个参数之前,则返回 true;否则,返回 false

以下是一个示例,展示如何使用 std::sort 对序列进行升序和降序排序:

#include <iostream>
#include <algorithm>
#include <vector>

// 自定义比较函数,用于降序排序
bool descending(int a, int b) {
    return a > b;
}

int main() {
    std::vector<int> vec = {5, 2, 9, 3, 7};

    // 升序排序
    std::sort(vec.begin(), vec.end());
    std::cout << "Ascending order: ";
    for (int num : vec) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    // 降序排序
    std::sort(vec.begin(), vec.end(), descending);
    std::cout << "Descending order: ";
    for (int num : vec) {
        std::cout << num << " ";
    }
    std::cout << std::endl;

    return 0;
}

在这个示例中,我们首先使用 std::sort 对序列进行升序排序,然后再次调用 std::sort,但这次传递了自定义的比较函数 descending,从而实现了降序排序。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1656997.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Anaconda】升级Anaconda Navigator提示JSONDecoderError,删除.condarc文件后搞定

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、报错&#xff1a;JSONDecoderError二、错误原因三、解决问题总结 前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 时间长未升级Ana…

本地搭建springboot服务并实现公网远程调试本地接口

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

2024/5/9 QTday4

完成定时器制作 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);connect(&timer2, &QTimer::timeout, this, &Widget::label_begin);connect(&…

数据库(MySQL)—— 索引

数据库&#xff08;MySQL&#xff09;—— 索引 什么是索引创建索引使用 CREATE INDEX 语句使用 ALTER TABLE 语句在创建表时定义索引特殊类型索引注意事项 举个例子无索引的情况有索引的情况为什么索引快索引的结构 今天我们来看看MySQL中的索引&#xff1a; 什么是索引 MyS…

0509_IO4

练习1&#xff1a; 创建一对父子进程&#xff1a; 父进程负责向文件中写入 长方形的长和宽 子进程负责读取文件中的长宽信息后&#xff0c;计算长方形的面积 1 #include <stdio.h>2 #include <string.h>3 #include <stdlib.h>4 #include <sys/types.h>…

PyCharm安装详细教程

PyCharm安装详细教程 PyCharm简介及其下载网站 PyCharm是由JetBrains打造的一款Python IDE(Integrated Development Environment&#xff0c;集成开发环境)&#xff0c;带有一整套可以帮助用户在使用Python语言开发时提高其效率的工具。PyCharm提供了代码编辑、调试、语法高亮…

【BUUCTF】[RoarCTF 2019]Easy Java1

工具&#xff1a;hackbar发包&#xff0c;bp抓包。 解题步骤&#xff1a;【该网站有时候send不了数据&#xff0c;只能销毁靶机重试】 这里的登录界面是个天坑【迷魂弹】 直接点击help&#xff0c;然后进行打开hackbar——通过post请求&#xff0c;再通过bp抓包&#xff0c;…

DetCLIPv3:面向多功能生成开放词汇的目标检测

DetCLIPv3:面向多功能生成开放词汇的目标检测 摘要IntroductionRelated worksMethod DetCLIPv3: Towards Versatile Generative Open-vocabulary Object Detection 摘要 现有的开词汇目标检测器通常需要用户预设一组类别&#xff0c;这大大限制了它们的应用场景。在本文中&…

长难句打卡5.9

For example, the Long Now Foundation has as its flagship project a mechanical clock that is designed to still be marking time thousands of years hence. 例如,今日永存资金会将机械钟表视为旗舰项目,因此该钟表旨在为未来几千年保持计时。 Foundation n.基金会flag…

如何快速注册企业邮箱?只需要三步

快速注册一个企业邮箱&#xff0c;只需要以下三个步骤&#xff1a;一是挑选适合的邮箱版本和邮箱价格&#xff0c;二是填写必要的企业信息&#xff0c;三是完成企业邮箱的基础配置。完成上述三个步骤&#xff0c;企业就能够拥有一个专属的企业邮箱。 一、挑选适合的邮箱版本 …

jpg照片怎么压缩到10k?压缩照片并不难

jpg照片怎么压缩到10k&#xff1f;随着数字摄影的普及&#xff0c;我们手机或电脑中存储的照片数量越来越多&#xff0c;而这些高分辨率的照片往往会占用大量的存储空间。为了节省空间&#xff0c;将JPG照片压缩到较小的文件大小成为了许多人的需求。本文将为您介绍几款可以将J…

《基于GNU-Radio和USRP的雷达通信系统的实现》文献阅读

文章目录 前言一、摘要二、引言三、联合系统实施1、基本原理2、实验方案 四、软件设置1、发射机2、接收机 五、实验结果1、实验设置2、波形3、室内外对比4、不同参数的结果 六、结论七、参考文献八、论文自取九、阅读收获 前言 本文记录《基于GNU-Radio和USRP的雷达通信系统的实…

新手做抖音小店,应该怎么做才挣钱?按照这个思路来!

大家好&#xff0c;我是电商糖果 别人在抖音创业开店&#xff0c;都可以赚到钱&#xff0c;他们是怎么做到的。 作为一个新手&#xff0c;应该怎么做呢&#xff1f; 我相信这是很多人想开店的朋友&#xff0c;前期都会思考的问题。 下面糖果就来说一下&#xff0c;新手做抖…

vue 代码样式问题

部分电脑存在样式错乱问题&#xff0c;部分电脑样式正常。最后发现是样式写在 el-col 里面导致的。 注意&#xff1a;写样式不要放在 el-row 或者 el-row &#xff0c;导致部分电脑会出现莫名其妙的样式问题 <el-row class"detail"><el-col class"it…

达梦数据库限制用户登录IP测试

达梦数据库创建用户时可以限制登录ip和时间段。 创建测试测试用户 create user test1 identified by Test_1234 ALLOW_IP "192.168.100.101"; 限定该用户只能通过192.168.100.101地址登录数据库 连接测试 上图可见&#xff0c;192.168.100.101客户端可以连接上19…

研发妹子说如果问题不解决她就给我一口大黑锅:记Oracle库SQL执行超时导致业务无法测试优化处理记录

备注&#xff1a;本次记录根据后期回顾整理&#xff0c;实际测试过程遇到问题较多&#xff0c;也花费了不少时间&#xff0c;后期整理也有些处理过程被遗漏。 一、问题现象及处理过程 研发中心开发部一同事反馈某Oracle 测试库 10.xxx.xxx.xxx 无法访问&#xff0c;业务无法测…

振动分析的一些概念

一.时域分析 振动测试领域中&#xff0c;通常使用标准是ISO 10816系列标准&#xff0c;其要去使用有效值&#xff08;RMS&#xff09;来表示震动信号的能量大小&#xff0c;并提供一组限制值&#xff0c;以帮助用户评估机器的振动水平是否正常。 1.位移&#xff1a; 峰峰&…

STM32--LoRa通信模块

ATK-LORA-01_V3.0(V3.0 是版本号&#xff0c;型号是 ATK-LORA-01 &#xff0c;下面均以 ATK-LORA-01表示该产品) 是 ALIENTEK 推出的一款体积小、微功率、低功耗、高性能远距离 LORA 无线串口模块。模块设计是采用高效的 ISM 频段射频 SX1278 扩频芯片&#xff0c;模…

【Java SE】对象的比较

&#x1f970;&#x1f970;&#x1f970;来都来了&#xff0c;不妨点个关注叭&#xff01; &#x1f449;博客主页&#xff1a;欢迎各位大佬!&#x1f448; 本期内容满满干货&#xff0c;将会深入介绍对象与对象之间是如何进行比较的&#xff0c;我们知道基本数据类型是可以直…

Linux0.11中MINIX 文件系统

阅读linux 的源码的时候对minix 文件系统有很多的疑惑&#xff0c;根据自己的认识将这些做一个总结。 MINIX 文件系统由六个部分组成&#xff0c;分别是引导块&#xff0c;超级块&#xff0c;i结点位图&#xff0c;逻辑块位图&#xff0c;i结点&#xff0c;数据块。 引导块&am…