博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ 1103: [POI2007]大都市meg(dfs序,树状数组)
阅读量:4306 次
发布时间:2019-06-06

本文共 1088 字,大约阅读时间需要 3 分钟。

本来还想链剖的,结果才发现能直接树状数组的= =

记录遍历到达点与退出点的时间,然后一开始每个到达时间+1,退出时间-1,置为公路就-1,+1,询问直接点1到该点到达时间求和就行了- -

CODE:

#include<cstdio>

#include<iostream>

#include<cstring>

#include<algorithm>

#include<vector>

using namespace std;

#define maxn 500010

#define maxm 250010

int f[maxn+10],up[maxm],down[maxm],clo;

vector<int > q[maxm];

#define lowbit(x) (x&(-x))

void add(int x,int y) {

for (x;x<=maxn;x+=lowbit(x)) f[x]+=y;

}

int sum(int x) {

int ans=0;

for (;x;x-=lowbit(x)) ans+=f[x];

return ans;

}

void dfs(int x) {

up[x]=++clo;

for (int i=0;i<q[x].size();i++) dfs(q[x][i]);

down[x]=++clo;

return ;

}

int main(){

int n,m;

scanf("%d",&n);

for (int i=1;i<n;i++) {

int x,y;

scanf("%d%d",&x,&y);

if (x>y) swap(x,y);

q[x].push_back(y);

}

dfs(1);

for (int i=2;i<=n;i++) add(up[i],1),add(down[i],-1);

scanf("%d",&m);

for (int i=1;i<=n+m-1;i++) {

char s[2];int x,y;

scanf("%s",s);

switch (s[0]){

case 'A':

scanf("%d%d",&x,&y);

if (x>y) swap(x,y);

add(up[y],-1);

add(down[y],1);

break;

case 'W':

scanf("%d",&x);

printf("%d\n",sum(up[x]));

break;

}

}

return 0;

}

转载于:https://www.cnblogs.com/New-Godess/p/4348907.html

你可能感兴趣的文章
互斥锁 synchronized分析
查看>>
java等待-通知机制 synchronized和waity()的使用实践
查看>>
win10 Docke安装mysql8.0
查看>>
docker 启动已经停止的容器
查看>>
order by 排序原理及性能优化
查看>>
Lock重入锁
查看>>
docker安装 rabbitMq
查看>>
git 常用命令 入门
查看>>
关闭selinx nginx无法使用代理
查看>>
shell 脚本部署项目
查看>>
spring cloud zuul网关上传大文件
查看>>
springboot+mybatis日志显示SQL
查看>>
工作流中文乱码问题解决
查看>>
maven打包本地依赖包
查看>>
spring boot jpa 实现拦截器
查看>>
jenkins + maven+ gitlab 自动化部署
查看>>
Pull Request流程
查看>>
Lambda 表达式
查看>>
函数式数据处理(一)--流
查看>>
java 流使用
查看>>