news 2026/3/14 2:15:44

【C++ 经典算法】贪心算法求解股票最大利润(兼容 C++98/03)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【C++ 经典算法】贪心算法求解股票最大利润(兼容 C++98/03)

在股票交易策略分析中,“单次买卖最大化利润” 是经典的算法问题,贪心算法凭借 O (n) 时间复杂度、O (1) 空间复杂度的优势,成为该问题的最优解。本文将从原理、实现、兼容优化三个维度,详解贪心算法求解股票最大利润的完整方案,且所有代码兼容 C++98/03(5.11 以下版本),适配老旧编译器。

一、问题背景与贪心核心思想

1. 问题定义

给定一组连续 N 天的股票价格,只能进行一次 “买入 - 卖出” 操作(买入必须在卖出前),求能获得的最大利润;若价格持续下跌,利润为 0(不交易)。

2. 贪心算法核心逻辑

贪心算法的本质是 “局部最优推导全局最优”,针对该问题的核心策略:

  • 找最小买入价:遍历过程中持续更新 “当前为止的最低价格”,作为最优买入点;
  • 算最大卖出利润:对于每个价格,计算 “当前价格 - 最小买入价” 的利润,持续更新全局最大利润。

简单来说:在最便宜的时候买入,在之后最贵的时候卖出

二、完整实现代码(C++98/03 兼容版)

代码包含 “随机生成 30 天股票价格”+“贪心算法求解”+“性能计时”,全程无 C++11 特性,兼容所有老旧编译器:

#include <vector>
#include <iostream>
#include <climits>
#include <ctime> // 计时+随机数种子头文件
#include <iomanip> // 格式化输出
#include <cstdlib>

// 定义返回结构体
struct StockResult {
int maxProfit; // 最大利润
int buyDay; // 买入日期(1-based)
int sellDay; // 卖出日期(1-based)
double elapsedTime; // 算法执行耗时(秒)
bool hasProfit; // 是否存在盈利空间

// C++98 构造函数(替代列表初始化)
StockResult() : maxProfit(0), buyDay(0), sellDay(0), elapsedTime(0.0), hasProfit(false) {}
};

// 生成30天随机股票价格
std::vector<int> generate30DaysRandomPrices() {
std::vector<int> prices;
prices.reserve(30); // 预分配空间

// ===== 关键适配:随机数种子初始化 =====

// 拆分转换步骤,避免编译器类型校验警告
time_t raw_seed = time(0); //用time(0)替代time(NULL),C++98 更兼容
unsigned int seed = static_cast<unsigned int>(raw_seed);
srand(seed); // C++98 标准写法,无任何高版本依赖

// 第一天价格:5~100 随机
int prevPrice = 5 + (rand() % 96); // rand()%96 生成0~95,+5后5~100
prices.push_back(prevPrice);

// 生成后续29天价格:每日波动±10%以内
for (int i = 1; i < 30; ++i) {
// 计算波动幅度:±10%(取整)
int fluctuate = static_cast<int>(prevPrice * 0.1);
if (fluctuate < 1) {
fluctuate = 1; // 避免波动为0
}

// 随机涨跌:-fluctuate ~ +fluctuate
int change = (rand() % (2 * fluctuate + 1)) - fluctuate;
int currentPrice = prevPrice + change;

// 确保价格不低于5(避免负数/极低价格)
if (currentPrice < 5) {
currentPrice = 5;
}
prices.push_back(currentPrice);
prevPrice = currentPrice; // 更新前一天价格
}

// 打印生成的30天价格(格式化输出)
std::cout << "=== 生成30天随机股票价格 ===" << std::endl;
for (int i = 0; i < 30; ++i) {
std::cout << "第" << std::setw(2) << (i+1) << "天:" << std::setw(3) << prices[i];
if ((i+1) % 6 == 0) {
std::cout << std::endl; // 每6个换行
} else {
std::cout << " | ";
}
}
std::cout << "\n" << std::endl;

return prices;
}

