【线段树】1108E2 Array and Segments (Hard version) & 1108E1 Array and Segments (Easy version)

The only difference between easy and hard versions is a number of elements in the array.

You are given an array a consisting of n integers. The value of the i-th element of the array is ai.

You are also given a set of m segments. The j-th segment is [lj;rj], where 1≤lj≤rj≤n.

You can choose some subset of the given set of segments and decrease values on each of the chosen segments by one (independently). For example, if the initial array a=[0,0,0,0,0] and the given segments are [1;3] and [2;4] then you can choose both of them and the array will become b=[−1,−2,−2,−1,0].

You have to choose some subset of the given segments (each segment can be chosen at most once) in such a way that if you apply this subset of segments to the array a and obtain the array b then the value在这里插入图片描述will be maximum possible.

Note that you can choose the empty set.

If there are multiple answers, you can print any.

If you are Python programmer, consider using PyPy instead of Python when you submit your code.

Input

The first line of the input contains two integers n and m (1≤n≤105,0≤m≤300) — the length of the array a and the number of segments, respectively.

The second line of the input contains n integers a1,a2,…,an (−106≤ai≤106), where ai is the value of the i-th element of the array a.

The next m lines are contain two integers each. The j-th of them contains two integers lj and rj (1≤lj≤rj≤n), where lj and rj are the ends of the j-th segment.

Output
In the first line of the output print one integer d— the maximum possible value maxi=1nbi−mini=1nbi if b is the array obtained by applying some subset of the given segments to the array a.

In the second line of the output print one integer q (0≤q≤m) — the number of segments you apply.

In the third line print q distinct integers c1,c2,…,cq in any order (1≤ck≤m) — indices of segments you apply to the array a in such a way that the value maxi=1nbi−mini=1nbi of the obtained array b is maximum possible.

If there are multiple answers, you can print any.

Input
5 4
2 -2 3 1 2
1 3
4 5
2 5
1 3

Output
6
2
1 4

Input
5 4
2 -2 3 1 4
3 5
3 4
2 4
2 5

Output
7
2
2 3

Input
1 0
1000000

Output
0
0

Note

In the first example the obtained array b will be [0,−4,1,1,2] so the answer is 6.
In the second example the obtained array b will be [2,−3,1,−1,4] so the answer is 7.

In the third example you cannot do anything so the answer is 0.

题意

  • 给定一个长度为n的序列,和m个区间。
  • 对一个区间的操作是:对整个区间的数-1
  • 可以选择任意个区间(可以为0个、每个区间最多被选择一次)进行操作后,要求最大化的序列极差(极差即最大值 - 最小值)。
  • easy version的范围是(1≤n≤300,0≤m≤300)
  • hard version的范围是(1≤n≤1e5,0≤m≤300)

思路

  • 操作是对区间内的数减1,因此最大值不会变的更大,要使得极差变大,只能够尽量使最小值变小。
  • 如果对一个区间进行操作,可能会有以下情况:
    1.这个区间只包含最小值,那么最小值变小,极差变大。 这是我们需要的。
    2.这个区间只包含最大值,那么最大值变大,极差变小 。这显然是要避免的。
    3.这个区间既包含最大值,又包含最小值,那么最大值和最小值同时变小,且变小幅度一致,极差不变。 这对我们来说是没有影响的。
  • 要最大化极差,显然要尽量进行1操作,不进行2操作,3操作无所谓。所以在选择区间时,只要是包含最小值的,我们都要选择这个区间进行操作。
  • 但是我们不知道最大值,最小值到底在哪里,他们在操作过程中是变化的。比如在样例1中,最开始最大值是第三位3,最后变成了第五位2。
  • 由于最大值最小值的不确定性,我们需要枚举最小值的位置。记录操作过程中,哪个位置作为最终最小值位置时,能够使得极差最大化。
  • 枚举了最小值的位置后,如果暴力进行区间修改操作,其复杂度是很大的,可以过easy version(我的easy version代码就是暴力的)。
