n(n<=18278)个串,第i个串Ti(Ti为纯小写字母串且长度不超过3),
得分Pi(-1e9<=Pi<=1e9),表示只要子串中出现一次Ti,就会获得Pi的得分
对于你可以构造的无限长的串S来说,S的最终得分,为其中每一个串出现次数*其得分之和
即,cnt[i]为S中与Pi相同的子串的个数
要求输出你可以构造出的非空S串的最大得分,无需输出S,
特别地,如果S可以无穷大,输出Infinity
AtCoder Beginner Contest 264 - _Backl1ght - 博客园
poj2949 Word Rings(图论/spfa判正环 最大平均环)_Code92007的博客-CSDN博客
18278=26+26*26+26*26*26
赛中暴力dp艹过去了,
dp[i][j][k]表示当前串长为i最后一个字母是j倒数第二个字母是k的最大收益,
暴力跑个2000轮,比较2000时的前缀最大值和1000时的前缀最大值,
若2000更大,说明可以到无穷,输出Infinity,否则输出前缀最大值
赛后学习一下别人的写法,
其实主要思路就是把dp的后两维变成图上的点,从而转换为最短路
Infinity即对应图上有正环的情形,spfa判环即可
这种相邻状态的转移,之前在欧拉回路(太鼓达人)的题里面也见过,典中典
例如从倒数第二个字母a,倒数第一个字母b,
转移到倒数第二个字母b,倒数第一个字母c的时候,
即视为状态从ab转移到了bc,
没有字母的初始状态可以视为##,而有一个字母的状态可以视为#x
这样状态是不超过27*27的,这700多个点的图上,最多18278条边,
O(n*m)跑BellmanFord/SPFA即可
- #include
- using namespace std;
- #define fi first
- #define se second
- typedef long long ll;
- typedef pair<int,int> P;
- const int N=27*27,M=N*27;
- int n,u,v,w,cnt[N];
- ll dis[N],a[M],mx;
- string s;
- bool vis[N];
- int f(string x){
- int n=x.size(),ans=0;
- for(int i=0;i
- int v=x[i]=='#'?0:(x[i]-'a'+1);
- ans=ans*27+v;
- }
- return ans;
- }
- bool spfa(int s){
- for(int i=0;i
-1e18; - mx=-1e18;
- queue<int>q;
- vis[s]=cnt[s]=1;
- dis[s]=0;
- q.push(s);
- while(!q.empty()){
- int u=q.front();
- q.pop();
- if(u!=s)mx=max(mx,dis[u]);//非空
- vis[u]=0;
- for(int i=1;i<=26;++i){
- int nu=u*27+i,v=nu%N;//## -> #a -> ab
- ll w=a[i]+(v/27?a[v]:0)+(nu/N?a[nu]:0);//第一个字母是#的,一定不包含代价
- if(dis[v]
- dis[v]=dis[u]+w;
- cnt[v]=cnt[u]+1;
- if(cnt[v]>N)return 1;
- if(!vis[v]){
- q.push(v);
- vis[v]=1;
- }
- }
- }
- }
- return 0;
- }
- int main(){
- cin>>n;
- for(int i=1;i<=n;++i){
- cin>>s>>w;
- int m=s.size();
- a[f(s)]=w;
- }
- if(spfa(f("##")))cout<<"Infinity"<
- else cout<
- return 0;
- }
-
相关阅读:
2022年美团春招真题:字符串排序
C++,内联函数、对象、类、this
【我不熟悉的css】07. css命名,bem规范,跟着组件库element-ui学习组件命名
Gstreamer中pad的链接
浅析 spring 事件驱动
为什么 AI 时代更应该 Learn in Public
PHP 有趣的函数与功能
五、Maven-单一架构案例(业务功能:批复奏折,业务功能:登录检查,打包部署)
Kafka【基础入门】
冒泡算法,leetcode第一题
-
原文地址:https://blog.csdn.net/Code92007/article/details/128049890