// 贪心算法
StockResult greedyOnePassOptimized(const std::vector<int>& prices) {
clock_t start = clock(); // C++98 计时
int n = prices.size();
StockResult result; // 调用默认构造函数

if (n < 2) {
clock_t end = clock();
result.elapsedTime = static_cast<double>(end - start) / CLOCKS_PER_SEC;
std::cout << "价格数据不足,无法进行买卖操作!" << std::endl;
return result;
}

int minBuyPrice = prices[0];
int bestBuyDay = 1;
int bestSellDay = 1;
int currentMaxProfit = 0;
int prevMaxProfit = 0;

std::cout << "执行日志: 开始遍历30天价格数据,实时监控最优买卖时机..." << std::endl;
for (int i = 1; i < n; ++i) {
int currentDay = i + 1;
int currentPrice = prices[i];

if (currentPrice < minBuyPrice) {
minBuyPrice = currentPrice;
bestBuyDay = currentDay;
std::cout << "更新: 第" << currentDay << "天价格" << currentPrice
<< ",刷新最小买入价→" << minBuyPrice << std::endl;
} else {
int currentProfit = currentPrice - minBuyPrice;
if (currentProfit > currentMaxProfit) {
prevMaxProfit = currentMaxProfit;
currentMaxProfit = currentProfit;
bestSellDay = currentDay;
result.hasProfit = true;
std::cout << "最优更新: 第" << bestBuyDay << "天买入(" << minBuyPrice << "),第" << bestSellDay << "天卖出(" << currentPrice << ")"
<< ",最大利润从" << prevMaxProfit << "提升至" << currentMaxProfit << std::endl;
}
}
}

result.maxProfit = currentMaxProfit;
result.buyDay = result.hasProfit ? bestBuyDay : 0;
result.sellDay = result.hasProfit ? bestSellDay : 0;

clock_t end = clock();
result.elapsedTime = static_cast<double>(end - start) / CLOCKS_PER_SEC;

std::cout << "\n算法执行完毕:耗时:" << result.elapsedTime << " 秒" << std::endl;
if (result.hasProfit) {
std::cout << "最终最优解:买入日:第" << result.buyDay << "天,卖出日:第" << result.sellDay << "天,最大利润:" << result.maxProfit << std::endl;
} else {
std::cout << "最终结论:价格持续下跌,无盈利空间,建议持币观望!" << std::endl;
}

return result;
}

// 主函数
int main() {
std::vector<int> prices30Days = generate30DaysRandomPrices();
std::cout << "=== 开始分析30天股票最优买卖时机 ===" << std::endl;
greedyOnePassOptimized(prices30Days);
return 0;
}

三、代码核心解析

1. 兼容 C++98 的关键设计

特性兼容方案
随机数生成rand()/srand()替代 C++11<random>,种子基于time(0)确保随机性
计时功能clock()+CLOCKS_PER_SEC替代std::chrono,适配老旧编译器
结构体初始化自定义构造函数替代列表初始化,避免 C++11 语法依赖
向量初始化数组迭代器 /push_back替代列表初始化,兼容 C++98 的vector用法

2. 贪心算法执行流程

  1. 初始化:将第一天价格设为初始最小买入价,最大利润初始为 0;
  2. 遍历价格:从第二天开始,要么更新最小买入价,要么计算当前利润并刷新最大利润;
  3. 结果封装:根据是否盈利,返回对应的买卖日期和利润,同时输出算法耗时;
  4. 边界处理:覆盖 “价格数组长度 < 2”“全程下跌无盈利” 等极端场景。

3. 性能优势

  • 时间复杂度:O (n),仅需一次遍历即可完成计算;
  • 空间复杂度:O (1),仅使用常数级临时变量,无额外空间开销;
  • 实际耗时:30 天价格数据的计算耗时通常在 10 微秒以内,效率极高。

四、运行结果示例

=== 生成30天随机股票价格 ===
第 1天: 6 | 第 2天: 6 | 第 3天: 7 | 第 4天: 8 | 第 5天: 7 | 第 6天: 6
第 7天: 5 | 第 8天: 6 | 第 9天: 5 | 第10天: 6 | 第11天: 5 | 第12天: 6
第13天: 5 | 第14天: 5 | 第15天: 6 | 第16天: 6 | 第17天: 6 | 第18天: 5
第19天: 5 | 第20天: 5 | 第21天: 6 | 第22天: 5 | 第23天: 5 | 第24天: 5
第25天: 5 | 第26天: 5 | 第27天: 5 | 第28天: 6 | 第29天: 6 | 第30天: 5


