【luogu P3952 时间复杂度】题解

对于2017 D1 T2 这道题

实实在在是个码力题,非常考验耐心。

其实大体的思路并不是非常难想出来,但是要注意的小细节比较多。

题目链接:https://www.luogu.org/problemnew/show/P3952

思路

对于每一个程序,先读入L和O(),并将其中的时间复杂度抠出来。

其次整行读入字符串,即所给定的程序。

判断第一个字符是F or E

F i x y 需要把x y拿出来,把i压进栈

E 退栈 压进i后为了方便退栈及退栈时判断,用一个flag标记

每做完一个程序,与前面抠出来的时间复杂度对比判断yesnoerr即可。

注意

1.我的readx和ready函数比较暴力,直接截取常数和n可能存在的位置并保存。所以在考虑情况时,常数与n的位置不能搞错,也不能少考虑。

2.在每搞完一个程序时,要把初始值和标记都初始化一遍。

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <algorithm>
  5 #include <stack>
  6 using namespace std;
  7 int L,t,anso,codeo;//anso=0 O(1)  anso=x O(n^x) 输入给的时间复杂度   codeo 自己算的时间复杂度  if anso==codeo yes else no     if codeo==-1 err
  8 string code[105];//按行输入所给代码(F,E)
  9 string ans;//读入给定的复杂度 O()
 10 int readx(string x)
 11 {
 12     int num;
 13     if(x[4] >= ‘0‘ && x[4] <= ‘9‘) num = x[4]-‘0‘;
 14     if(x[5] >= ‘0‘ && x[5] <= ‘9‘) num = (x[4]-‘0‘)*10+x[5]-‘0‘;
 15     if(x[4] == ‘n‘) num = 1000000;
 16     return num;
 17 } //抠出给定的复杂度 x
 18 int ready(string x)
 19 {
 20     int num;
 21     if(x[6] >= ‘0‘ && x[6] <= ‘9‘ && x[7] <= ‘0‘ || x[7] >= ‘9‘) num = x[6]-‘0‘;
 22     if(x[7] >= ‘0‘ && x[7] <= ‘9‘ && x[8] <= ‘0‘ || x[8] >= ‘9‘) num = x[7]-‘0‘;
 23     if(x[6] >= ‘0‘ && x[6] <= ‘9‘ && x[7] >= ‘0‘ && x[7] <= ‘9‘) num = (x[6]-‘0‘)*10+x[7]-‘0‘;
 24     if(x[7] >= ‘0‘ && x[7] <= ‘9‘ && x[8] >= ‘0‘ && x[8] <= ‘9‘) num = (x[7]-‘0‘)*10+x[8]-‘0‘;
 25     if(x[6] == ‘n‘) num = 1000000;
 26     if(x[7] == ‘n‘) num = 1000000;
 27     return num;
 28 }//抠出给定的复杂度 y
 29 int check1()
 30 {
 31     int res;
 32     if(ans[2] == ‘1‘) res = 0;
 33         if(ans[2] == ‘n‘)
 34         {
 35             if(ans[4]<=‘9‘ && ans[4]>=‘0‘)
 36             res = ans[4]-‘0‘;
 37             if(ans[5]<=‘9‘ && ans[5]>=‘0‘)
 38             res = (ans[4]-‘0‘)*10 + ans[5]-‘0‘;
 39         }//记录输入给的复杂度
 40     return res;
 41 }
 42 int check2()
 43 {
 44         stack<int> s;
 45         int flag=-1;//标记
 46         bool fe[26]={0};//上面都是便于栈操作,fe来记录变量名
 47         int res=0,now=0;//now来记录当前循环中的时间复杂度,res是整个程序的时间复杂度
 48         bool cflag[26]={0};//记录变量是否重复   0 则没用过    1 用过  changeflag
 49         int xnum,ynum;
 50
 51     for(int i=1;i<=L;i++)
 52     {
 53
 54         if(code[i][0]==‘F‘)
 55         {
 56             int k=code[i][2]-‘a‘;
 57             if(cflag[k]) return -1;
 58             s.push(k);
 59             cflag[k] = 1;
 60
 61             xnum=readx(code[i]); ynum=ready(code[i]);
 62             if(ynum-xnum>1000)
 63             {
 64                 if(flag==-1)
 65                 {
 66                     now++;
 67                     res=max(res,now);
 68                     fe[k]=1;
 69                 }
 70             }
 71             if(xnum>ynum)
 72             {
 73                 if(flag==-1) flag=k;
 74             }
 75         }
 76
 77         if(code[i][0]==‘E‘)
 78         {
 79         if(s.empty()) return -1;
 80         int k=s.top();
 81         s.pop();cflag[k]=false;
 82         if(flag==k) flag=-1;
 83         if(fe[k])
 84             {
 85                 fe[k]=false;
 86                 now--;
 87             }
 88         }
 89     }
 90     if(s.size()) return -1;
 91     return res;
 92 }
 93 int main()
 94 {
 95     scanf("%d",&t);
 96     while(t--)
 97     {
 98
 99         scanf("%d ",&L);getline(cin,ans);
100         anso=check1();
101
102         for(int i=1;i<=L;i++)getline(cin,code[i]);
103         codeo=check2();
104
105         if(codeo==-1) cout<<"ERR"<<endl;
106         else
107         {
108             if(anso!=codeo)  cout<<"No"<<endl;
109             if(anso==codeo)  cout<<"Yes"<<endl;
110         }
111     }
112     return 0;
113 } 

原文地址:https://www.cnblogs.com/MisakaAzusa/p/8470118.html

