JZOJsenior5474.【NOIP2017提高组】day1T2时间复杂度

本文介绍了一个程序复杂度评估的问题,需要编写程序来判断学生给出的程序时间复杂度是否正确。通过对循环结构进行分析,利用栈模拟的方法实现了复杂度的计算,并提供了具体的实现思路和伪代码。

problem

Description

   小明正在学习一种新的编程语言 A++,刚学会循环语句的他激动地写了好多程序并给出了他自己算出的时间复杂度,可他的编程老师实在不想一个一个检查小明的程序,于是你的机会来啦!下面请你编写程序来判断小明对他的每个程序给出的时间复杂度是否正确。
 A++语言的循环结构如下:

 其中“ F i x y ”表示新建变量 i (变量 i 不可与未被销毁的变量重名)并初始化为 x,然后判断 i 和 y 的大小关系,若 i 小于等于 y 则进入循环,否则不进入。每次循环结束后 i 都会被修改成 i +1,一旦 i 大于 y 终止循环。
  x 和 y 可以是正整数(x 和 y 的大小关系不定) 或变量 n。n 是一个表示数据规模的变量,在时间复杂度计算中需保留该变量而不能将其视为常数,该数 远大于 100。
 “ E ”表示循环体结束。循环体结束时,这个循环体新建的变量也被销毁。
 注:本题中为了书写方便,在描述复杂度时,使用大写英文字母“O”表示通常意义下“Θ”的概念。

Input

    输入文件第一行一个正整数 t,表示有 t(t ≤ 10)个程序需要计算时间复杂度。每个程序我们只需抽取其中  “F i x y” ” 和“ “E” ”即可计算时间复杂度 。 注意: 循环结构允许嵌套 。
    接下来每个程序的第一行包含一个正整数 L 和一个字符串,L 代表程序行数,字符串表示这个程序的复杂度,“O(1)”表示常数复杂度,“O(n^w)”表示复杂度为�� �� ,其中w是一个小于100的正整数(输入中不包含引号),输入保证复杂度只有O(1)和O(n^w)两种类型。
    接下来 L 行代表程序中循环结构中的“F i x y”或者 “E”。
    程序行若以“F”开头,表示进入一个循环,之后有空格分离的三个字符(串)i x y,其中 i 是一个小写字母(保证不为“n”),表示新建的变量名,x 和 y 可能是正整数或n ,已知若为正整数则一定小于 100。
    程序行若以“E”开头,则表示循环体结束。

Output

   输出文件共 t 行,对应输入的 t 个程序,每行输出“Yes”或“No”或者“ERR”(输出中不包含引号),若程序实际复杂度与输入给出的复杂度一致则输出“Yes”,不一致则输出“No”,若程序有语法错误(其中语法错误只有: ① F 和 E 不匹配 ②新建的变量与已经存在但未被销毁的变量重复两种情况),则输出“ERR”。
  注意:即使在程序不会执行的循环体中出现了语法错误也会编译错误,要输出“ERR”。

Sample Input

8
2 O(1)
F i 1 1
E
2 O(n^1)
F x 1 n
E
1 O(1)
F x 1 n
4 O(n^2)
F x 5 n
F y 10 n
E
E
4 O(n^2)
F x 9 n
E
F y 2 n
E
4 O(n^1)
F x 9 n
F y n 4
E
E
4 O(1)
F y n 4
F x 9 n
E
E
4 O(n^2)
F x 1 n
F x 1 10
E
E

Sample Output

Yes
Yes
ERR
Yes
No
Yes
Yes
ERR

【输入输出样例 1 说明】
第一个程序 i 从 1 到 1 是常数复杂度。
第二个程序 x 从 1 到 n 是 n 的一次方的复杂度。
第三个程序有一个 F 开启循环却没有 E 结束,语法错误。
第四个程序二重循环,n 的平方的复杂度。
第五个程序两个一重循环,n 的一次方的复杂度。
第六个程序第一重循环正常,但第二重循环开始即终止(因为n远大于100,100大于4)。
第七个程序第一重循环无法进入,故为常数复杂度。
第八个程序第二重循环中的变量 x 与第一重循环中的变量重复,出现语法错误②,输出ERR。

Data Constraint

对于 30%的数据:不存在语法错误,数据保证小明给出的每个程序的前 L/2 行一定为以 F 开头的语句,第 L/2+1 行至第 L 行一定为以 E 开头的语句,L<=10,若 x、y 均为整数,x 一定小于 y,且只有 y 有可能为 n。
对于 50%的数据:不存在语法错误,L<=100,且若 x、y 均为整数,x 一定小于 y,且只有 y 有可能为 n。
对于 70%的数据:不存在语法错误,L<=100。
对于 100%的数据:L<=100。


analysis

CCF为了送我们分要我们打个编译器太恶心了

