P3686 [CERC2016] 爵士之旅 Jazz Journey

题解

可以发现,对于巡演的路线 (x o y)(a o b),若 (min(x,y) eq min(a,b))(max(x,y) eq max(a,b)),则其答案互不影响。不妨设 (x<y),我们需要对每一种 ((x,y),(y,x)) 求出最小值。

将巡演路线中 (x o y) 记作 (()(y o x) 记作 ()),则它形成了一个仅由左右括号组成的序列。设 (A)(x o y) 的最优价钱,(B) 同理;(AB)(x o y o x) 的最优价钱,(BA) 同理。则有:(A=min(operatorname{cost}(x o y),AB),AB=min(A+B,operatorname{cost}(x o y o x)))(B,BA) 同理。

不妨设 (AB<BA),于是,最优策略是:不断地删去括号序列中的 (()) 子序列(不必连续),然后不断地删去 ()() 子序列,直到只剩下若干 (()())

(AB>BA),则只需要将所有左、右括号调换,然后 (A,B) 调换,(AB,BA) 调换,再执行上面的策略即可。

注意 (max(AB,BA)) 可能会达到 (2 imes 10^9),因此 (inf) 要开得大一点。

代码
#include <cstdio>
#include <cstring>
#include <cctype>
#include <algorithm>
#include <utility>
#include <map>
#include <string>
#include <stack>
using namespace std;
#define For(Ti,Ta,Tb) for(int Ti=(Ta);Ti<=(Tb);++Ti)
#define Dec(Ti,Ta,Tb) for(int Ti=(Ta);Ti>=(Tb);--Ti)
template<typename T> void Read(T &_x){
	_x=0;int _f=1;
	char ch=getchar();
	while(!isdigit(ch)) _f=(ch=='-'?-1:_f),ch=getchar();
	while(isdigit(ch)) _x=_x*10+(ch^48),ch=getchar();
	_x*=_f;
}
template<typename T,typename... Args> void Read(T &_x,Args& ...others){
	Read(_x);Read(others...);
}
typedef long long ll;
typedef pair<int,int> pii;
const ll Inf=0x3f3f3f3f3fLL;
const int N=3e5+5;
int n,d,m;
map<pii,string> mp;
void Insert(int x,int y){
	if(x<y) mp[{x,y}]+='A';
	else mp[{y,x}]+='B';
}
map<pii,ll> one,two;
void Insert2(map<pii,ll> &fli,int u,int v,int p){
	pii x(u,v);
	if(!fli.count(x)) fli.insert({x,p});
	else fli[x]=min(fli[x],ll(p));
}
ll Find(const map<pii,ll> &fli,int u,int v){
	pii x(u,v);
	if(!fli.count(x)) return Inf;
	return fli.at(x);
}
int rem[N];
ll Calc(const string &s,ll cost,char c){
	ll res=0;
	stack<int> stk;
	for(int i=0;i<int(s.size());++i){
		if(rem[i]) continue;
		if(s[i]==c) stk.push(i);
		else{
			if(stk.empty()) continue;
			res+=cost,rem[stk.top()]=rem[i]=1;
			stk.pop();
		}
	}
	return res;
}
int main(){
	Read(n,d);
	int pre;
	For(i,1,d){
		int x;Read(x);
		if(i>1) Insert(pre,x);
		pre=x;
	}
	Read(m);
	For(i,1,m){
		int u,v,p;char temp[5];
		Read(u,v);scanf("%s",temp);Read(p);
		if(temp[0]=='O') Insert2(one,u,v,p);
		else Insert2(two,u,v,p);
	}
	ll ans=0;
	for(const auto &pr:mp){
		int x=pr.first.first,y=pr.first.second;
		string s=pr.second;
		for(int i=0;i<int(s.size());++i) rem[i]=0;
		ll A=Find(one,x,y),B=Find(one,y,x);
		ll AB=Find(two,x,y),BA=Find(two,y,x);
		A=min(A,AB),B=min(B,BA);
		AB=min(AB,A+B),BA=min(BA,A+B);
		if(BA<AB){
			for(char &c:s){
				if(c=='A') c='B';
				else c='A';
			}
			swap(A,B);swap(AB,BA);
		}
		ans+=Calc(s,AB,'A');
		ans+=Calc(s,BA,'B');
		for(int i=0;i<int(s.size());++i){
			if(rem[i]) continue;
			if(s[i]=='A') ans+=A;
			else ans+=B;
		}
	}
	printf("%lld
",ans);
	return 0;
}
Written by Alan_Zhao
原文地址:https://www.cnblogs.com/alan-zhao-2007/p/p3686-sol.html