【链接】h在这里写链接


【题意】


给你一棵树,每个节点上都有一个权值.
然后给你m个询问,每个询问(x,y,a,b);
表示询问x->y这条路径上权值在[a,b]范围内的节点的权值和.

【题解】


树链剖分题。
在树链上建一个线段树,线段树的每个节点存3个值,max[i],min[i],sum[i]分别表示这个区间里面的数的最大值、最小值、以及权值和。
然后在线段树上做查找(a,b)范围内的数的权值和。
如果该节点在路径上,那么如果a<=min[i] && max[i] <= b的话,直接返回sum[i],否则如果不在范围里,直接范围0(肯定不在的话);
如果没办法确定在不在,则继续往下走。

【错的次数】


2

【反思】


关了同步之后,不能用puts("");

【代码】

#include <iostream>
#include <algorithm>
#include <vector>
using namespace std; const int N = 1e5; int n, m, c[N + 10], uppest[N + 10], idx[N + 10], cnt, sz[N + 10], dep[N + 10];
long long sum[(N + 10) << 2];
int mi[(N + 10) << 2], ma[(N + 10) << 2], a, b, fat[N + 10];
vector <int> g[N + 10]; void dfs1(int x, int fa) {
    sz[x] = 1;
    for (int y : g[x]) {
        if (y == fa) continue;
        dep[y] = dep[x] + 1;
        dfs1(y, x);
        sz[x] += sz[y];
        fat[y] = x;
    }
} void dfs2(int x, int chain) {
    idx[x] = ++cnt;
    uppest[x] = chain;
    int ma = 0;
    for (int y : g[x]) {
        if (dep[y] > dep[x] && sz[y] > sz[ma]) {
            ma = y;
        }
    }
    if (ma == 0) return;
    dfs2(ma, chain);
    for (int y : g[x])
        if (dep[y] > dep[x] && y != ma)
            dfs2(y, y);
} void updata(int pos, int x, int l, int r, int rt) {
    if (l == r) {
        mi[rt] = ma[rt] = x;
        sum[rt] = x;
        return;
    }
    int m = (l + r) >> 1;
    if (pos <= m)
        updata(pos, x, l, m, rt << 1);
    else
        updata(pos, x, m + 1, r, rt << 1 | 1);
    ma[rt] = max(ma[rt << 1 | 1], ma[rt << 1]);
    mi[rt] = min(mi[rt << 1 | 1], mi[rt << 1]);
    sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
} void pre() {
    for (int i = 1; i <= n; i++)
        updata(idx[i], c[i], 1, n, 1);
} long long get_ans(int L, int R, int l, int r, int rt) {
    if (L <= l && r <= R) {
        if (ma[rt] < a) return 0;
        if (mi[rt] > b) return 0;
        if (a <= mi[rt] && ma[rt] <= b) return sum[rt];
    }
    int m = (l + r) >> 1;
    long long temp = 0;
    if (L <= m)
        temp += get_ans(L, R, l, m, rt << 1);
    if (m < R)
        temp += get_ans(L, R, m + 1, r, rt << 1 | 1);
    return temp;
} long long calc(int x, int y) {
    long long temp = 0;
    while (uppest[x] != uppest[y]) {
        if (dep[uppest[x]] > dep[uppest[y]]) swap(x, y);
        //dep[x] < dep[y] ok
        temp += get_ans(idx[uppest[y]], idx[y], 1, n, 1);
        y = fat[uppest[y]];
    }
    if (dep[x] > dep[y]) swap(x, y);
    temp += get_ans(idx[x], idx[y], 1, n, 1);
    return temp;
} void deal() {
    for (int i = 1; i <= m; i++) {
        int s, t;
        cin >> s >> t >> a >> b;
        cout << calc(s, t);
        if (i == m)
            cout << endl;
        else
            cout << ' ';
    }
} int main() {
    //freopen("F:\\rush.txt", "r", stdin);
    ios::sync_with_stdio(0), cin.tie(0);     while (cin >> n >> m) {
        for (int i = 1; i <= n; i++) cin >> c[i];
        for (int i = 1; i <= n; i++) g[i].clear();
        for (int i = 1; i <= n - 1; i++) {
            int x, y;
            cin >> x >> y;
            g[x].push_back(y), g[y].push_back(x);
        }         dep[1] = 0;
        dfs1(1, 0);
        cnt = 0;
        dfs2(1, 1);         pre();         deal();
    }
    return 0;
}

最新文章

  1. SQLite使用教程4 附加数据库
  2. 包含为 HTTP 定义的状态代码的值(枚举)
  3. Optimal Logging
  4. 树莓派入手(烧写系统,调整分区,配置Java环境,串口GPS配置) 分类: Raspberry Pi 2015-04-09 21:13 145人阅读 评论(0) 收藏
  5. jQueryMobile之listview
  6. Qt线程QThread简析(8个线程等级,在UI线程里可调用thread-&gt;wait()等待线程结束,exit()可直接退出线程,setStackSize设置线程堆栈,首次见到Qt::HANDLE,QThreadData和QThreadPrivate)
  7. ThinkPad E431/E531 ubuntu 14.04 安装无线网卡驱动
  8. Kafka Eagle 源码解读
  9. WIN2003+IIS6+FastCGI+PHP5.3的安装配置
  10. Javaweb-request与response
  11. C#工作总结(一):Fleck的WebSocket使用
  12. js 时间戳转换为‘yyyy-MM-dd hh:mm’格式(es6语法)
  13. hosts.allow和hosts.deny支持哪些服务
  14. nginx实现unigui群集
  15. magento2 重置后台密码
  16. 设置PyCharm中的Python代码模版
  17. 每日英语:Secrets Of Effective Office Humor
  18. 面试官问:JS的this指向
  19. 51Nod 1006:最长公共子序列Lcs(打印LCS)
  20. 蚂蚁金服SOFAMesh在多语言上的实践

热门文章

  1. jni中调用java方法获取当前apk的签名文件md5值
  2. c#+windows api SetWindowsHookEx 全局钩子 demo 下载
  3. PhoneGap/Cordova Android应用签名公布注意事项
  4. 论Nim中的 proc 和 method
  5. 如何分解json值设置到text文本框中
  6. ActionListener三种实现
  7. sparksql不支持hive中的分区名称大写
  8. 【Codeforces Round #456 (Div. 2) B】New Year&#39;s Eve
  9. code blocks主题颜色配置
  10. Filebeat的下载(图文讲解)