【每日算法Day 101】字节跳动 AI Lab 精选面试编程题

简介: 【每日算法Day 101】字节跳动 AI Lab 精选面试编程题

今天字节三面结束了,超越妹妹保佑我通过吧!今天更新两道同学之前面试 AI Lab 时遇到的题。

0-1 背包问题(浮点数)

0-1 背包问题,一共 n < 20 个物品,每个物品价格 p[i] (浮点数),重量 w[i] (浮点数),背包容量 M (浮点数)。求最大能装的价值是多少?

输入:
20 678.91
23.56 51.56
31.45 23.56
62.54 45.62
15.32 42.23
12.32 65.32
65.12 32.45
15.65 45.78
62.15 98.32
32.15 45.62
15.44 95.32
45.65 99.45
32.15 22.48
23.56 51.56
31.45 23.56
62.54 45.62
15.32 42.23
12.32 65.32
65.12 32.45
15.65 45.78
62.15 98.32
输出:
1050.07

题解

因为这里全部都是浮点数,所以没有办法直接用普通的动态规划来做,这里我提供几个思路。

方法1:

如果小数点只有两位的话,很简单,所有数字统一乘以 100 ,那么就都变成整数了。然后就可以直接用普通的 0-1 背包方法来做。

方法2:

因为 n < 20 ,所以直接二进制枚举所有物品可能,然后取出重量小于背包容量,且价格最高的那一种就行了。时间复杂度  ,勉强可以接受。

方法3:

n 个物品平均分成两份,对每一份做二进制枚举,然后保存所有可能的总重量和对应的总价格。保存在两个数组中,记为 ab ,分别表示两份的所有可能情况。

预处理出 b 中重量小于等于 b[j].w 的最大价格,保存在 maxp[j] 中。

分别按照 w 排序,然后用双指针,从重量小的开始遍历 a 中每个元素 a[i] ,在 b 中找出重量最高的那个满足 a[i].w + b[j].w 不超过背包容量的 j

然后 i 移动到 i+1 ,也就是重量增加了,那么 j 只能减小了,直到减小到 a[i].w + b[j].w 再次不超过背包容量。然后直接取预处理好的 maxp[j]+a[i].p 和最优答案比较就行了。

最终时间复杂度是  ,相比上面直接二进制枚举所有情况,大大降低了呀。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mod = 1e9 + 7;
const int N = 22;
struct node {
    double w, p;
    bool operator < (const node& rhs) const {
        return w < rhs.w;
    }
} a[1<<N], b[1<<N];
int main() {
    int n;
    double M;
    scanf("%d%lf", &n, &M);
    printf("%f\n", M);
    vector<double> w(n, 0), p(n, 0);
    for (int i = 0; i < n; ++i) {
        scanf("%lf%lf", &w[i], &p[i]);
    }
    printf("%f\n", tmp);
    int ca = 0, cb = 0;
    for (int s = 0; s < (1<<(n/2)); ++s) {
        double tot_w = 0, tot_p = 0;
        for (int i = 0; i < n/2; ++i) {
            if (s&(1<<i)) {
                tot_w += w[i];
                tot_p += p[i];
                if (tot_w > M) break;
            }
        }
        if (tot_w <= M) {
            a[ca].w = tot_w;
            a[ca].p = tot_p;
            ca++;
        }
    }
    for (int s = 0; s < (1<<(n-n/2)); ++s) {
        double tot_w = 0, tot_p = 0;
        for (int i = 0; i < n-n/2; ++i) {
            if (s&(1<<i)) {
                tot_w += w[n/2+i];
                tot_p += p[n/2+i];
                if (tot_w > M) break;
            }
        }
        if (tot_w <= M) {
            b[cb].w = tot_w;
            b[cb].p = tot_p;
            cb++;
        }
    }
    sort(a, a+ca);
    sort(b, b+cb);
    vector<double> maxp(cb, 0);
    maxp[0] = b[0].p;
    for (int i = 1; i < cb; ++i) {
        maxp[i] = max(maxp[i-1], b[i].p);
    }
    int j = cb-1;
    double res = 0;
    for (int i = 0; i < ca; ++i) {
        while (j >= 0 && a[i].w+b[j].w > M) --j;
        if (j < 0) break;
        res = max(res, a[i].p+maxp[j]);
    }
    printf("%f\n", res);
    return 0;
}

