- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
java解法
- 解题思路
- 本题要求从一组价格中选择三个人,找出一个组合,其总价尽量接近给定的预算(不超过预算)。我们需要通过一种优化的算法来寻找这个组合,使得时间复杂度尽可能低。
本题本质上是一个三数之和问题,目标是找到三个数,它们的和尽量接近一个目标值(即预算)。
解题步骤
输入:
首先读取一组价格(以逗号分隔的数字),然后读取预算值。
处理:
边界条件检查: 如果价格数组长度小于 3,无法选择 3 个价格,直接返回 -1。
排序: 对价格进行排序,方便使用双指针法优化搜索过程。
双指针法:
外层循环固定一个价格 prices[i],然后使用双指针法在剩余的部分(即 prices[i+1] 到 prices[prices.length-1])中寻找另外两个价格,使得三个数的和尽量接近预算。
双指针调整:
计算当前三数之和 total = prices[i] + prices[left] + prices[right]。
如果当前和等于预算,返回该值。
如果当前和小于预算,说明当前的组合太小,增加 left(即选择更大的数)。
如果当前和大于预算,说明当前的组合太大,减小 right(即选择更小的数)。
记录最大合法值: 如果没有找到正好等于预算的组合,记录最大合法组合 maxSpend,最终返回该值。
输出:
输出最大合法组合的总价,如果没有找到有效的组合,则返回 -1
import java.util.Arrays;
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
Integer[] prices = Arrays.stream(sc.nextLine().split(","))
.map(Integer::parseInt)
.toArray(Integer[]::new);
int budget = Integer.parseInt(sc.nextLine());
System.out.println(maxSpending(prices, budget));
}
public static int maxSpending(Integer[] prices, int budget) {
if (prices.length < 3) return -1;
Arrays.sort(prices);
int maxSpend = -1;
for (int i = 0; i < prices.length - 2; i++) {
int left = i + 1;
int right = prices.length - 1;
while (left < right) {
int total = prices[i] + prices[left] + prices[right];
if (total == budget) {
return total;
}
else if (total < budget) {
maxSpend = Math.max(maxSpend, total);
left++;
}
else {
right--;
}
}
}
return maxSpend;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
C++解法
- 解题思路
- 本题的目标是从一个价格列表中选择三个数,使得这三数之和尽量接近给定的预算(不超过预算)。要求输出三个数的和,如果没有符合条件的组合,返回 -1。
我们可以通过以下步骤来解决这个问题:
输入处理:
第一行是一个逗号分隔的字符串,表示各个商品的价格。
第二行是预算值,表示消费者能接受的最大支出。
问题分析:
由于要从多个价格中选择三个数,我们可以将问题转化为三数之和问题,目标是找到三个数,它们的和尽量接近预算,并且不超过预算。
可以使用双指针法来优化搜索。排序后,固定一个元素,使用双指针扫描剩下的部分,快速找到符合条件的三数之和。
算法步骤:
排序: 首先对价格数组进行升序排序,排序的目的是为了后续使用双指针法来高效寻找三数之和。
双指针法:
外层循环遍历每个元素,将其作为当前三数之和的第一个数。
内层使用两个指针,left 指向当前元素的下一个位置,right 指向数组的最后一个位置。通过调整 left 和 right 来寻找与当前元素配对的另两个数,使得三数之和尽量接近预算。
更新最大值: 如果找到的三数之和小于预算,更新最大合法支出;如果三数之和等于预算,直接返回该值。
如果没有找到合适的组合,返回最大合法支出或 -1
#include
#include
#include
#include
using namespace std;
int findMaxSpending(vector<int>& prices, int budget) {
if (prices.size() < 3) return -1;
int maxSpend = -1;
sort(prices.begin(), prices.end());
for (int i = 0; i < prices.size() - 2; i++) {
int left = i + 1;
int right = prices.size() - 1;
while (left < right) {
int sum = prices[i] + prices[left] + prices[right];
if (sum == budget) {
return sum;
}
else if (sum < budget) {
maxSpend = max(maxSpend, sum);
left++;
}
else {
right--;
}
}
}
return maxSpend == -1 ? -1 : maxSpend;
}
int main() {
string pricesInput;
getline(cin, pricesInput);
string budgetInput;
getline(cin, budgetInput);
vector<int> prices;
stringstream ss(pricesInput);
string temp;
while (getline(ss, temp, ',')) {
prices.push_back(stoi(temp));
}
int budget = stoi(budgetInput);
cout << findMaxSpending(prices, budget) << endl;
return 0;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
解决方案
输入处理:
第一行是一个逗号分隔的字符串,表示商品的价格列表。
第二行是一个整数,表示预算。
核心思路:
三数之和问题:这是一个经典的算法问题,目标是从给定的数组中选出三个数,使得它们的和尽量接近目标值(即预算)。
排序 + 双指针法:首先将价格数组排序。然后固定一个元素(作为当前三数和的第一个元素),再通过双指针法来寻找另外两个元素,使得三数之和尽量接近预算。
双指针法的具体步骤:固定当前元素后,使用两个指针分别指向当前元素之后的位置和数组的最后位置:
如果三数之和小于预算,则增加左指针(左边的数增大可能使和更接近预算)。
如果三数之和大于预算,则减少右指针(右边的数减小可能使和更接近预算)。
每次都计算当前三数之和,并记录一个合法的最大和。
返回结果: 如果找到符合条件的组合,则返回最大合法和。如果没有找到,返回 -1
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
let lines = [];
rl.on("line", (input) => {
lines.push(input);
if (lines.length === 2) {
const prices = lines[0].split(",").map(Number);
const budget = Number(lines[1]);
console.log(findMaxSpending(prices, budget));
lines = [];
}
});
function findMaxSpending(prices, budget) {
if (prices.length < 3) return -1;
let maxSpend = -1;
prices.sort((a, b) => a - b);
for (let i = 0; i < prices.length - 2; i++) {
let left = i + 1;
let right = prices.length - 1;
while (left < right) {
const sum = prices[i] + prices[left] + prices[right];
if (sum === budget) {
return sum;
}
else if (sum < budget) {
maxSpend = Math.max(maxSpend, sum);
left++;
}
else {
right--;
}
}
}
return maxSpend === -1 ? -1 : maxSpend;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: