\(noip模拟27\;solutions\)

这次吧,我本来以为我能切掉两个题,结果呢??只切掉了一个

不过,隔壁Varuxn也以为能切两个,可惜了,他一个都没切。。。。。。

确实他分比我高一点,但是吧,这个人就改题非常慢,所以结论就是

我牛逼,牛逼到家了

所以我应该是挂掉了100pts,下次注意,下次AK

\(T1\;妹子图(graph)\)

这个题有一堆做法,而且旗鼓相当,复杂度都一样,跑的也差不多快

我的做法是在无向图上跑魔改拓扑排序+魔改DIJ,非常的恶心,但是吧,理解之后极其简单

按照堆优化的DIJ搞了一个优先队列,这样我们先把半仙的家能够连到的地方压到队列中

因为我们是按照边权小的在前面,所以我们当前找到的一定是当前困难抵抗值能够到达的最大的图

显然有一些困难抵抗值能够见到的妹子种类数是一样的,这些我们就不必保存了

这样的话,我们的数组里最多只有2e5个点,直接保证了时间和空间复杂度

我们开始弹队列,然后更新这个数组,这里面存的东西就是

能够看到当前这么多妹子的最小困难抵抗值,和能够看到的妹子

最后的时候直接在这个数组里二分查找,搞一搞边界就好了

复杂度是\(O(nlogn+qlogn)\)

AC_code

#include<bits/stdc++.h>
using namespace std;
#define re register int
#define ll long long
const int N=5e5+5;
int n,m,qq,hom,opt,mod;
int c[N];
ll num[N],fro[N],bas[N],cnt;
int to[N*2],nxt[N*2],val[N*2],head[N],rp;
void add_edg(int x,int y,int z){
to[++rp]=y;
val[rp]=z;
nxt[rp]=head[x];
head[x]=rp;
}
int ton[N],sum;
bool vis[N];
struct node{
int val,tag;
node(){}
node(int x,int y){
val=x;tag=y;
}
bool operator < (node x)const{
return val>x.val;
}
};
priority_queue<node> q;
signed main(){
scanf("%d%d%d%d%d",&n,&m,&qq,&hom,&opt);
if(opt)scanf("%d",&mod);
for(re i=1;i<=n;i++){
scanf("%d",&c[i]);
}
for(re i=1,x,y,z;i<=m;i++){
scanf("%d%d%d",&x,&y,&z);
add_edg(x,y,z);
add_edg(y,x,z);
}
ton[c[hom]]++;sum++;
for(re i=head[hom];i;i=nxt[i])
q.push(node(val[i],to[i]));
int cge=0;bas[++cnt]=1;num[cnt]=sum;
while(!q.empty()){
int dis=q.top().val;
int now=q.top().tag;
q.pop();
if(vis[now])continue;
vis[now]=true;
cge=max(cge,dis);
ton[c[now]]++;
if(ton[c[now]]==1){
sum++;
if(bas[cnt]!=cge)
bas[++cnt]=cge;
num[cnt]=sum;
}
for(re i=head[now];i;i=nxt[i]){
q.push(node(val[i],to[i]));
}
}
bas[++cnt]=1e9+1;
for(re i=1;i<=cnt;i++){
fro[i]=fro[i-1]+1ll*num[i-1]*(bas[i]-bas[i-1]);
//cout<<bas[i]<<" "<<num[i]<<" "<<fro[i]<<endl;
}
ll lans=0;
for(re i=1;i<=qq;i++){
ll x,y;
scanf("%lld%lld",&x,&y);
if(opt==1){
x=(x^lans)%mod+1;
y=(y^lans)%mod+1;
}
if(x>y)swap(x,y);
int ld=upper_bound(bas+1,bas+cnt+1,x)-bas;
int rd=upper_bound(bas+1,bas+cnt+1,y)-bas-1;
//cout<<ld<<" "<<rd<<" "<<num[ld]<<" "<<bas[ld]<<endl;
ll ans=fro[rd]-fro[ld];
ans+=1ll*(bas[ld]-x)*num[ld-1];
ans+=1ll*(y-bas[rd]+1)*num[rd];
printf("%lld\n",ans);
lans=ans;
}
}


\(T2\;妹子腿(tree)\)

真是没想到啊,没想到,这个竟然是个大暴力,主要是他还能过,气死人,比线段树还快

考场上没敢写,气人。

就是维护被标记的点的标记时间以及和要查询的点的距离,

这个距离就是树剖+LCA,完事!!!

注意优化,要不然过不了,如果要标记的点已经在不断的扩散中被标记了,就不要加进去了。

AC_code