最小长度子数组

给一个正数数组,找出最小长度连续子数组,其和大于等于 m

题解

这题还是用双指针,首先用 i 遍历每一个位置,然后维护 a[j] ~ a[i] 之间的元素和。如果发现和大于等于 m ,那就更新最小长度,同时增大 j 直到区间和小于 m 。最终时间复杂度是  的。

代码

#include <bits/stdc++.h>
using namespace std;
int main() {
    int n, m;
    scanf("%d%d", &n, &m);
    vector<int> a(n, 0);
    for (int i = 0; i < n; ++i) {
        scanf("%d", &a[i]);
    }
    int j = 0, sum = 0, res = INT_MAX;
    for (int i = 0; i < n; ++i) {
        sum += a[i];
        while (sum >= m) {
            res = min(res, i-j+1);
            sum -= a[j++];
        }
    }
    printf("%d\n", res);
    return 0;
}
相关文章
|
5天前
|
算法 Java 调度
《面试专题-----经典高频面试题收集四》解锁 Java 面试的关键:深度解析并发编程进阶篇高频经典面试题(第四篇)
《面试专题-----经典高频面试题收集四》解锁 Java 面试的关键:深度解析并发编程进阶篇高频经典面试题(第四篇)
14 0
|
28天前
|
消息中间件 算法 Java
抖音面试:说说延迟任务的调度算法?
Netty 框架是以性能著称的框架,因此在它的框架中使用了大量提升性能的机制,例如 Netty 用于实现延迟队列的时间轮调度算法就是一个典型的例子。使用时间轮调度算法可以实现海量任务新增和取消任务的时间度为 O(1),那么什么是时间轮调度算法呢?接下来我们一起来看。 ## 1.延迟任务实现 在 Netty 中,我们需要使用 HashedWheelTimer 类来实现延迟任务,例如以下代码: ```java public class DelayTaskExample { public static void main(String[] args) { System.ou
29 5
抖音面试:说说延迟任务的调度算法?
|
6天前
|
安全 Java API
《面试专题-----经典高频面试题收集三》解锁 Java 面试的关键:深度解析并发编程基础篇高频经典面试题(第三篇)
《面试专题-----经典高频面试题收集三》解锁 Java 面试的关键:深度解析并发编程基础篇高频经典面试题(第三篇)
11 0
|
13天前
|
存储 算法 Java
面试高频算法题汇总「图文解析 + 教学视频 + 范例代码」之 二分 + 哈希表 + 堆 + 优先队列 合集
面试高频算法题汇总「图文解析 + 教学视频 + 范例代码」之 二分 + 哈希表 + 堆 + 优先队列 合集
|
2月前
|
机器学习/深度学习 算法 固态存储
深度学习算法工程师面试问题总结| 深度学习目标检测岗位面试总结
本文给大家带来的百面算法工程师是深度学习目标检测岗位面试总结,文章内总结了常见的提问问题,旨在为广大学子模拟出更贴合实际的面试问答场景。在这篇文章中,我们还将介绍一些常见的深度学习目标检测面试问题,并提供参考的回答及其理论基础,以帮助求职者更好地准备面试。通过对这些问题的理解和回答,求职者可以展现出自己的深度学习目标检测领域的专业知识、解决问题的能力以及对实际应用场景的理解。同时,这也是为了帮助求职者更好地应对深度学习目标检测岗位的面试挑战,提升面试的成功率和竞争力。
|
23天前
|
数据采集 算法 数据挖掘
LeetCode 题目 80:删除排序数组中的重复项 II【算法面试高频题】
LeetCode 题目 80:删除排序数组中的重复项 II【算法面试高频题】
|
2月前
|
算法 搜索推荐 索引
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题(下)
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题
29 0
|
2月前
|
算法 程序员 索引
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题(中)
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题
30 0
|
2月前
|
算法 C语言 C++
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题(上)
数据结构与算法⑥(第二章OJ题,下)后八道链表面试题
18 0
|
2月前
|
算法 测试技术
数据结构与算法⑤(第二章OJ题,上)前五道链表面试题(下)
数据结构与算法⑤(第二章OJ题,上)前五道链表面试题
26 0