这道不难的题引发了我不少思考

我第一个版本是用vector嵌套vector写成的,后来发现没必要还存储那些已经超过24h的船,完全可以删除前面的船,因此把外层vector换成了deque。

即用deque存船,用vector存储船上人的国籍

然后喜提TLE,只有70分

然后把cincout换成scanfprintf,仍然TLE

把scanfprintf换成快读快写函数,仍然TLE

意识到可以把存船换成存人,遂把deque换成queue,把存了一个int时间和一个vector国籍的船类换成的只存int时间和int国籍的人类,这样deque套vector直接变成了单单一个queue,但仍然TLE

(后来意识到这三个优化全属于小打小闹,意义不大,第三个优化有点意义,但时间提升其实也不大,还多浪费了一点空间)

然后意识到我应该分析循环次数,发现外层循环循环了n次,内层循环了Ki次+2100000。n是小于100000的数,每个n总要处理不可能缩小,可是n2100000显然达到了十的十次方超时间复杂度了。内层循环的Ki一共运行了sigmaKi次,而sigmaKi是小于300000的,最开始我看错了,我以为Ki小于300000,Ki缩小不了所以才用了个2100000级别的内层运算。结果发现Ki的和才300000次,因而很明显缩小的重点就在于这个两个100000的循环了

这段分析非常重要,让我突然一直到该优化哪里,此前优化的地方都是小打小闹,要么是常数级的优化,要么是至多1~2个数量级的优化,这段分析才真正找到了症结所在。

这种分析我已经好久没做忘却了,这道题让我拾起来这种记忆。给了我三个收获,一是要认真读题,别漏看了sigma,二是写算法前先算好自己的内层循环一共会被执行多少次。三是算法TLE了先检查循环次数是否有数量级级别的大问题,没有的话在进行常数级的小优化。

为什么有两个100000级别的循环呢,因为我用b[i]存储i国籍的人有没有,我用1和0两个状态来存储有没有,而不是用数字存储有多少,这导致我让24h以前的人出队只能将b数组全部清零然后全部新统计,而不能直接在原本b数组的基础上修修改改,因为我不知道出队后b[i]==1的数据该不该归零,毕竟24h以前的人中有没有那个国籍的人我不知道。那么,只要改存数字,把粗暴地损失了信息的b[i]=1改成b[i]++,就可以存储剩下的人了,就可以让那些人出队了,出队后只要b[i]--;我就可以判断那个国籍的人到底还有没有,从而可以直接在b数组上修修改改了,就不需要再全部清零再全部重新统计了。这样就把十的五次方级别的操作直接改成了常数级别的操作。

然后另一个100000级别的循环是我每次处理一艘船的答案的时候都是直接设置一个新的ans量并初始化为0,然后遍历b数组。可是,ans也可以略加修改变成只在原本的基础上修修改改,从而直接把遍历b数组这种是的五次方级别的操作改成了常数级别的操作。

就写出来的Code-3这样AC了。我看题解中很多人都强调要按人头来存,但根据我的分析很明显我这个算法按船来存按人头来存都是不会TLE的,因为我用船来存的时候用了vector。为了证明这一点,我有写了第二份AC-Code,就是Code-4

