• “蔚来杯“2022牛客暑期多校训练营3


    "蔚来杯"2022牛客暑期多校训练营3_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ牛客ACM提高训练营是面向ICPC/CCPC/CSP/NOI比赛备战的选手,由金牌选手命题,提高训练高难度的编程比赛,ACM/NOI拔高训练营。icon-default.png?t=M666https://ac.nowcoder.com/acm/contest/33188

    补不完啊,好多,还要学东西,继续努力

    A.Ancestor(LCA)

    题意:给你一个长度k数组,里面记录了点的编号,有两棵树,两棵树每个点都有权值.我们每次在长为k的数组里删除一个点(每次删除后计算了会加回来)并且看其他点的lca(最近公共祖先),是否保证这两个祖先a的值大于b的值.输出删点方案个数

    思路:直接对于两棵树都维护一个前缀lca和后缀lca,遍历k数组,选一个点,用前后缀lca求出其他点的lca(就把这点之前的前缀lca和后面的后缀lca进行求lca,判断两棵树上两个lca值大小即可)

    1. #include
    2. using namespace std;
    3. const int N=100100;
    4. int numk[N],vala[N],valb[N];
    5. int qiana[N],houa[N];
    6. int qianb[N],houb[N];
    7. int ansa[N],ansb[N];
    8. vector<int>to[N];
    9. int si[N],deep[N],fa[N],top[N],son[N];
    10. void dfs1(int x,int fa1)
    11. {
    12. si[x]=1,deep[x]=deep[fa1]+1;
    13. son[x]=0,fa[x]=fa1;
    14. for(int i=0;isize();i++)
    15. {
    16. int tt=to[x][i];
    17. if(tt==fa[x])continue;
    18. dfs1(tt,x);
    19. si[x]+=si[tt];
    20. if(si[son[x]]
    21. }
    22. return ;
    23. }
    24. void dfs2(int x,int tx)
    25. {
    26. top[x]=tx;
    27. if(son[x]!=0)dfs2(son[x],top[x]);
    28. for(int i=0;isize();i++)
    29. {
    30. if(to[x][i]!=fa[x]&&to[x][i]!=son[x])
    31. {
    32. dfs2(to[x][i],to[x][i]);
    33. }
    34. }
    35. return ;
    36. }
    37. int lca(int x,int y)
    38. {
    39. while(top[x]!=top[y])
    40. {
    41. if(deep[top[x]]swap(x,y);
    42. x=fa[top[x]];
    43. }
    44. return deep[x]
    45. }
    46. int main()
    47. {
    48. int fa,t,u,v,n,k;
    49. scanf("%d%d",&n,&k);
    50. for(int i=1;i<=k;i++)
    51. scanf("%d",&numk[i]);
    52. for(int i=1;i<=n;i++)
    53. scanf("%d",&vala[i]);
    54. for(int i=2;i<=n;i++)
    55. {
    56. scanf("%d",&fa);
    57. to[fa].push_back(i);
    58. }
    59. dfs1(1,0);
    60. dfs2(1,1);
    61. qiana[1]=numk[1];
    62. int f=numk[1];
    63. for(int i=2;i<=k;i++)
    64. {
    65. f=lca(f,numk[i]);
    66. qiana[i]=f;
    67. }
    68. houa[k]=numk[k];
    69. f=numk[k];
    70. for(int i=k-1;i>=1;i--)
    71. {
    72. f=lca(f,numk[i]);
    73. houa[i]=f;
    74. }
    75. ansa[1]=houa[2],ansa[k]=qiana[k-1];
    76. for(int i=2;i<=k-1;i++)
    77. {
    78. ansa[i]=lca(qiana[i-1],houa[i+1]);
    79. }
    80. //
    81. for(int i=1;i<=n;i++)
    82. to[i].clear();
    83. //
    84. for(int i=1;i<=n;i++)
    85. scanf("%d",&valb[i]);
    86. for(int i=2;i<=n;i++)
    87. {
    88. scanf("%d",&fa);
    89. to[fa].push_back(i);
    90. }
    91. dfs1(1,0);
    92. dfs2(1,1);
    93. qianb[1]=numk[1];
    94. f=numk[1];
    95. for(int i=2;i<=k;i++)
    96. {
    97. f=lca(f,numk[i]);
    98. qianb[i]=f;
    99. }
    100. houb[k]=numk[k];
    101. f=numk[k];
    102. for(int i=k-1;i>=1;i--)
    103. {
    104. f=lca(f,numk[i]);
    105. houb[i]=f;
    106. }
    107. ansb[1]=houb[2],ansb[k]=qianb[k-1];
    108. for(int i=2;i<=k-1;i++)
    109. {
    110. ansb[i]=lca(qianb[i-1],houb[i+1]);
    111. }
    112. int ans=0;
    113. for(int i=1;i<=k;i++)
    114. {
    115. if(vala[ansa[i]]>valb[ansb[i]])
    116. ans++;
    117. }
    118. printf("%d",ans);
    119. return 0;
    120. }

    C.Concatenation

    题意:给你n个字符串,求出将字符串排列后字典序最小的串.

    思路:本来一开始想tire+dfs序写的,感觉挺麻烦,互让想到之前洛谷一个原题,直接进行sort排序即可

    1. #include
    2. using namespace std;
    3. string s[2000006];
    4. bool cmp(string &a,string &b)
    5. {
    6. return a+b
    7. //这里保证相邻的两个string排列起来肯定是字典序最小的
    8. }
    9. void solve()
    10. {
    11. int n;
    12. cin>>n;
    13. for(int i=1;i<=n;i++)
    14. cin>>s[i];
    15. sort(s+1,s+1+n,cmp);
    16. for(int i=1;i<=n;i++)
    17. cout<
    18. cout<
    19. }
    20. signed main()
    21. {
    22. cin.tie(0);
    23. cout.tie(0);
    24. ios::sync_with_stdio(0);
    25. solve();
    26. return 0;
    27. }

    J.Journey

    题意:NIO 和 Desprado2 是好朋友,他们住在同一个城市。 不过,每次蔚来开车去desprado2,等红灯的时间都比较长,蔚来对此非常苦恼。 他们的城市有许多十字路口。 蔚来是个很倒霉的家伙,每次直行、左转或在十字路口掉头,都会遇到红灯,必须等待。 十字路口右转不用等红灯。蔚来讨厌红灯,所以他想知道自己最少会遇到多少个红灯。 你能帮助他吗?

    思路:这题题面挺恶心的.我们看着就是最短路,这个时候就考虑图怎么建,其实可以不建图,把每个点的位置和朝向当作一个图上的点,然后进行最短路求法即可.这里因为边权都为1(每次等红灯和不等就是作为边权,0,1).可以采用01bfs(有deque模拟优先队列省去了优先队列复杂度,具体实现就是把0边权到达的点存在队首,边权1更新的点放在队尾,这样保证边权长度一定是和优先队列一样的,就是用deque代替优先队列).

    1. #include
    2. using namespace std;
    3. typedef pair<int,int> pii;
    4. const int N=500010;
    5. int g[N][4];
    6. int sx,sy,tx,ty;
    7. mapint>ma;
    8. struct node
    9. {
    10. int x,y,step;
    11. };
    12. void __01bfs()
    13. {
    14. dequequ;
    15. qu.push_back({sx,sy,0});
    16. while(qu.size())
    17. {
    18. auto [x,y,step]=qu.front();
    19. qu.pop_front();
    20. if(x==tx&&y==ty)
    21. {
    22. printf("%d\n",step);
    23. return ;
    24. }
    25. if(ma[{x,y}]==1)
    26. continue;
    27. ma[{x,y}]=1;
    28. int right=-1;
    29. for(int i=0;i<4;i++)
    30. if(g[y][i]==x)
    31. right=(i+1)%4;
    32. for(int i=0;i<4;i++)
    33. {
    34. if(i==right)
    35. qu.push_front({y,g[y][i],step});
    36. else
    37. qu.push_back({y,g[y][i],step+1});
    38. }
    39. }
    40. printf("-1\n");
    41. return ;
    42. }
    43. void solve()
    44. {
    45. int n;
    46. scanf("%d",&n);
    47. for(int i=1;i<=n;i++)
    48. scanf("%d%d%d%d",&g[i][0],&g[i][1],&g[i][2],&g[i][3]);
    49. scanf("%d%d%d%d",&sx,&sy,&tx,&ty);
    50. __01bfs();
    51. return ;
    52. }
    53. signed main()
    54. {
    55. cin.tie(0);
    56. cout.tie(0);
    57. ios::sync_with_stdio(0);
    58. solve();
    59. return 0;
    60. }

  • 相关阅读:
    创建ES索引
    探索隧道ip如何助力爬虫应用
    我的创作纪念日
    leetcode栈与队列(上)之理论篇(java实现)
    同一份数据全域共享,HashData UnionStore实时性背后的故事
    azure devops工具实践分析
    用于安装和维护光纤单模和多模的光纤网络测试套件
    最新 IntelliJ IDEA 旗舰版和社区版下载安装教程(图解)
    pandas|Task03索引
    [高等数学]同济版高等数学【第七版】上下册教材+习题全解PDF
  • 原文地址:https://blog.csdn.net/qq_49593247/article/details/126190945