HDU 4912 LCA+策略

点击打开链接

题意:给了一个树,然后m条路径,问最多可以选多少条路径而没有一个点是重复使用的,如第二组样例,3条路径是4—2—5和6—3—7和2—1—3,那么只能选前两个使得所选路径最多

思路:没啥思路,看了正解竟然是LCA+贪心,好嘛可以这样考虑,对于所有的可选路径,我们先选择最下面的对上面是没有影响的,那么我们可以对每条路经的最上面的那个点进行排序,就按深度由大到小排序,然后这个最上面的点不就是LCA嘛,然后因为是由下到上的,那么对于每次之后就要将以LCA为根的子树节点全部标记,因为它作为深度最深的点以后它的下面在出现肯定是不符合的了       PS:主要是不敢去想用贪心,自己就是想当然的写根本不知道这么贪心对不对(弱哭)

#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
const int inf=0x3f3f3f3f;
const ll INF=0x3f3f3f3f3f3f3f3f;
const int maxn=100010;
bool vis[maxn];
int L[maxn*2],E[maxn*2],H[maxn],dis[maxn],dp[2*maxn][20],num[maxn],head[maxn],kkk,pre[maxn];
struct node{
    int to,next;
}EE[maxn*10];
void add_edge(int u,int v){
    EE[kkk].to=v;EE[kkk].next=head[u];head[u]=kkk++;
}
int k,n;
void dfs(int t,int deep){
    k++;E[k]=t;L[k]=deep;H[t]=k;
    for(int i=head[t];i!=-1;i=EE[i].next){
        int tt=EE[i].to;
        if(!vis[tt]){
            vis[tt]=1;pre[tt]=t;
            dfs(tt,deep+1);
            k++;E[k]=t;L[k]=deep;
        }
    }
}
void RMQ_init(){
    for(int i=1;i<=2*n-1;i++) dp[i][0]=i;
    for(int i=1;(1<ri) swap(le,ri);
    int kk=0;
    while((1<<(kk+1))<=ri-le+1) kk++;
    if(L[dp[le][kk]]L[H[b.f]];
}
void visdfs(int x){
    vis[x]=1;
    for(int i=head[x];i!=-1;i=EE[i].next){
        int t=EE[i].to;
        if(t==pre[x]||vis[t]) continue;
        visdfs(t);
    }
}
int main(){
    int q,u,v;
    while(scanf("%d%d",&n,&q)!=-1){
        for(int i=0;i<=n;i++) vis[i]=0;
        memset(head,-1,sizeof(head));
        for(int i=0;i

你可能感兴趣的:(图论,LCA,线段树)