关于重复定义变量与memset函数导致超时的分析

简述

先上俩组代码
code1:

void solve(){
    int n;
    PII a[200010];//注意此处
    cin>>n;
    rep(i,1,n){
        cin>>a[i].fi;
        a[i].se=i;
    }
    sort(a+1,a+1+n);
    a[n+1].fi=INF;
    a[0].fi=INF;
    int k;
    rep(i,1,n){
        if(a[i].fi!=a[i-1].fi&&a[i].fi!=a[i+1].fi){
            cout<<a[i].se<<endl;
            return;
        }
 
    }
    cout<<-1<<endl;
}
int main(){
    int t;
    cin>>t;
    while(t--){
        solve();
    }
    return 0;
}

code2:

void solve(){
    int n;
    n=read();
    PII a[n+10];//注意此处
    rep(i,1,n){
        a[i].fi=read();
        a[i].se=i;
    }
    sort(a+1,a+1+n);
    a[n+1].fi=INF;
    a[0].fi=INF;
    int k;
    rep(i,1,n){
        if(a[i].fi!=a[i-1].fi&&a[i].fi!=a[i+1].fi){
            write(a[i].se);
            printf("\n");
            return;
        }
 
    }
    cout<<-1<<endl;
}
int main(){
    int t;
    t=read();
    while(t--){
        solve();
    }
    return 0;
}

显然上面俩组代码只有注释处不同,我们将其拿到cf上交俩发

code1耗时
在这里插入图片描述
code2耗时
在这里插入图片描述
俩个代码耗时竟然相差了十倍!!!

分析

当输入数据为 t = 2 e 4 t=2e4 t=2e4,
接下来跟着 2 e 4 2e4 2e4
n = 3 n=3 n=3
a 1 = 1 , a 2 = 1 , a 3 = 1 a_1=1,a_2=1,a_3=1 a1=1,a2=1,a3=1
每次while循环中会重复定义一个2e5的数组a,此处的a显然为局部变量,众所周知局部变量会在本次运行完成后进行销毁操作,可能是此处的申请空间与销毁空间耗时太多,导致时间花费较大,具体的情况笔者也不了解,此为猜测,不太懂编译原理相关的东西.


再来看看定义一个 2 e 5 2e5 2e5大小的全局变量,每次在 s o l v e solve solve中使用memset(a,0,sizeof a)清空,此时memset几乎等价于for(int i=1;i<=2e5;i++) a[i]=0,当输入上面的数据时,时间复杂度为
O ( 2 e 4 ∗ 1 e 5 ) = 2 e 9 O(2e4*1e5)=2e9 O(2e41e5)=2e9
显然 c + + c++ c++一秒内是跑不完的,导致超时

使用memset的解决方案:
memset(a,0,sizeof a)换成for(int i=1;i<=n;i++) a[i]=0即可

此时依旧运行上面的样例,时间复杂度为 O ( 2 e 4 ∗ 3 ) O(2e4*3) O(2e43),显然很快


记录一场cf被hack的经历
本来的code1为cin/cout输入输出,直接被某同学当场hack掉

code1为加入快读版本,在cf选手的不断努力下从800+ms跑到998ms,极限ac
在这里插入图片描述

珍爱 r a t i n g rating rating,避免不好的代码习惯(大雾)

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值