精妙之处
  • 我们可以对操作区间进行整理,使得在枚举过程中,不需要重复进行区间操作。
  • 比如我们现在枚举的最小值位置是p,则我们需要选择包含了p的所有区间。
  • 那么p前面一个状态p-1,进行的区间操作是包含了p-1的所有区间,这些p-1区间中,有些是包含了p的,有些是没包含p的,那么我们需要先删除掉没包含p的区间的影响(即把这些区间减掉的1给加回来)。同时还需要补上包含了p但是不包含p-1的区间(这样选择,就不会和p-1里面保留了的区间重复)。
  • 包含了p-1,但不包含p的区间,显然是以p-1结尾的区间。我们用add[]数组保存以r[i]结尾的区间的编号。
  • 包含了p,但不包含p-1的区间,显然是以p开始的区间。我们用sub[]数组保存以l[i]开始的区间的编号。
  • 区间修改操作,就要用到线段树了。
  • 最后输出答案时,使用最终确定的最小值的位置P,包含P的区间留下,不包含P的舍弃。

AC代码

#include <bits/stdc++.h>

using namespace std;
const int maxn = 1e5 + 5;
const int maxm = 305;

int n, m;
int a[maxn], l[maxm], r[maxm], ans[maxn];
vector<int> add[maxn], sub[maxn];

struct node
{
    int l, r;
    int maxx, minn, lazy;
} tree[maxn << 2];

void push_up(int k)
{
    tree[k].maxx = max(tree[k << 1].maxx, tree[k << 1 | 1].maxx);
    tree[k].minn = min(tree[k << 1].minn, tree[k << 1 | 1].minn);
}

void push_down(int k)
{
    if(tree[k].l != tree[k].r)
    {
        tree[k << 1].maxx += tree[k].lazy, tree[k << 1].minn += tree[k].lazy;
        tree[k << 1 | 1].maxx += tree[k].lazy, tree[k << 1 | 1].minn += tree[k].lazy;
        tree[k << 1].lazy += tree[k].lazy;
        tree[k << 1 | 1].lazy += tree[k].lazy;
    }
    tree[k].lazy = 0;
}

void build(int k, int l, int r)
{
    tree[k].l = l;
    tree[k].r = r;
    if(l == r)
    {
        tree[k].maxx = a[l];
        tree[k].minn = a[l];
        tree[k].lazy = 0;
        return ;
    }
    int mid = (l + r) >> 1;
    build(k << 1, l, mid);
    build(k << 1 | 1, mid + 1, r);
    push_up(k);
}

void update(int k, int l, int r, int x)
{
    if(tree[k].lazy)
        push_down(k);
    tree[k].maxx += x;
    tree[k].minn += x;
    if(tree[k].l == l && tree[k].r == r)
    {
        tree[k].lazy += x;
        return ;
    }
    int mid = (tree[k].l + tree[k].r) >> 1;
    if(r <= mid)
        update(k << 1, l, r, x);
    else if(l > mid)
        update(k << 1 | 1, l, r, x);
    else
    {
        update(k << 1, l, mid, x);
        update(k << 1 | 1, mid + 1, r, x);
    }
    push_up(k);
}

int query_max(int k, int l, int r)
{
    int maxx;
    if(tree[k].lazy)
        push_down(k);
    if(tree[k].l == l && tree[k].r == r)
        return tree[k].maxx;
    int mid = (tree[k].l + tree[k].r) >> 1;
    if(r <= mid)
        maxx = query_max(k << 1, l, r);
    else if(l > mid)
        maxx = query_max(k << 1 | 1, l, r);
    else
        maxx = max(query_max(k << 1, l, mid), query_max(k << 1 | 1, mid + 1, r));
    return maxx;
}

int query_min(int k, int l, int r)
{
    int minn;
    if(tree[k].lazy)
        push_down(k);
    if(tree[k].l == l && tree[k].r == r)
        return tree[k].minn;
    int mid = (tree[k].l + tree[k].r) >> 1;
    if(r <= mid)
        minn = query_min(k << 1, l, r);
    else if(l > mid)
        minn = query_min(k << 1 | 1, l, r);
    else
        minn = min(query_min(k << 1, l, mid), query_min(k << 1 | 1, mid + 1, r));
    return minn;
}

