• CF865D Buy Low Sell High(反悔贪心)


    自己想了好一会,AC后看了下好像和网上挺多人思路不太一样(但本质是一样的),所以就来写这篇题解 

    首先这题之所以能反悔的根本原因和性质在于你在第i天买股票,第j天卖出,可以拆成第i天买股票,第k(i <= k <= j)天卖出和第k天买股票,第j天卖出两个过程(I)

    我们首先可以从大到小倒序扫描,假设当前扫到某个数i,然后贪心地想如果后面的最大值可以大于当前这个值,那就买入这个数,在最大值的位置卖出,这时我们要分两种情况考虑,

    (1)

    如果后面那个数(即最大值,记为k)已经买入过了,那么意味它后面肯定还有一个比它更大的值j,于是我们可以根据性质I,把i塞入堆里,标记为已经买过,而把k更新为标记没有买入过,重新塞入堆里.

    (2)

    如果这个数k没有被买入过,那么我们应该直接把它弹出(因为它不能像性质I那样作为中转点,所以只能卖出),同时把i标记为已经买入过,塞入堆里

    代码如下

    /*CF865D Buy Low Sell High*/
    #include<cstdio>
    #include<iostream>
    #include<cstring>
    #include<algorithm>
    #include<queue>
    using namespace std;
    #define ll long long
    struct Num{
    	int v,op;
    };
    bool operator < (Num A,Num B) {
    	return A.v < B.v;
    }
    priority_queue<Num>q;
    int read(){
    	char c = getchar();
    	int x = 0;
    	while(c < '0' || c > '9')		c = getchar();
    	while(c >= '0' && c <= '9')		x = x * 10 + c - 48,c = getchar();
    	return x;
    }
    ll ans = 0;
    const int maxn = 1e6 + 10;
    int p[maxn];
    int main()
    {
    	int n = read();
    	for(int i = 1; i <= n; ++i)
    		p[i] = read();
    	for(int i = n; i >= 1; --i){
    		if(i == n){
    			q.push(Num{p[i],1});
    			continue;
    		}
    		Num x = q.top();
    		if(x.v <= p[i]){
    			q.push(Num{p[i],1});
    		}
    		else{
    			if(x.op == -1){
    				int v = x.v;
    				q.pop();
    				q.push(Num{v,1});
    				ans += v - p[i];
    				q.push(Num{p[i],-1});
    			}
    			else{
    				int v = x.v;
    				ans += v - p[i];
    				q.pop();
    				q.push(Num{p[i],-1});
    			}
    		}
    	}
    	cout<<ans<<endl;
    	return 0;
    }
    

      

  • 相关阅读:
    python可视化---plot()函数
    蜂鸣器的相关知识及简单应用
    将一个文件夹中多个视频的视频帧保存在多个文件夹下
    键盘按键控制程序的简单案例
    Tkinter模块的详细总结
    控制LED灯发光
    python socket 套接字编程 单进程服务器 实现多客户端访问
    python 报错RuntimeError: dictionary changed size during iteration
    HTTP请求行、请求头、请求体详解(转)
    python UDP套接字通信
  • 原文地址:https://www.cnblogs.com/y-dove/p/13562592.html
Copyright © 2020-2023  润新知