• AtCoder Beginner Contest 264 G.String Fair(最短路/暴力dp 补写法)


    题目

    n(n<=18278)个串,第i个串Ti(Ti为纯小写字母串且长度不超过3),

    得分Pi(-1e9<=Pi<=1e9),表示只要子串中出现一次Ti,就会获得Pi的得分

    对于你可以构造的无限长的串S来说,S的最终得分,为其中每一个串出现次数*其得分之和

    \large \sum_{i=1}^{n}cnt[i]*P[i],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即可

    代码

    1. #include
    2. using namespace std;
    3. #define fi first
    4. #define se second
    5. typedef long long ll;
    6. typedef pair<int,int> P;
    7. const int N=27*27,M=N*27;
    8. int n,u,v,w,cnt[N];
    9. ll dis[N],a[M],mx;
    10. string s;
    11. bool vis[N];
    12. int f(string x){
    13. int n=x.size(),ans=0;
    14. for(int i=0;i
    15. int v=x[i]=='#'?0:(x[i]-'a'+1);
    16. ans=ans*27+v;
    17. }
    18. return ans;
    19. }
    20. bool spfa(int s){
    21. for(int i=0;i-1e18;
    22. mx=-1e18;
    23. queue<int>q;
    24. vis[s]=cnt[s]=1;
    25. dis[s]=0;
    26. q.push(s);
    27. while(!q.empty()){
    28. int u=q.front();
    29. q.pop();
    30. if(u!=s)mx=max(mx,dis[u]);//非空
    31. vis[u]=0;
    32. for(int i=1;i<=26;++i){
    33. int nu=u*27+i,v=nu%N;//## -> #a -> ab
    34. ll w=a[i]+(v/27?a[v]:0)+(nu/N?a[nu]:0);//第一个字母是#的,一定不包含代价
    35. if(dis[v]
    36. dis[v]=dis[u]+w;
    37. cnt[v]=cnt[u]+1;
    38. if(cnt[v]>N)return 1;
    39. if(!vis[v]){
    40. q.push(v);
    41. vis[v]=1;
    42. }
    43. }
    44. }
    45. }
    46. return 0;
    47. }
    48. int main(){
    49. cin>>n;
    50. for(int i=1;i<=n;++i){
    51. cin>>s>>w;
    52. int m=s.size();
    53. a[f(s)]=w;
    54. }
    55. if(spfa(f("##")))cout<<"Infinity"<
    56. else cout<
    57. return 0;
    58. }

  • 相关阅读:
    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