int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 1; i <= n; i++)
        scanf("%d", &a[i]);

    for(int i = 1; i <= m; i++)
    {
        scanf("%d%d", &l[i], &r[i]);

        //统计区间
        sub[ l[i] ].push_back(i);
        add[ r[i] ].push_back(i);
    }

    //构造线段树
    build(1, 1, n);

    int d = -1, p;//d记录最大极差,p记录最小值的位置
    for(int i = 1; i <= n; i++)//枚举最小值出现的位置
    {
        for(int j = 0; j < add[i - 1].size(); j++)//消除包含i-1,不包含i区间的影响
        {
            int id = add[i - 1][j];//注意这里是i-1
            update(1, l[id], r[id], 1);
        }

        for(int j = 0; j < sub[i].size(); j++)//添加包含i,不包含i-1区间的影响
        {
            int id = sub[i][j];
            update(1, l[id], r[id], -1);
        }

        //查询极差 注意这里虽然知道最小值是a[i],但是由于使用了线段树,有lazy标记,标记可能没有更新到最底层,所以不能直接使用a[i]。
        int det = query_max(1, 1, n) - query_min(1, 1, n);
        if(det > d)
        {
            d = det;
            p = i;
        }
    }

    printf("%d\n", d);
    int t = 0;
    for(int i = 1; i <= m; i++)//统计区间
    {
        if(l[i] <= p && p <= r[i])
            ans[t++] = i;
    }
    printf("%d\n", t);
    for(int i = 0; i < t; i++)
    {
        if(i != 0)
            printf(" ");
        printf("%d", ans[i]);
    }
    printf("\n");
    return 0;
}

### 回答1: 线段相交问题是计算几何学中的常见问题,主要目标是确定两个给定线段是否相交。在解决这个问题时,我们可以使用几何知识和数学方法,下面是一种常见的解决方案: 首先,我们可以将每个线段表示为两个端点的坐标。对于线段AB,我们可以表示为A(x1, y1)和B(x2, y2)。同样的,对于线段CD,我们可以表示为C(x3, y3)和D(x4, y4)。 然后,我们可以利用一系列关系来判断线段是否相交。首先,我们可以通过比较两个线段的最小和最大x坐标来判断它们是否在同一平面上。如果线段AB的最小x坐标大于线段CD的最大x坐标,或者线段AB的最大x坐标小于线段CD的最小x坐标,则可以判断它们不会相交。 接下来,我们可以利用向量的叉积来判断两个线段是否共线。我们可以计算向量AB和向量AC的叉积以及向量CD和向量CA的叉积。如果这两个叉积乘积小于0,则可以判断线段AB和线段CD相交。 最后,我们需要考虑一些特殊情况,例如两个线段共线但没有重叠部分的情况,或者两个线段有一个公共端点的情况。这些情况可以通过包含更多的条件来进一步判断。 通过以上的方法,我们可以相对准确地判断两个线段是否相交。这个问题在计算几何学和计算机图形学中有广泛的应用,例如在碰撞检测、路径规划和游戏开发中都可以使用到。 ### 回答2: line-segments-intersect是一个用于确定两个线段是否相交的算法。这个算法的目标是判断给定的两个线段是否存在交点,若存在,则认为两个线段相交,否则认为它们不相交。 这个算法可以通过以下步骤来实现: 1. 首先,我们需要确定每个线段的两个端点的坐标。假设第一个线段的端点分别为A(x1, y1)和B(x2, y2),第二个线段的端点分别为C(x3, y3)和D(x4, y4)。 2. 接下来,我们需要利用线段AB的斜率和CD的斜率来判断它们是否平行。如果两条线段的斜率相等,那么它们是平行的,此时它们不会相交。 3. 如果线段AB和CD不平行,我们进一步判断它们是否相交。我们可以使用线段的方程来计算两条线段的交点。如果两条线段的交点的x坐标和y坐标都在两个线段的范围内,则认为它们相交。 4. 如果以上条件都不满足,则认为两条线段不相交。 通过以上步骤,我们可以确定两个线段是否相交。这个算法可以在计算机程序中实现,并可以用于各种应用场景,如计算几何问题、计算路径交叉等。但需要注意的是,线段的相交判断需要考虑特殊情况,如端点重合、线段长度为0等,以保证算法的准确性和稳定性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值