#include<bits/stdc++.h>
using namespace std;
#define re register int
#define fi first
#define se second
#define pa pair<int,int>
#define mpa(x,y) make_pair(x,y)
const int N=1e5+5;
int n,m;
int to[N*2],nxt[N*2],head[N],rp;
void add_edg(int x,int y){
to[++rp]=y;
nxt[rp]=head[x];
head[x]=rp;
}
int fa[N],dep[N];
int siz[N],son[N],top[N];
void dfs1(int x){
siz[x]=1;son[x]=0;
for(re i=head[x];i;i=nxt[i]){
int y=to[i];
if(y==fa[x])continue;
fa[y]=x;dep[y]=dep[x]+1;
dfs1(y);
siz[x]+=siz[y];
if(!son[x]||siz[y]>siz[son[x]])son[x]=y;
}
}
void dfs2(int x,int f){
top[x]=f;
if(son[x])dfs2(son[x],f);
for(re i=head[x];i;i=nxt[i]){
int y=to[i];
if(y==fa[x]||y==son[x])continue;
dfs2(y,y);
}
}
int LCA(int x,int y){
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]])swap(x,y);
x=fa[top[x]];
}
return dep[x]<dep[y]?x:y;
}
int DIS(int x,int y){
return dep[x]+dep[y]-2*dep[LCA(x,y)];
}
pa ji[N];
int cnt;
signed main(){
scanf("%d%d",&n,&m);
for(re i=1,x,y;i<n;i++){
scanf("%d%d",&x,&y);
add_edg(x,y);add_edg(y,x);
}
dfs1(1);dfs2(1,1);
for(re i=1,typ,x;i<=m;i++){
scanf("%d%d",&typ,&x);
if(typ==1){
int flag=0;
for(re j=1;j<=cnt;j++){
if(DIS(ji[j].fi,x)<=i-ji[j].se){
flag=1;break;
}
}
if(flag==1)continue;
ji[++cnt]=mpa(x,i);
}
if(typ==2)cnt=0;
if(typ==3){
int flag=0;
for(re j=1;j<=cnt;j++){
if(DIS(ji[j].fi,x)<=i-ji[j].se){
flag=1;break;
}
}
if(flag==1)printf("wrxcsd\n");
else printf("orzFsYo\n");
}
}
}


\(T3\;妹子腰(序列)\)

这个题我一眼就是原题,而且是昨天的原题

后来发现不对,是极长上升而不是最长上升

所以我忘记了如何维护这个东西了,考完试又去回顾了一下,好难哟!!!

就是一个\(O(log^2n)\)的线段树,注意维护一个底线,来保证是极长的。。

AC_code

#include<bits/stdc++.h>
using namespace std;
#define re register int
#define ll long long
const int N=2e5+5;
const ll inf=0x3f3f3f3f3f3f3f3f;
const ll mod=998244353;
int n,a[N],maxd;
struct node{
#define ls x<<1
#define rs x<<1|1
ll sum[N*4];
int mxra[N*4];
ll work(int x,int l,int r,int maxn){
//cout<<x<<" "<<l<<" "<<r<<endl;
if(l==r){
if(mxra[x]>maxn)return sum[x];
return 0;
}
int mid=l+r>>1;
if(maxn>mxra[x])return 0;
else if(maxn>mxra[rs])return work(ls,l,mid,maxn);
else{
ll ret=0;
ret=(ret+work(rs,mid+1,r,maxn))%mod;
ret=(ret+work(ls,l,mid,mxra[rs]))%mod;
return ret;
}
}
void pushup(int x){
mxra[x]=max(mxra[ls],mxra[rs]);
//sum[x]=(sum[ls]+sum[rs])%mod;
return ;
}
void ins(int x,int l,int r,int pos,int mx,int v){
if(l==r){
mxra[x]=mx;
sum[x]=v;
return ;
}
int mid=l+r>>1;
if(pos<=mid)ins(ls,l,mid,pos,mx,v);
else ins(rs,mid+1,r,pos,mx,v);
pushup(x);return ;
}
ll query(int x,int l,int r,int pos){
if(pos==0)return 0;
if(r<=pos){
ll ret=work(x,l,r,maxd);
maxd=max(maxd,mxra[x]);
return ret;
}
int mid=l+r>>1;ll ret=0;
if(pos>mid)ret=(ret+query(rs,mid+1,r,pos))%mod;
ret=(ret+query(ls,l,mid,pos))%mod;
return ret;
}
#undef ls
#undef rs
}xds;
ll ans,dp[N];
signed main(){
scanf("%d",&n);
for(re i=1;i<=n;i++)scanf("%d",&a[i]);
for(re i=1;i<=n;i++){
maxd=0;
dp[i]=xds.query(1,1,n,a[i]-1);
//cout<<dp[i]<<" ";
if(dp[i]==0)dp[i]=1;
xds.ins(1,1,n,a[i],i,dp[i]);
//cout<<dp[i]<<endl;
}
int mx=0;
for(re i=n;i>=1;i--){
if(a[i]>mx)ans=(ans+dp[i])%mod,mx=a[i];
}
printf("%lld",ans);
}