Code-1

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <map>
#include <vector>
#include <queue>
#include <list>
using namespace std; inline int read(){
int x=0,f=1;char c=getchar();
while(!isdigit(c)){if(c=='-')f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
inline void write(int x){
if(x==0){putchar('0');return;}
int len=0,k1=x,c[10005];
if(k1<0)k1=-k1,putchar('-');
while(k1)c[len++]=k1%10+'0',k1/=10;
while(len--)putchar(c[len]);
} struct ship
{
int t;
vector<int> na;
ship(int T,vector<int> NA){t=T;na.assign(NA.begin(),NA.end());}
};
queue<ship ,list< ship > > Q;
int b[100005],n,t,k,ana;
int main()
{
Q.push(ship(0,vector<int>()));
n=read();
for(int i=1;i<=n;i++)
{
t=read();k=read();
vector<int> tna;
for(int j=1;j<=k;j++)
{
ana=read();
tna.push_back(ana);
}
for(Q.push(ship(t,tna));Q.back().t-Q.front().t>=86400;Q.pop())
for(auto p=Q.front().na.begin();p<Q.front().na.end();p++)
b[*p]--;
for(auto p=Q.back().na.begin();p<Q.back().na.end();p++)
b[*p]++;
int ans=0;
for(int j=1;j<=100000;j++)if(b[j]>0)ans++;
write(ans);putchar('\n');
}
return 0;
}

Code-2

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <map>
#include <vector>
#include <queue>
#include <list>
using namespace std; inline int read(){
int x=0,f=1;char c=getchar();
while(!isdigit(c)){if(c=='-')f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
inline void write(int x){
if(x==0){putchar('0');return;}
int len=0,k1=x,c[10005];
if(k1<0)k1=-k1,putchar('-');
while(k1)c[len++]=k1%10+'0',k1/=10;
while(len--)putchar(c[len]);
} struct p
{
int t,na;
p(int T,int NA){t=T;na=NA;}
};
queue<p> Q;
int b[100005],n,t,k,ana;
int main()
{
Q.push(p(0,0));
n=read();
for(int i=1;i<=n;i++)
{
t=read();k=read();
for(int j=1;j<=k;j++)
{
ana=read();b[ana]++;p h=Q.front();
for(Q.push(p(t,ana));t-h.t>=86400;Q.pop())
b[Q.front().na]--;
}
int ans=0;
for(int j=1;j<=100000;j++)if(b[j]>0)ans++;
write(ans);putchar('\n');
}
return 0;
}

Code3

#include <iostream>
#include <queue>
using namespace std;
struct p{int t,na; p(int T,int NA){t=T;na=NA;} };
int b[100005],n,t,k,ana,ans;
queue<p> Q;
int main()
{
Q.push(p(0,0));
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&t,&k);
for(int j=1;j<=k;j++)
{
scanf("%d",&ana);
b[ana]++;
if(b[ana]==1)ans++;
for(Q.push(p(t,ana));t-Q.front().t>=86400&&!Q.empty();Q.pop())
{
b[Q.front().na]--;
if(b[Q.front().na]==0)ans--;
}
}
printf("%d\n",ans);
}
return 0;
}

Code-4

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <map>
#include <vector>
#include <queue>
#include <list>
using namespace std; struct ship
{
int t;
vector<int> na;
ship(int T,vector<int> NA){t=T;na.assign(NA.begin(),NA.end());}
};
queue<ship> Q;
int b[100005],n,t,k,ana,ans;
int main()
{
Q.push(ship(0,vector<int>()));
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&t,&k);
vector<int> tna;
for(int j=1;j<=k;j++)
{
scanf("%d",&ana);
tna.push_back(ana);
}
for(Q.push(ship(t,tna));t-Q.front().t>=86400;Q.pop())
for(auto p=Q.front().na.begin();p<Q.front().na.end();p++)
{b[*p]--;if(b[*p]==0)ans--;}
for(auto p=Q.back().na.begin();p<Q.back().na.end();p++)
{b[*p]++;if(b[*p]==1)ans++;}
printf("%d",ans);
}
return 0;
}

P2058的更多相关文章

  1. 洛谷 P2058 海港 解题报告

    P2058 海港 题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况: ...

  2. 洛谷P2058 仪仗队

    P2058 仪仗队 24通过 34提交 题目提供者shengmingkexue 标签数论(数学相关) 难度普及+/提高 提交该题 讨论 题解 记录 最新讨论 暂时没有讨论 题目描述 作为体育委员,C君 ...

  3. 洛谷 P2058 海港 题解

    P2058 海港 题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况: ...

  4. P2058 海港

    原题链接  https://www.luogu.org/problemnew/show/P2058 这道题昨天qyf大佬刚给我们讲了一下,所以今天就把它做啦! qyf大佬的思路和我的是一样的(其实是借 ...

  5. 洛谷 P2058 海港(模拟)

    题目链接:https://www.luogu.com.cn/problem/P2058 这是一道用手写队列模拟的一道题,没有什么细节,只是注意因为数不会很大,所以直接用数作为数组下标即可,不用用map ...

  6. [P2058][NOIP2015]海港 (模拟)

    %%%ADMAN #include<cstdio> using namespace std; int n,tot,now,ans,h; ],k[],a[],sum[]; int main( ...

  7. (寒假集训)洛谷 P2058 海港

    小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况:对于第i艘到达的船,他记录了 ...

  8. 洛谷P2058 海港

    题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况:对于第i艘到达的船 ...

  9. 洛谷——P2058 海港

    题目传送 由于于题目保证输入的ti是递增的,所以发现当我们统计完一艘船的答案后,这个答案多少会对下一艘船的答案有贡献.同时还发现如果对每个艘船都记录他的乘客在整个数据出现的所有国籍中相应出现的次数,在 ...

  10. 【HDOJ】P2058 The sum problem

    题意很简单就是给你一个N和M,让你求在1-N的那些个子序列的值等于M 首先暴力法不解释,简单超时 再仔细想一想可以想到因为1-N是一个等差数列,可以运用我们曾经学过的只是来解决 假设开始的位置为s,结 ...

