P2058
这道不难的题引发了我不少思考
我第一个版本是用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的更多相关文章
- 洛谷 P2058 海港 解题报告
P2058 海港 题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况: ...
- 洛谷P2058 仪仗队
P2058 仪仗队 24通过 34提交 题目提供者shengmingkexue 标签数论(数学相关) 难度普及+/提高 提交该题 讨论 题解 记录 最新讨论 暂时没有讨论 题目描述 作为体育委员,C君 ...
- 洛谷 P2058 海港 题解
P2058 海港 题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况: ...
- P2058 海港
原题链接 https://www.luogu.org/problemnew/show/P2058 这道题昨天qyf大佬刚给我们讲了一下,所以今天就把它做啦! qyf大佬的思路和我的是一样的(其实是借 ...
- 洛谷 P2058 海港(模拟)
题目链接:https://www.luogu.com.cn/problem/P2058 这是一道用手写队列模拟的一道题,没有什么细节,只是注意因为数不会很大,所以直接用数作为数组下标即可,不用用map ...
- [P2058][NOIP2015]海港 (模拟)
%%%ADMAN #include<cstdio> using namespace std; int n,tot,now,ans,h; ],k[],a[],sum[]; int main( ...
- (寒假集训)洛谷 P2058 海港
小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况:对于第i艘到达的船,他记录了 ...
- 洛谷P2058 海港
题目描述 小K是一个海港的海关工作人员,每天都有许多船只到达海港,船上通常有很多来自不同国家的乘客. 小K对这些到达海港的船只非常感兴趣,他按照时间记录下了到达海港的每一艘船只情况:对于第i艘到达的船 ...
- 洛谷——P2058 海港
题目传送 由于于题目保证输入的ti是递增的,所以发现当我们统计完一艘船的答案后,这个答案多少会对下一艘船的答案有贡献.同时还发现如果对每个艘船都记录他的乘客在整个数据出现的所有国籍中相应出现的次数,在 ...
- 【HDOJ】P2058 The sum problem
题意很简单就是给你一个N和M,让你求在1-N的那些个子序列的值等于M 首先暴力法不解释,简单超时 再仔细想一想可以想到因为1-N是一个等差数列,可以运用我们曾经学过的只是来解决 假设开始的位置为s,结 ...
随机推荐
- [ABC265F] Manhattan Cafe
Problem Statement In an $N$-dimensional space, the Manhattan distance $d(x,y)$ between two points $x ...
- cp {,bak}用法(转载)
cp filename{,bak} cp filename{,.bak} 这个命令是用来把filename备份成filename.bak的 等同于命令 cp filename filename.bak ...
- 年底了,网站被挂马了,关于IIS被陌生DLL劫持(新人发帖,写的不好的地方,请多多担待)
一上班被分到两个需要杀毒的站点,情况是SEO被劫持 出现一些博彩信息,但是打开确实正常内容,使用站长工具的网站被黑检测功能,发现网站的HEAD前面加载一对加密的东西 一开始我使用D盾扫描网站,删除了一 ...
- Unity3d_Rewired官方文档翻译:概念(一):InputManager、Players、Actions
仅翻译了官方文档中的Essentials(要点).Concepts(概念)两部分,这是文档中最重要的部分,理解了这两部分的内容应该足以让你将Rewired运用到你的项目中,之后再去阅读文档的其他部分也 ...
- static、final、private是否可以修饰抽象方法?
1.static和abstract:是不能共存的.static是为了方便调用,abstract是为了给子类重写,没有方法体. 2.final和abstract:是互相冲突的,final修饰的方法不能重 ...
- HDU 3660 树形DP
原题地址 题意 Alice与Bob在一棵树的树根一同出游,两人从Bob开始轮换选择道路一直走到树叶,Alice会尽可能使走过的总长最小,而Bob相反.不过他们都不能让总长落到[l, r]之外 现在给出 ...
- SaaS 营销,如何利用 RPA 实现自动化获客?
大家好,这次给大家带来如何利用 RPA 实现自动化获客. 一.RPA 是什么?难吗? RPA 对大家来说,可能挺陌生的,其实它很简单. Robotic Process Automation(简称 RP ...
- 【玩转鲲鹏 DevKit系列】如何快速迁移无源码应用?
本文分享自华为云社区<[玩转鲲鹏 DevKit系列]如何快速迁移无源码应用?>,作者: 华为云社区精选. 为了帮助广大用户和开发者快速将无源码应用从 x86 迁移到鲲鹏,鲲鹏 DevKit ...
- 小熊派开发实践丨小熊派+合宙Cat.1接入云服务器
摘要:使用小熊派开发板,以合宙的AIR724为通信模组(Cat.1),以AT指令方式,通过mqtt协议接入云服务器. 本贴使用小熊派开发板+合宙的Air724(Cat.1模组),接入自己搭建的EMQ服 ...
- 划重点丨详解Java流程控制语句知识点
摘要:流程控制语句就是用来控制程序中各语句执行的顺序,下面将详细介绍java流程控制语句. 流程控制语句就是用来控制程序中各语句执行的顺序,下面将详细介绍java流程控制语句. Q: break后面加 ...