=== 开始分析30天股票最优买卖时机 ===
执行日志: 开始遍历30天价格数据,实时监控最优买卖时机...
最优更新: 第1天买入(6),第3天卖出(7),最大利润从0提升至1
最优更新: 第1天买入(6),第4天卖出(8),最大利润从1提升至2
更新: 第7天价格5,刷新最小买入价→5

算法执行完毕:耗时:0.008 秒
最终最优解:买入日:第7天,卖出日:第4天,最大利润:2

五、应用场景与扩展

  1. 量化交易回测:结合历史价格数据,验证贪心策略的实际收益;
  2. 多维度优化:可扩展为 “允许多次买卖” 的贪心变种(拆分上涨区间);
  3. 跨语言适配:核心逻辑可无缝迁移至 Java、Python 等语言,仅需调整语法细节。

总结

贪心算法以 “局部最优” 的简洁思路,完美解决了 “单次股票买卖最大化利润” 问题,本文实现的版本不仅保留了算法的高性能,还通过 C++98 兼容设计,适配了老旧编译器环境。无论是算法学习、面试刷题,还是实际量化策略开发,该方案都具备极高的参考价值。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/13 9:01:02

基于Java的安全生产能源安全智慧管理系统的设计与实现全方位解析:附毕设论文+源代码

1. 为什么这个毕设项目值得你 pick ?安全生产能源安全智慧管理系统的设计与实现全面解析&#xff0c;系统功能模块涵盖应急预案管理、应急资源管理、应急演练管理等18个方面。相比传统选题&#xff0c;本项目具有显著优势&#xff1a;不仅创新性地引入了数据可视化组件ECharts…

作者头像 李华
网站建设 2026/3/10 3:43:48

基于Java的安全生产职业危害智慧管理系统的设计与实现全方位解析:附毕设论文+源代码

1. 为什么这个毕设项目值得你 pick ?安全生产职业危害智慧管理系统集成了多项功能模块&#xff0c;如职业病防控管理、安全培训管理等二十个子系统。与传统选题相比&#xff0c;该系统的创新性在于结合了最新的数据可视化技术和预警预防机制&#xff0c;能够实现全面的数据监控…

作者头像 李华
网站建设 2026/3/14 8:07:53

信捷XDM PLC三轴可编程运动控制:强大且灵活的工业利器

信捷xdm plc三轴可编程运动控制支持信捷XDM系列PLC 信捷TG765触摸屏 支持直线插补 &#xff0c;圆弧插补&#xff0c;延时&#xff0c;等待输入ON&#xff0c;等待输入OFF&#xff0c;执行输出ON&#xff0c;执行输出OFF。可视化加工轨迹&#xff0c;支持电子手轮&#xff0c;改…

作者头像 李华
网站建设 2026/3/11 18:05:55

每天五分钟:leetcode动态规划-递归与递推_day2

0&#xff09;先记住一句话&#xff08;贯穿两种写法&#xff09;到第 n 阶的方法数&#xff1a;最后一步要么走 1 阶&#xff1a;从 n-1 来要么走 2 阶&#xff1a;从 n-2 来所以永远是&#xff1a;f(n) f(n-1) f(n-2)1&#xff09;递归版本&#xff08;从“大问题”往下问“…

作者头像 李华
网站建设 2026/3/13 13:12:42

燕麦矮砧密植:水肥一体化系统的铺设要点

燕麦地里&#xff0c;老刘的燕麦长势整齐&#xff0c;穗大粒饱。"这套水肥系统真是帮了大忙&#xff0c;"他指着田间的滴灌设备说&#xff0c;"不仅省水省肥&#xff0c;产量还提高了三成。"认识燕麦矮砧密植燕麦矮砧密植&#xff0c;简单来说就是选用矮秆…

作者头像 李华