思路:每个替代物品就可以降低成本,可以在代替物品和被代替物品之间连一条边,边权为折后价,然后判断从每个物品出发到达物品1的最小总成本是多少,所以可以创建一个超级源点,然后在超级源点与每个物品之间建一条边权为该点成本的边
关于等级的处理:物品1的等级一定在规定等级范围内,所以可以遍历所有包括物品1等级的等级范围然后在该进行一次有等级范围限制的dijkstra最后取min即可
ac代码:
#include<bits/stdc++.h>
#define endl '\n'
#define ll long long
#define INF 0x3f3f3f3f
#define pb push_back
#define int long long
// #define int unsigned long long
#define Mirai ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
using namespace std;
typedef pair<int,int> pii;
const int N=1e5+10;
int m,n,k,P,L,X,l,r;
vector<pii> g[N];
int dist[N];
bool vis[N];
int level[N];
void init()
{
for(int i=0;i<=n;i++)
{
dist[i]=INF;
vis[i]=false;
}
}
void dij(int l,int r)
{
priority_queue<pii,vector<pii>,greater<pii>> q;
init();
dist[0]=0;
q.push({dist[0],0});
while(q.size())
{
int u=q.top().second;
q.pop();
if(vis[u])continue;
vis[u]=true;
for(auto [v,w]:g[u])
{
if(dist[v]>dist[u]+w&&level[v]>=l&&level[v]<=r)
{
dist[v]=dist[u]+w;
q.push({dist[v],v});
}
}
}
}
void solve()
{
cin>>m>>n;
for(int i=1;i<=n;i++)
{
cin>>P>>level[i]>>X;
g[0].pb({i,P});//超级源点
for(int j=0;j<X;j++)
{
int v,w;
cin>>v>>w;
g[v].pb({i,w});
}
}
int ans=INF;
for(int i=level[1]-m;i<=level[1];i++)
//level[1]一定在等级范围内
//枚举所有由level[1]并且长度为m的区间
{
dij(i,i+m);
ans=min(ans,dist[1]);
}
cout<<ans<<endl;
}
signed main()
{
Mirai;
int T=1;
//cin>>T;
while(T--)
{
solve();
}
}