一眼栈模拟题
奇诡的是,大部分纪中OIers只扫了一遍**时间复杂度**O(m)就解决了
(不执行有语法错误也输出ERR)我觉得很eggsache,于是就O(m)扫了三遍……
第一遍,check一下有木有语法错误,这个真的很简单
第二遍是initialization,处理一下每个F对应哪个E(其实前两个可以合并在一起)
第三个就是solve了,对于每个F循环有五种情况:

  • x和y都为常数且x≤y:O(1)循环
  • x和y都为常数且x>y:不循环,直接跳到循坏尾
  • x为常数且y为n:O(n)循环
  • x为n且y为常数:不循环,直接跳到循环尾
  • x和y都为n:O(1)循环

对于进入每个循环,用栈模拟
每进入一个O(n)循环就把n的指数+1,每退出一个就-1,同时记录最大值
这样既可在O(tm)时间复杂度内求解

(gmh:蛤我比较常数拿字符串来比较我好方啊)


code

150lines4500+bytes码农题毁我青春
拿c++打这种题是很容易GG的

var
        bz:array['a'..'z']of boolean;
        stack:array[0..1000]of char;
        stack1,next:array[0..1000]of longint;
        a:array[0..1000]of string;
        n,m,i,j,k,top,xx,yy,ans:longint;
        temp:char;
        s:string;
function max(x,y:longint):longint;
begin
        if x>y then exit(x);
        exit(y);
end;
function check:boolean;
var
        i:longint;
begin
        fillchar(bz,sizeof(bz),true);
        fillchar(stack,sizeof(stack),0);
        top:=0;
        if (m mod 2=1)or(a[1][1]='E') then exit(false);
        for i:=1 to m do
        begin
                if a[i][1]='F' then
                begin
                        if not bz[a[i][3]] then exit(false);
                        bz[a[i][3]]:=false;
                        inc(top);
                        stack[top]:=a[i][3];
                end
                else
                begin
                        bz[stack[top]]:=true;
                        dec(top);
                        if top<0 then exit(false);
                end;
        end;
        if top>0 then exit(false);
        exit(true);
end;
procedure init;
var
        i:longint;
begin
        top:=0;
        fillchar(stack1,sizeof(stack1),0);
        for i:=1 to m do
        begin
                if a[i][1]='F' then
                begin
                        inc(top);
                        stack1[top]:=i;
                end
                else
                begin
                        next[stack1[top]]:=i;
                        dec(top);
                end;
        end;
end;
procedure turn(st:string;var x,y:string);
begin
        delete(st,1,4);
        x:=copy(st,1,pos(' ',st)-1);
        delete(st,1,pos(' ',st));
        y:=st;
end;
function solve:longint;
var
        i,mx:longint;
        x,y:string;
begin
        fillchar(stack,sizeof(stack),0);
        top:=0;
        solve:=0;
        mx:=0;
        i:=0;
        while i<m do
        begin
                inc(i);
                if a[i][1]='F' then
                begin
                        inc(top);
                        turn(a[i],x,y);
                        if (x<>'n')and(y<>'n') then
                        begin
                                val(x,xx);
                                val(y,yy);
                                if xx<=yy then
                                begin
                                        stack[top]:='0';
                                end
                                else
                                begin
                                        dec(top);
                                        i:=next[i];
                                end;
                        end
                        else if (x<>'n')and(y='n') then
                        begin
                                stack[top]:='1';
                                inc(mx);
                                solve:=max(solve,mx);
                        end
                        else if (x='n')and(y<>'n') then
                        begin
                                dec(top);
                                i:=next[i];
                        end
                        else if (x='n')and(y='n')then
                        begin
                                stack[top]:='0';
                        end;
                end
                else
                begin
                        if stack[top]='1' then dec(mx);
                        dec(top);
                end;
        end;
        exit(solve);
end;
begin
        assign(input,'complexity.in');reset(input);
        assign(output,'complexity.out');rewrite(output);
        readln(n);
        for k:=1 to n do
        begin
                readln(m,temp,s);
                for i:=1 to m do readln(a[i]);
                if not check then writeln('ERR')
                else
                begin
                        fillchar(next,sizeof(next),0);
                        init;
                        if s[3]='n' then
                        begin
                                val(copy(s,pos('^',s)+1,length(s)-pos('^',s)-1),ans);
                                if ans=solve then writeln('Yes')
                                else writeln('No');
                        end
                        else
                        begin
                                if solve=0 then writeln('Yes')
                                else writeln('No');
                        end;
                end;
        end;
        close(input);close(output);