随机推荐

  1. 网安靶场环境_DVWA-读取文件报错File not found! Cookie中有两个security键

    DVWA-文件包含漏洞-读取文件报错-ERROR: File not found! Cookie中有两个security键 1 问题复现 (1)登录DVMA后,设置DVWA Security为Low. ...

  2. Oracle数据库卸载器 - 开源研究系列文章

    今天无事,把网上搜到的Oracle数据库卸载器的软件更新到C#的Winform界面的操作上. 1. 程序目录: 与笔者的其它软件类似,目录如下: 2. 使用的类: 这里主要使用了一个处理函数: 3. ...

  3. MybatisPlus高级特性之ActiveRecord模式

    1.是什么? ActiveRecord是一种领域模型模式,特点是一个模型类对应关系型数据库中的一个表:而模型类的一个实例对应表中的一行记录.ActiveRecord 一直广受解释型动态语言 ( PHP ...

  4. 反向 Debug 了解一下?揭秘 Java DEBUG 的基本原理

    Debug 的时候,都遇到过手速太快,直接跳过了自己想调试的方法.代码的时候吧-- 一旦跳过,可能就得重新执行一遍,准备数据.重新启动可能几分钟就过去了. 好在IDE 们都很强大,还给你后悔的机会,可 ...

  5. 数字孪生为何需要融合GIS?以智慧城市项目为例说明

    数字孪生和地理信息系统(GIS)是两个在现代科技中崭露头角的概念,它们的融合为智慧城市项目带来了革命性的机会.本文将解释数字孪生为何需要融合GIS,并以智慧城市项目为例进行说明. 数字孪生是一种虚拟模 ...

  6. IDEA插件(1 UI美化)

    一.IDEA 插件怎么安装?(图文讲解) IntelliJ IDEA 支持丰富的插件,熟练使用相关插件,能够有效提高我们的开发效率以及用户体验.那么,要如何在 IDEA 中安装插件呢?这里有两种方式: ...

  7. 记一次windows病毒联合排查全过程

    8月2日通过态势感知平台,发现大量内部DNS服务器有恶意请求,且告警描述为:试图解析僵尸网络C&C服务器xmr-eu2.nanopool.org的地址,通过截图可以看到,用户每5分钟会定期向目 ...

  8. 一篇文章彻底搞懂TiDB集群各种容量计算方式

    背景 TiDB 集群的监控面板里面有两个非常重要.且非常常用的指标,相信用了 TiDB 的都见过: Storage capacity:集群的总容量 Current storage size:集群当前已 ...

  9. 如何在IIS上部署docsify以及404问题

    操作步骤 创建一个文件夹,在文件夹中新建2个文件 index.html:入口文件,整个网站只需要这个html文件,其他文件都是md文件 README.md:主页内容,如果没有这个文件,访问时提示404 ...

  10. 震惊!火爆全网的ChatGPT背后使用的数据库居然是……

    摘要:ChatGPT承认了自己背后使用的数据库是Cassandra. OpenAI最近发布的AI驱动的智能聊天机器人ChatGPT在互联网上掀起了一阵风暴,热衷于尝试这一新AI成果的网民不在少数.Ch ...