时间: 2024-07-30 22:12:52

【luogu P3952 时间复杂度】题解的相关文章

luogu P3952 时间复杂度题解

显然这是一道大模拟 我们要做的就是读入一堆字符串,然后模拟这个循环. 定义某一层的复杂度为执行完这一层循环之后,消耗的复杂度. 某层循环的复杂度=\(max \{\)所有并列的下一层循环的复杂度\(\}\).通俗点说,就是在某层循环中有分支的时候,这一层的复杂度=\(max \{\)所有分支的复杂度\(\}+\)本层复杂度. 最后复杂度=\(max \{\)所有的第一层循环复杂度\(\}\) 考虑到会有分支,所以我们采用递归来实现(当然其本质是栈,但是我不会写). 由于要使程序不至于\(RE\)

洛谷P3952 时间复杂度

题目:https://www.luogu.org/problemnew/show/3952 题目描述 小明正在学习一种新的编程语言 A++,刚学会循环语句的他激动地写了好多程序并 给出了他自己算出的时间复杂度,可他的编程老师实在不想一个一个检查小明的程序, 于是你的机会来啦!下面请你编写程序来判断小明对他的每个程序给出的时间复杂度是否正确. A++语言的循环结构如下: F i x y 循环体 E 其中F i x y表示新建变量 i(变量 i 不可与未被销毁的变量重名)并初始化为 x, 然后判断 

邮递员送信(luogu 1629)题解

[问题描述] 有一个邮递员要送东西,邮局在节点1.他总共要送N-1样东西,其目的地分别是2~N.由于这个城市的交通比较繁忙,因此所有的道路都是单行的,共有M条道路,通过每条道路需要一定的时间.这个邮递员每次只能带一样东西.求送完这N-1样东西并且最终回到邮局最少需要多少时间. [样例输入] 5 10    2 3 5    1 5 5    3 5 6    1 2 8    1 3 8    5 3 4    4 1 8    4 5 3    3 5 6    5 4 2 [样例输出] 83

luogu 3952 时间复杂度

noip2017 D1T2 某zz选手考场上写了1.5h 考完之后发现自己写的是错的 但是结果A了??? 题目大意: 一种新的编程语言 A++ 给出一个程序只有循环语句 并给出这个程序的时间复杂度 判断每个程序给出的时间复杂度是否正确. A++语言的循环结构如下: F i x y 循环体 E 其中F i x y表示新建变量i(变量i不可与未被销毁的变量重名)并初始化为 x 然后判断 i 和 y 的大小关系,若 i 小于等于 y 则进入循环 否则不进入 每次循环结束后 i 都会被修改成 i +1,

洛谷 P3952 时间复杂度

做这道题的最大收获就是坚持不懈,勇往直前,这道题是一个"码力题",不能说无脑,但绝对恶心.总共花了3h+才调出来.果然当时NOIp放弃这道题是明智的 好了,闲话放一边,我们来搞一搞这道题. 这道题思路很简单,就是模拟.t<=10和L<=100也提示了我们时间不是问题,要大胆的去模拟.我是在线判断程序是否ERR,离线判断时间复杂度计算是否正确.如果程序ERR,就标记一下,输入完之后直接结束这次循环(continue),不再进行接下来的计算.至于判断Yes和No,我先将输入的时

luogu 3952 时间复杂度(模拟)

时间复杂度 这道题从两个月前开始做,一直没做出来,最后今晚决心一定要做出来.于是开始认真的在打草纸上写思路,最后在AC的那一刻,差点哭了出来!! 题目大意 这个自己看吧,noip2017的D1T2 solution 先介绍一下这道题我们用到的每个变量他们的用处 stack[]记录变量的循环层 vis[]记录变量在栈中是否出现过 cmp函数,这个可以用作比较循环中a和b的大小 \[ \begin{cases} a<b -> 进入新的一层循环\a>b -> 无法进入新的循环,循环终止\

Luogu P1342 请柬 题解

差不多是Dijkstra的裸题吧... 这道题可以分为来回两个阶段. 去的时候很简单,直接用一次Dijkstra,然后统计答案. 回来的时候就有些巧妙了,虽然表面上是每个点回到起点,但是何尝不可将其看成从起点出发,逆着每个点过来的路去找一次每个点?所以只需要存边的时候处理一下,然后直接跑Dijkstra就行了. 附上代码. #include<bits/stdc++.h> #define clean(a,i) memset(a,i,sizeof(a)) #define ll long long

Luogu P4014 分配问题 题解

闲扯 蒟蒻的第一道自己想出怎么建图的题!!虽然是一个没什么技术含量的图 想了想,还是写篇题解纪念一下. 题面 题面 Solution 要求最小费用和最大费用,同时限制了流量,考虑费用流. 虚拟一个超级源点,从这个点分别向 \(N\) 个任务连一条流量为 \(1\) ,费用为 \(0\) 的边. 虚拟一个超级汇点,才从 \(N\) 个物品分别向该点连一条流量为 \(1\) ,费用为 \(0\) 的边. 因为每个人只能做一件,且每个工作只能做一次,所以连的边流量都为一.而第 \(i\) 个人做第 \

P3952 时间复杂度

—————————————————————————————————————————————————— noip罕见的纯模拟题,细节还是很多的 虽然没有看题解,但也错了很多地方 1,Yes与YES 2,ERR与Err,这个真的毒瘤,一个全大写一个却不是 3,ERR后向栈中添加元素防止RE 4,最后判断栈空. ————————————————————————————————————————- #include<bits/stdc++.h> using namespace std; int t,ln