end.
# P2615 [NOIP 2015 提高] 神奇的幻方 ## 题目背景 NOIp2015 提高 Day1T1 ## 题目描述 幻方是一种很神奇的 $N\times N$ 矩阵:它由数字 $1,2,3,\cdots \cdots ,N \times N$ 构成,且每行、每列及两条对角线上的数字之和都相同。 当 $N$ 为奇数时,我们可以通过下方法构建一个幻方: 首先将 $1$ 写在第一行的中间。 之后,按如下方式从小到大依次填写每个数 $K \ (K=2,3,\cdots,N \times N)$ : 1. 若 $(K-1)$ 在第一行但不在最后一列,则将 $K$ 填在最后一行, $(K-1)$ 所在列的右一列; 2. 若 $(K-1)$ 在最后一列但不在第一行,则将 $K$ 填在第一列, $(K-1)$ 所在行的上一行; 3. 若 $(K-1)$ 在第一行最后一列,则将 $K$ 填在 $(K-1)$ 的正下方; 4. 若 $(K-1)$ 既不在第一行,也不在最后一列,如果 $(K-1)$ 的右上方还未填数,则将 $K$ 填在 $(K-1)$ 的右上方,否则将 $K$ 填在 $(K-1)$ 的正下方。 现给定 $N$ ,请按上述方法构造 $N \times N$ 的幻方。 ## 输入格式 一个正整数 $N$,即幻方的大小。 ## 输出格式 共 $N$ 行,每行 $N$ 个整数,即按上述方法构造出的 $N \times N$ 的幻方,相邻两个整数之间用单空格隔开。 ## 输入输出样例 #1 ### 输入 #1 ``` 3 ``` ### 输出 #1 ``` 8 1 6 3 5 7 4 9 2 ``` ## 输入输出样例 #2 ### 输入 #2 ``` 25 ``` ### 输出 #2 ``` 327 354 381 408 435 462 489 516 543 570 597 624 1 28 55 82 109 136 163 190 217 244 271 298 325 353 380 407 434 461 488 515 542 569 596 623 25 27 54 81 108 135 162 189 216 243 270 297 324 326 379 406 433 460 487 514 541 568 595 622 24 26 53 80 107 134 161 188 215 242 269 296 323 350 352 405 432 459 486 513 540 567 594 621 23 50 52 79 106 133 160 187 214 241 268 295 322 349 351 378 431 458 485 512 539 566 593 620 22 49 51 78 105 132 159 186 213 240 267 294 321 348 375 377 404 457 484 511 538 565 592 619 21 48 75 77 104 131 158 185 212 239 266 293 320 347 374 376 403 430 483 510 537 564 591 618 20 47 74 76 103 130 157 184 211 238 265 292 319 346 373 400 402 429 456 509 536 563 590 617 19 46 73 100 102 129 156 183 210 237 264 291 318 345 372 399 401 428 455 482 535 562 589 616 18 45 72 99 101 128 155 182 209 236 263 290 317 344 371 398 425 427 454 481 508 561 588 615 17 44 71 98 125 127 154 181 208 235 262 289 316 343 370 397 424 426 453 480 507 534 587 614 16 43 70 97 124 126 153 180 207 234 261 288 315 342 369 396 423 450 452 479 506 533 560 613 15 42 69 96 123 150 152 179 206 233 260 287 314 341 368 395 422 449 451 478 505 532 559 586 14 41 68 95 122 149 151 178 205 232 259 286 313 340 367 394 421 448 475 477 504 531 558 585 612 40 67 94 121 148 175 177 204 231 258 285 312 339 366 393 420 447 474 476 503 530 557 584 611 13 66 93 120 147 174 176 203 230 257 284 311 338 365 392 419 446 473 500 502 529 556 583 610 12 39 92 119 146 173 200 202 229 256 283 310 337 364 391 418 445 472 499 501 528 555 582 609 11 38 65 118 145 172 199 201 228 255 282 309 336 363 390 417 444 471 498 525 527 554 581 608 10 37 64 91 144 171 198 225 227 254 281 308 335 362 389 416 443 470 497 524 526 553 580 607 9 36 63 90 117 170 197 224 226 253 280 307 334 361 388 415 442 469 496 523 550 552 579 606 8 35 62 89 116 143 196 223 250 252 279 306 333 360 387 414 441 468 495 522 549 551 578 605 7 34 61 88 115 142 169 222 249 251 278 305 332 359 386 413 440 467 494 521 548 575 577 604 6 33 60 87 114 141 168 195 248 275 277 304 331 358 385 412 439 466 493 520 547 574 576 603 5 32 59 86 113 140 167 194 221 274 276 303 330 357 384 411 438 465 492 519 546 573 600 602 4 31 58 85 112 139 166 193 220 247 300 302 329 356 383 410 437 464 491 518 545 572 599 601 3 30 57 84 111 138 165 192 219 246 273 301 328 355 382 409 436 463 490 517 544 571 598 625 2 29 56 83 110 137 164 191 218 245 272 299 ``` ## 说明/提示 对于 $100\%$ 的数据,对于全部数据, $1 \leq N \leq 39$ 且 $N$ 为奇数。 用C语言写
最新发布
09-30
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值