zoukankan      html  css  js  c++  java
  • HDU4784 Dinner Coming Soon(dp)

    当时区域赛的一道题。题意大概是这样的,有一个1~N的图,然后你要从1-》N,其中每经过一条边需要消耗你的时间和金钱,每到一个地方可以选择什么都不做,或者买一包盐,卖一包盐,身上不能同时有超过B包盐,然后你还可以从当前所在的世界i转道(i+1)modK的世界,每次转换花费时间1,然后问你在T时间内到达n点的时候你最多能有多少钱。 那个时候不会做是因为不知道如何控制dp的方向,我记得那个时候我已经把状态写好了,转移也是知道的,就是不知道转移的方向,后来发现时间的那一维是降的,于是我们就可以建一个优先队列,把一开始的状态压进去,然后拓展新的状态,这样就可以做出来了,调了很久,因为我心里认为它是无向图。。。。。吸取教训呀要!

    #pragma warning(disable:4996)
    #include<iostream>
    #include<cstring>
    #include<cstdio>
    #include<string>
    #include<algorithm>
    #include<queue>
    #include<cmath>
    #define maxn 150
    using namespace std;
    
    int n, m, B, K, R, T;
    
    int p[6][maxn];
    
    struct Edge
    {
    	int u, v, t, m;
    	Edge(){}
    	Edge(int ui, int vi, int ti, int mi) :u(ui), v(vi), t(ti), m(mi){}
    };
    
    vector<Edge> G[maxn];
    
    int dp[6][maxn][5][250]; // dp[K][N][B][T] k维空间,第n个点,有b袋盐,剩余时间为T时的最多有多少钱
    bool vis[6][maxn][5][250];
    struct Node
    {
    	int k, n, b, t;
    	Node(){}
    	Node(int ki, int ni, int bi, int ti) :k(ki), n(ni), b(bi), t(ti){}
    	bool operator < (const Node X) const{
    		return t < X.t;
    	}
    };
    
    
    int main()
    {
    	int cas; cin >> cas; int ca = 0;
    	while (cas--)
    	{
    		scanf("%d%d%d%d%d%d", &n, &m, &B, &K, &R, &T);
    		for (int i = 0; i < K; i++){
    			for (int j = 1; j <= n; j++){
    				scanf("%d", &p[i][j]);
    			}
    		}
    		for (int i = 0; i <= n; i++) G[i].clear();
    		int ui, vi, ti, mi;
    		for (int i = 0; i < m; i++){
    			scanf("%d%d%d%d", &ui, &vi, &ti, &mi);
    			G[ui].push_back(Edge(ui, vi, ti, mi));
    		}
    		memset(dp, -1, sizeof(dp));
    		memset(vis, 0, sizeof(vis));
    		dp[0][1][0][T] = R;
    		vis[0][1][0][T] = true;
    		priority_queue<Node> que;
    		que.push(Node(0, 1, 0, T));
    		while (!que.empty()){
    			Node nxt = que.top(); que.pop();
    			if (nxt.k == 0 && nxt.n == n) continue;
    			int val = dp[nxt.k][nxt.n][nxt.b][nxt.t];
    			for (int i = 0; i < G[nxt.n].size(); i++){
    				Edge vv = G[nxt.n][i];
    				if (nxt.k != 0 && (vv.v == 1 || vv.v == n)) continue;
    				if (nxt.t >= vv.t&&val >= vv.m){
    					dp[nxt.k][vv.v][nxt.b][nxt.t - vv.t] = max(dp[nxt.k][vv.v][nxt.b][nxt.t - vv.t], val - vv.m);
    					if (!vis[nxt.k][vv.v][nxt.b][nxt.t - vv.t]){
    						que.push(Node(nxt.k, vv.v, nxt.b, nxt.t - vv.t));
    						vis[nxt.k][vv.v][nxt.b][nxt.t - vv.t] = true;
    					}
    					if (vv.v == 1 || vv.v == n) continue;
    					if (nxt.b < B && (val - vv.m) >= p[nxt.k][vv.v]){
    						dp[nxt.k][vv.v][nxt.b + 1][nxt.t - vv.t] = max(dp[nxt.k][vv.v][nxt.b + 1][nxt.t - vv.t],
    							val - vv.m - p[nxt.k][vv.v]);
    						if (!vis[nxt.k][vv.v][nxt.b + 1][nxt.t - vv.t]){
    							que.push(Node(nxt.k, vv.v, nxt.b + 1, nxt.t - vv.t));
    							vis[nxt.k][vv.v][nxt.b + 1][nxt.t - vv.t] = true;
    						}
    					}
    					if (nxt.b>0){
    						dp[nxt.k][vv.v][nxt.b - 1][nxt.t - vv.t] = max(dp[nxt.k][vv.v][nxt.b - 1][nxt.t - vv.t],
    							val - vv.m + p[nxt.k][vv.v]);
    						if (!vis[nxt.k][vv.v][nxt.b - 1][nxt.t - vv.t]){
    							que.push(Node(nxt.k, vv.v, nxt.b - 1, nxt.t - vv.t));
    							vis[nxt.k][vv.v][nxt.b - 1][nxt.t - vv.t] = true;
    						}
    					}
    				}
    			}
    			if (nxt.t < 1) continue;
    			if (nxt.n == 1 || nxt.n == n) continue;
    			dp[(nxt.k + 1) % K][nxt.n][nxt.b][nxt.t - 1] = max(dp[(nxt.k + 1) % K][nxt.n][nxt.b][nxt.t - 1],
    				val);
    			if (!vis[(nxt.k + 1) % K][nxt.n][nxt.b][nxt.t - 1]){
    				que.push(Node((nxt.k + 1) % K, nxt.n, nxt.b, nxt.t - 1));
    				vis[(nxt.k + 1) % K][nxt.n][nxt.b][nxt.t - 1] = true;
    			}
    			if (nxt.b < B&&val >= p[(nxt.k + 1) % K][nxt.n]){
    				dp[(nxt.k + 1) % K][nxt.n][nxt.b + 1][nxt.t - 1] = max(dp[(nxt.k + 1) % K][nxt.n][nxt.b + 1][nxt.t - 1],
    					val - p[(nxt.k + 1) % K][nxt.n]);
    				if (!vis[(nxt.k + 1) % K][nxt.n][nxt.b + 1][nxt.t - 1]){
    					que.push(Node((nxt.k + 1) % K, nxt.n, nxt.b + 1, nxt.t - 1));
    					vis[(nxt.k + 1) % K][nxt.n][nxt.b + 1][nxt.t - 1] = true;
    				}
    			}
    			if (nxt.b>0){
    				dp[(nxt.k + 1) % K][nxt.n][nxt.b - 1][nxt.t - 1] = max(dp[(nxt.k + 1) % K][nxt.n][nxt.b - 1][nxt.t - 1],
    					val + p[(nxt.k + 1) % K][nxt.n]);
    				if (!vis[(nxt.k + 1) % K][nxt.n][nxt.b - 1][nxt.t - 1]){
    					que.push(Node((nxt.k + 1) % K, nxt.n, nxt.b - 1, nxt.t - 1));
    					vis[(nxt.k + 1) % K][nxt.n][nxt.b - 1][nxt.t - 1] = true;
    				}
    			}
    		}
    		int ans = -1;
    		for (int i = 0; i <= B; i++){
    			for (int j = 0; j <= T; j++){
    				ans = max(ans, dp[0][n][i][j]);
    			}
    		}
    		printf("Case #%d: ", ++ca);
    		if (ans == -1) puts("Forever Alone");
    		else printf("%d
    ", ans);
    	}
    	return 0;
    }
    
  • 相关阅读:
    GPG实践
    keepass口令管理实践
    2019-2020 20191204《信息安全导论》第十一周学习总结
    2019-2020 20191204第十周学习总结
    2019-2020 20191211第八周学习总结
    熟悉编程语言
    斐波那契数列
    辗转相除法
    20191210_获奖感言及学习总结
    2019-2020-1 20191210第11周学习总结
  • 原文地址:https://www.cnblogs.com/chanme/p/3612622.html
Copyright © 2011-2022 走看看