CCF认证 2017-12 商路
2018-02-06 22:35
330 查看
首先说明,这道题我只拿了60分,超时,还没有想到怎么优化或者更好的算法
如果有优化方法,其他算法或者建议,欢迎在评论区留言
接下来说我的方法:
因为要计算一个城市接下来通向哪里可以取得最大值,必须知道所有下级城市的商路能取得的最大值(我是这么认为的)
最下级城市的价值一定为0(已知),已知最大价值后就可以计算所有上级直接通向这里可以得到的价值,取最大的那个(保存在一个数组ans里)
如果一个城市的所有直接下级都处理过了,那么它的最大价值就已知了,就可以计算它的上级直接通向它能取得的价值
所以要从下级向上级计算,再用类似拓扑排序的方法,查看是否计算了所有直接下级
最后把所有的加起来,具体看代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
typedef long long ll;
const ll MOD=1e18;
const int N=1e5+10;
ll v
,f
,u
,s
;
ll ans
;
int vis
;
void init()
{
memset(ans,0,sizeof(ans));
memset(vis,0,sizeof(vis));
}
int main()
{
int T,n;
scanf("%d",&T);
while(T--)
{
init(); //初始化
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d%d%d",&u[i],&s[i],&v[i],&f[i]);
vis[u[i]]++; //vis数组保存这个点的直接下级数量
}
int q
,q0=0,q1=0;
for(int i=1;i<=n;i++) //这里类似拓扑排序,直接下级数量为零入队
if(!vis[i]) q[q1++]=i;
while(q0<q1)
{
int k=q[q0++];
int t=u[k],d=s[k];
vis[t]--;
if(!vis[t]) q[q1++]=t; //直接下级全都处理过了,入队
while(t) //寻找所有上级,计算上级城市到这里的获利,取最大值
{
ans[t]=max(ans[t],(ans[k]+v[t]-(f[t]-d)*(f[t]-d))); //ans里保存了以这个城市为开头的商路最大价值
d+=s[t];
t=u[t];
}
}
ll sum=0;
for(int i=1;i<=n;i++) sum=(sum+ans[i])%MOD; //计算总价值
printf("%lld\n",sum);
}
return 0;
}
如果有优化方法,其他算法或者建议,欢迎在评论区留言
接下来说我的方法:
因为要计算一个城市接下来通向哪里可以取得最大值,必须知道所有下级城市的商路能取得的最大值(我是这么认为的)
最下级城市的价值一定为0(已知),已知最大价值后就可以计算所有上级直接通向这里可以得到的价值,取最大的那个(保存在一个数组ans里)
如果一个城市的所有直接下级都处理过了,那么它的最大价值就已知了,就可以计算它的上级直接通向它能取得的价值
所以要从下级向上级计算,再用类似拓扑排序的方法,查看是否计算了所有直接下级
最后把所有的加起来,具体看代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
typedef long long ll;
const ll MOD=1e18;
const int N=1e5+10;
ll v
,f
,u
,s
;
ll ans
;
int vis
;
void init()
{
memset(ans,0,sizeof(ans));
memset(vis,0,sizeof(vis));
}
int main()
{
int T,n;
scanf("%d",&T);
while(T--)
{
init(); //初始化
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d%d%d",&u[i],&s[i],&v[i],&f[i]);
vis[u[i]]++; //vis数组保存这个点的直接下级数量
}
int q
,q0=0,q1=0;
for(int i=1;i<=n;i++) //这里类似拓扑排序,直接下级数量为零入队
if(!vis[i]) q[q1++]=i;
while(q0<q1)
{
int k=q[q0++];
int t=u[k],d=s[k];
vis[t]--;
if(!vis[t]) q[q1++]=t; //直接下级全都处理过了,入队
while(t) //寻找所有上级,计算上级城市到这里的获利,取最大值
{
ans[t]=max(ans[t],(ans[k]+v[t]-(f[t]-d)*(f[t]-d))); //ans里保存了以这个城市为开头的商路最大价值
d+=s[t];
t=u[t];
}
}
ll sum=0;
for(int i=1;i<=n;i++) sum=(sum+ans[i])%MOD; //计算总价值
printf("%lld\n",sum);
}
return 0;
}
相关文章推荐
- CCF认证 2017-12 行车路线
- CCF认证 2017-12 Crontab
- CCF认证 2017-12 行车路线
- CCF认证 2017-12 行车路线
- CCF认证 2017-12 最小差值
- CCF认证 2017-12 游戏
- 12/04/2017 - 12/10/2017
- 2017_12_05学习Java心得笔记
- [CCF BY C++]2017-12 游戏
- 2017 Multi-University Training Contest - Team 6 【solved:5 / 12】
- C语言趣味一百道 第16道 2017_12_18
- 日记2017,12,20
- 2017/12/28C语言笔记整理
- CCF认证 2017-09 JSON查询
- CCF认证 2015-12 画图
- webstorm 2017(2017.01更新)11 12 (注册,激活,破解,码,一起支持正版,最新可用)
- 2017 1 12
- 2017_12_15 js获取项目路径,js调用问题,jsp获取js传递url中参数
- 【BigHereo 12】-----The First Half Of My 2017
- 2017_7_12 CSS基础