noip模拟27[妹子图·腿·腰](fengwu半仙的妹子们)的更多相关文章

  1. 2021.7.29考试总结[NOIP模拟27]

    T1 牛半仙的妹子图 做法挺多的,可以最小生成树或者最短路,复杂度O(cq),c是颜色数. 我考场上想到了原来做过的一道题影子,就用了并查集,把边权排序后一个个插入,记录权值的前缀和,复杂度mlogm ...

  2. 7.29考试总结(NOIP模拟27)[牛半仙的妹子图·Tree·序列]

    前言 从思路上来讲是比较成功的,从分数上就比较令人失望了. 考场上是想到了前两个题的正解思路,其实最后一个题是半个原题,只可惜是我看不懂题... 这波呀,这波又是 语文素养限制OI水平.. 改题的时候 ...

  3. NOIP 模拟 $27\; \rm 牛半仙的妹子序列$

    题解 \(by\;zj\varphi\) 明显一道极长上升子序列的题. 直接线段树维护单调栈,最后单调栈求出可以贡献的序列,答案相加就行. Code #include<bits/stdc++.h ...

  4. NOIP 模拟 $27\; \rm 牛半仙的妹子Tree$

    题解 \(by\;zj\varphi\) 很妙的虚树题. 考虑若没有操作 \(2\),那么直接记录一下扩散到它的最短时间和询问时间相比即可,可以当作一个树上最短路. 有 \(2\) 操作怎么办,将操作 ...

  5. NOIP模拟赛 混合图

    [题目描述] Hzwer神犇最近又征服了一个国家,然后接下来却也遇见了一个难题. Hzwer的国家有n个点,m条边,而作为国王,他十分喜欢游览自己的国家.他一般会从任意一个点出发,随便找边走,沿途欣赏 ...

  6. 20190820 Tue 集训总结&NOIP模拟 27

    低谷度过了? 但是skyh阿卡了,还是反衬出我的辣鸡. T1知道要sort,却忘了判重,正解不如暴力分高,555. T2成功化出正解柿子,然后化过头了,化出了无法DP的柿子. 果然不够强,大神们一眼就 ...

  7. NOIP模拟27

    两个机房又和在一起考试 开场看了看T1,感觉挺水的,过. T2,这个式子有点奇怪,暂时没什么思路,过 T3,好像保留最后几位换个根处理一下就行了,过,先去打T1 于是T1大概打了0.5h,连暴力带正解 ...

  8. NOIP模拟27(命悬一线)

    考得太悬了!

  9. NOIP 模拟 $27\; \rm 牛半仙的妹子图$

    题解 \(by\;zj\varphi\) 颜色数很少,考虑枚举颜色数. 建出来一棵最小生成树,可以证明在最小生成树上,一个点到另一个点的路径上的最大权值最小(易证,考虑 \(\rm kruskal\) ...

随机推荐

  1. 网络游戏逆向分析-3-通过发包函数找功能call

    网络游戏逆向分析-3-通过发包函数找功能call 网络游戏和单机游戏的分析有相似点,但是区别还是很大的. 网络游戏和单机游戏的区别: 网络游戏是需要和服务器进行交互的,网游中的所有功能几乎都会先发送封 ...

  2. 【linux】驱动-13-阻塞与非阻塞

    目录 前言 13. 阻塞与非阻塞 13.1 阻塞与非阻塞 13.2 休眠与唤醒 13.2.1 内核休眠函数 13.2.2 内核唤醒函数 13.3 等待队列(阻塞) 13.3.1 定义等待队列头部 13 ...

  3. Eclipse安装PyDev失败的解决办法

    在Eclipse上安装Pydev,首先尝试了三种方法,均失败. Eclipse版本号如下: 尝试的方法 第一种: 菜单栏Help>Install New Software,输入网址, http: ...

  4. golang 用defer 捕获error 需小心

    有时一个函数内需要根据最后是否出错,决定是否执行某个操作.这时候如果函数的分支又比较多,就会比较麻烦了. defer 处理这个情况刚好合适 func main() { var err error by ...

  5. 『无为则无心』Python基础 — 13、Python流程控制语句(条件语句)

    目录 1.流程控制基本概念 2.选择结构(条件语句) (1)条件语句概念 (2)if语句语法 (3)if...else...语句 (4)多重判断 (5)if语句嵌套 3.应用:猜拳游戏 4.三元运算符 ...

  6. linux密码策略

    1.密码过期策略 # vim /etc/login.defs PASS_MAX_DAYS 99999 # 一个密码最长可以使用的天数: PASS_MIN_DAYS 0 # 更换密码的最小天数: PAS ...

  7. Java核心反射机制

    Java核心反射机制: 基本反射: 反射是一种动态类的处理机制,通过Class类来实现反射机制: Class类的基本信息: Module java.base Package java.lang Cla ...

  8. Linux:Ubuntu银河麒麟防火墙操作

    查看防火墙状态 #防火墙状态 sudo ufw status inactive状态是防火墙 关闭 状态 active状态是防火墙 开启 状态 开启防火墙 #开启防火墙 sudo ufw enable ...

  9. angularjs的一点总结

    一,错误小结 1.出现类似于下面的错误,就是说明 $sessionStorage 这个服务未找到 Error: [$injector:unpr] Unknown provider: $sessionS ...

  10. CG-CTF WxyVM

    一. 之前一直以为虚拟机是那种vmp的强壳,下午看了一些文章才逐渐明白虚拟机这个概念,目前ctf中题目出现的都是在程序中相等于内嵌了一个虚拟机,将程序代码转换成自己定义的指令,通过内嵌的虚拟机进行解释 ...