首页 > 其他分享 >2025-01-04:不包含相邻元素的子序列的最大和。用go语言,给定一个整数数组 nums 和一个由二维数组 queries 组成的查询列表,其中每个查询的格式为 queries[i] = [pos

2025-01-04:不包含相邻元素的子序列的最大和。用go语言,给定一个整数数组 nums 和一个由二维数组 queries 组成的查询列表,其中每个查询的格式为 queries[i] = [pos

时间:2025-01-04 16:01:39浏览次数:1  
标签:nums int self v11 tree st 数组 queries 查询

2025-01-04:不包含相邻元素的子序列的最大和。用go语言,给定一个整数数组 nums 和一个由二维数组 queries 组成的查询列表,其中每个查询的格式为 queries[i] = [posi, xi]。

对于每个查询 i,首先将 nums[posi] 的值更新为 xi,然后计算在这一更新后,数组 nums 中所有不包含相邻元素的子序列的最大和。

最后,返回所有查询的结果之和。需要注意的是,由于最终答案可能非常大,因此要对其进行 1000000007 的取余处理。

请根据以上描述进行相关的处理。

1 <= nums.length <= 5 * 10000。

-100000 <= nums[i] <= 100000。

1 <= queries.length <= 5 * 10000。

queries[i] == [posi, xi]。

0 <= posi <= nums.length - 1。

-100000 <= xi <= 100000。

答案2025-01-04:

chatgpt

题目来自leetcode3165。

大体步骤如下:

1.定义了一个常量 MOD 为 1000000007,表示取余处理的数值。

2.实现了一个函数 maximumSumSubsequence,该函数接受一个整数数组 nums 以及一个查询列表 queries。首先创建一个长度为 nums 数组长度四倍的线段树 tree,然后初始化这颗线段树根据传入的 nums 数组。接着对 queries 中的每个查询进行处理:更新 nums 中指定位置的值,并计算不包含相邻元素的子序列的最大和,并将结果取余加到 ans 中。最终返回 ans。

3.定义了一个结构体 SegNode,包含四个成员变量 v00、v01、v10、v11,表示线段树中的四种情况。

4.实现了两个 SegNode 结构体的方法:Set 和 Best,分别用于设置节点的值和获取最佳值。

5.定义了一个结构体 SegTree,包含了一个整数 n 和一个指向 SegNode 结构体数组的指针 tree。

6.实现了一个 NewSegTree 函数用于创建一个 SegTree 结构体并初始化相关信息。

7.实现了 SegTree 结构体的方法:Init、Update、Query、internalInit、internalUpdate、pushup。这些方法用于初始化线段树、更新节点值、查询最佳值等功能。

8.在 main 函数中,给定了一个示例数组 nums 和查询 queries,然后调用 maximumSumSubsequence 函数计算不包含相邻元素的子序列的最大和,并打印结果。

总的时间复杂度:

  • 初始化线段树的时间复杂度为 O(n)。

  • 每次查询的时间复杂度为 O(logn)。

  • 因此,总的时间复杂度为 O(n + q*logn),其中 n 为数组长度,q 为查询次数。

总的额外空间复杂度:

  • 线段树的空间复杂度为 O(n)。

  • 因此,总的额外空间复杂度为 O(n),其中 n 为数组长度。

Go完整代码如下:

package main

import (
	"fmt"
	"math"
)

const MOD = 1000000007

func maximumSumSubsequence(nums []int, queries [][]int) int {
    n := len(nums)
    tree := NewSegTree(n)
    tree.Init(nums)

    ans := int64(0)
    for _, q := range queries {
        tree.Update(q[0], q[1])
        ans = (ans + tree.Query()) % MOD
    }
    return int(ans)
}

type SegNode struct {
    v00, v01, v10, v11 int64
}

func NewSegNode() *SegNode {
    return &SegNode{0, 0, 0, 0}
}

func (sn *SegNode) Set(v int64) {
    sn.v00, sn.v01, sn.v10 = 0, 0, 0
    sn.v11 = int64(math.Max(float64(v), 0))
}

func (sn *SegNode) Best() int64 {
    return sn.v11
}

type SegTree struct {
    n    int
    tree []*SegNode
}

func NewSegTree(n int) *SegTree {
    tree := make([]*SegNode, n * 4 + 1)
    for i := range tree {
        tree[i] = NewSegNode()
    }
    return &SegTree{n, tree}
}

func (st *SegTree) Init(nums []int) {
    st.internalInit(nums, 1, 1, st.n)
}

func (st *SegTree) Update(x, v int) {
    st.internalUpdate(1, 1, st.n, x + 1, int64(v))
}

func (st *SegTree) Query() int64 {
    return st.tree[1].Best()
}

func (st *SegTree) internalInit(nums []int, x, l, r int) {
    if l == r {
        st.tree[x].Set(int64(nums[l - 1]))
        return
    }
    mid := (l + r) / 2
    st.internalInit(nums, x * 2, l, mid)
    st.internalInit(nums, x * 2 + 1, mid + 1, r)
    st.pushup(x)
}

func (st *SegTree) internalUpdate(x, l, r int, pos int, v int64) {
    if l > pos || r < pos {
        return
    }
    if l == r {
        st.tree[x].Set(v)
        return
    }
    mid := (l + r) / 2
    st.internalUpdate(x * 2, l, mid, pos, v)
    st.internalUpdate(x * 2 + 1, mid + 1, r, pos, v)
    st.pushup(x)
}

func (st *SegTree) pushup(x int) {
    l, r := x * 2, x * 2 + 1
    st.tree[x].v00 = max(st.tree[l].v00 + st.tree[r].v10, st.tree[l].v01 + st.tree[r].v00)
    st.tree[x].v01 = max(st.tree[l].v00 + st.tree[r].v11, st.tree[l].v01 + st.tree[r].v01)
    st.tree[x].v10 = max(st.tree[l].v10 + st.tree[r].v10, st.tree[l].v11 + st.tree[r].v00)
    st.tree[x].v11 = max(st.tree[l].v10 + st.tree[r].v11, st.tree[l].v11 + st.tree[r].v01)
}


func main() {
	nums := []int{3,5,9}
	queries := [][]int{{1,-2},{0,-3}}
	result := maximumSumSubsequence(nums, queries)
	fmt.Println(result)
}

在这里插入图片描述

Rust完整代码如下:

use std::cmp::max;

const MOD: i64 = 1_000_000_007;

#[derive(Clone)]
struct SegNode {
    v00: i64,
    v01: i64,
    v10: i64,
    v11: i64,
}

impl SegNode {
    fn new() -> Self {
        SegNode {
            v00: 0,
            v01: 0,
            v10: 0,
            v11: 0,
        }
    }

    fn set(&mut self, v: i64) {
        self.v00 = 0;
        self.v01 = 0;
        self.v10 = 0;
        self.v11 = max(v, 0);
    }

    fn best(&self) -> i64 {
        self.v11
    }
}

struct SegTree {
    n: usize,
    tree: Vec<SegNode>,
}

impl SegTree {
    fn new(n: usize) -> Self {
        let tree = vec![SegNode::new(); n * 4];
        SegTree { n, tree }
    }

    fn init(&mut self, nums: &[i32]) {
        self.internal_init(nums, 1, 1, self.n);
    }

    fn update(&mut self, pos: usize, v: i32) {
        self.internal_update(1, 1, self.n, pos + 1, v as i64);
    }

    fn query(&self) -> i64 {
        self.tree[1].best()
    }

    fn internal_init(&mut self, nums: &[i32], x: usize, l: usize, r: usize) {
        if l == r {
            self.tree[x].set(nums[l - 1] as i64);
            return;
        }
        let mid = (l + r) / 2;
        self.internal_init(nums, x * 2, l, mid);
        self.internal_init(nums, x * 2 + 1, mid + 1, r);
        self.push_up(x);
    }

    fn internal_update(&mut self, x: usize, l: usize, r: usize, pos: usize, v: i64) {
        if l > pos || r < pos {
            return;
        }
        if l == r {
            self.tree[x].set(v);
            return;
        }
        let mid = (l + r) / 2;
        self.internal_update(x * 2, l, mid, pos, v);
        self.internal_update(x * 2 + 1, mid + 1, r, pos, v);
        self.push_up(x);
    }

    fn push_up(&mut self, x: usize) {
        let l = x * 2;
        let r = x * 2 + 1;
        self.tree[x].v00 = max(
            self.tree[l].v00 + self.tree[r].v10,
            self.tree[l].v01 + self.tree[r].v00,
        );
        self.tree[x].v01 = max(
            self.tree[l].v00 + self.tree[r].v11,
            self.tree[l].v01 + self.tree[r].v01,
        );
        self.tree[x].v10 = max(
            self.tree[l].v10 + self.tree[r].v10,
            self.tree[l].v11 + self.tree[r].v00,
        );
        self.tree[x].v11 = max(
            self.tree[l].v10 + self.tree[r].v11,
            self.tree[l].v11 + self.tree[r].v01,
        );
    }
}

fn maximum_sum_subsequence(nums: &[i32], queries: &[(usize, i32)]) -> i64 {
    let n = nums.len();
    let mut tree = SegTree::new(n);
    tree.init(nums);
    let mut ans = 0;

    for (x, v) in queries {
        tree.update(*x, *v);
        ans = (ans + tree.query()) % MOD;
    }

    ans
}

fn main() {
    let nums = vec![3, 5, 9];
    let queries = vec![(1, -2), (0, -3)];
    let result = maximum_sum_subsequence(&nums, &queries);
    println!("{}", result);
}

在这里插入图片描述

C完整代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <math.h>

#define MOD 1000000007

typedef struct {
    long long v00, v01, v10, v11;
} SegNode;

typedef struct {
    int n;
    SegNode* tree;
} SegTree;

SegNode newSegNode() {
    SegNode node;
    node.v00 = 0;
    node.v01 = 0;
    node.v10 = 0;
    node.v11 = 0;
    return node;
}

void setSegNode(SegNode* sn, long long v) {
    sn->v00 = 0;
    sn->v01 = 0;
    sn->v10 = 0;
    sn->v11 = fmax(v, 0);
}

long long bestSegNode(SegNode* sn) {
    return sn->v11;
}

SegTree* newSegTree(int n) {
    SegTree* tree = (SegTree*)malloc(sizeof(SegTree));
    tree->n = n;
    tree->tree = (SegNode*)malloc(sizeof(SegNode) * (4 * n + 1));

    for (int i = 0; i < 4 * n + 1; i++) {
        tree->tree[i] = newSegNode();
    }

    return tree;
}

void pushup(SegTree* st, int x);

void internalInit(SegTree* st, int* nums, int x, int l, int r) {
    if (l == r) {
        setSegNode(&st->tree[x], (long long)nums[l - 1]);
        return;
    }
    int mid = (l + r) / 2;
    internalInit(st, nums, x * 2, l, mid);
    internalInit(st, nums, x * 2 + 1, mid + 1, r);
    pushup(st, x);
}

void pushup(SegTree* st, int x) {
    int l = x * 2;
    int r = x * 2 + 1;
    st->tree[x].v00 = fmax(st->tree[l].v00 + st->tree[r].v10, st->tree[l].v01 + st->tree[r].v00);
    st->tree[x].v01 = fmax(st->tree[l].v00 + st->tree[r].v11, st->tree[l].v01 + st->tree[r].v01);
    st->tree[x].v10 = fmax(st->tree[l].v10 + st->tree[r].v10, st->tree[l].v11 + st->tree[r].v00);
    st->tree[x].v11 = fmax(st->tree[l].v10 + st->tree[r].v11, st->tree[l].v11 + st->tree[r].v01);
}

void internalUpdate(SegTree* st, int x, int l, int r, int pos, long long v) {
    if (l > pos || r < pos) {
        return;
    }
    if (l == r) {
        setSegNode(&st->tree[x], v);
        return;
    }
    int mid = (l + r) / 2;
    internalUpdate(st, x * 2, l, mid, pos, v);
    internalUpdate(st, x * 2 + 1, mid + 1, r, pos, v);
    pushup(st, x);
}

long long query(SegTree* st) {
    return bestSegNode(&st->tree[1]);
}

void initSegTree(SegTree* st, int* nums) {
    internalInit(st, nums, 1, 1, st->n);
}

void updateSegTree(SegTree* st, int pos, int v) {
    internalUpdate(st, 1, 1, st->n, pos + 1, v);
}

long long maximumSumSubsequence(int* nums, int numsSize, int(*queries)[2], int queriesSize) {
    SegTree* tree = newSegTree(numsSize);
    initSegTree(tree, nums);
    long long ans = 0;

    for (int i = 0; i < queriesSize; i++) {
        updateSegTree(tree, queries[i][0], queries[i][1]);
        ans = (ans + query(tree)) % MOD;
    }

    // Free allocated memory
    free(tree->tree);
    free(tree);
    return ans;
}

int main() {
    int nums[] = { 3, 5, 9 };
    int queries[2][2] = { {1, -2}, {0, -3} };
    long long result = maximumSumSubsequence(nums, 3, queries, 2);
    printf("%lld\n", result);
    return 0;
}

在这里插入图片描述

C++完整代码如下:

#include <iostream>
#include <vector>
#include <algorithm>
#include <cmath>

const int MOD = 1000000007;

class SegNode {
public:
    long long v00, v01, v10, v11;

    SegNode() : v00(0), v01(0), v10(0), v11(0) {}

    void set(long long v) {
        v00 = 0;
        v01 = 0;
        v10 = 0;
        v11 = std::max(v, 0LL);
    }

    long long best() const {
        return v11;
    }
};

class SegTree {
private:
    int n;
    std::vector<SegNode> tree;

    void pushup(int x) {
        int l = x * 2;
        int r = x * 2 + 1;
        tree[x].v00 = std::max(tree[l].v00 + tree[r].v10, tree[l].v01 + tree[r].v00);
        tree[x].v01 = std::max(tree[l].v00 + tree[r].v11, tree[l].v01 + tree[r].v01);
        tree[x].v10 = std::max(tree[l].v10 + tree[r].v10, tree[l].v11 + tree[r].v00);
        tree[x].v11 = std::max(tree[l].v10 + tree[r].v11, tree[l].v11 + tree[r].v01);
    }

    void internalInit(const std::vector<int>& nums, int x, int l, int r) {
        if (l == r) {
            tree[x].set(static_cast<long long>(nums[l - 1]));
            return;
        }
        int mid = (l + r) / 2;
        internalInit(nums, x * 2, l, mid);
        internalInit(nums, x * 2 + 1, mid + 1, r);
        pushup(x);
    }

    void internalUpdate(int x, int l, int r, int pos, long long v) {
        if (l > pos || r < pos) {
            return;
        }
        if (l == r) {
            tree[x].set(v);
            return;
        }
        int mid = (l + r) / 2;
        internalUpdate(x * 2, l, mid, pos, v);
        internalUpdate(x * 2 + 1, mid + 1, r, pos, v);
        pushup(x);
    }

public:
    SegTree(int n) : n(n) {
        tree.resize(n * 4);
    }

    void init(const std::vector<int>& nums) {
        internalInit(nums, 1, 1, n);
    }

    void update(int pos, int v) {
        internalUpdate(1, 1, n, pos + 1, static_cast<long long>(v));
    }

    long long query() const {
        return tree[1].best();
    }
};

long long maximumSumSubsequence(const std::vector<int>& nums, const std::vector<std::pair<int, int>>& queries) {
    int n = nums.size();
    SegTree tree(n);
    tree.init(nums);
    long long ans = 0;

    for (const auto& query : queries) {
        tree.update(query.first, query.second);
        ans = (ans + tree.query()) % MOD;
    }

    return ans;
}

int main() {
    std::vector<int> nums = { 3, 5, 9 };
    std::vector<std::pair<int, int>> queries = { {1, -2}, {0, -3} };
    long long result = maximumSumSubsequence(nums, queries);
    std::cout << result << std::endl;
    return 0;
}

在这里插入图片描述

Python完整代码如下:

# -*-coding:utf-8-*-

class SegNode:
    def __init__(self):
        self.v00 = 0
        self.v01 = 0
        self.v10 = 0
        self.v11 = 0

    def set(self, v):
        self.v00 = 0
        self.v01 = 0
        self.v10 = 0
        self.v11 = max(v, 0)

    def best(self):
        return self.v11


class SegTree:
    def __init__(self, n):
        self.n = n
        self.tree = [SegNode() for _ in range(n * 4 + 1)]

    def init(self, nums):
        self._internal_init(nums, 1, 1, self.n)

    def update(self, x, v):
        self._internal_update(1, 1, self.n, x + 1, v)

    def query(self):
        return self.tree[1].best()

    def _internal_init(self, nums, x, l, r):
        if l == r:
            self.tree[x].set(nums[l - 1])
            return
        mid = (l + r) // 2
        self._internal_init(nums, x * 2, l, mid)
        self._internal_init(nums, x * 2 + 1, mid + 1, r)
        self._pushup(x)

    def _internal_update(self, x, l, r, pos, v):
        if l > pos or r < pos:
            return
        if l == r:
            self.tree[x].set(v)
            return
        mid = (l + r) // 2
        self._internal_update(x * 2, l, mid, pos, v)
        self._internal_update(x * 2 + 1, mid + 1, r, pos, v)
        self._pushup(x)

    def _pushup(self, x):
        l, r = x * 2, x * 2 + 1
        self.tree[x].v00 = max(self.tree[l].v00 + self.tree[r].v10, self.tree[l].v01 + self.tree[r].v00)
        self.tree[x].v01 = max(self.tree[l].v00 + self.tree[r].v11, self.tree[l].v01 + self.tree[r].v01)
        self.tree[x].v10 = max(self.tree[l].v10 + self.tree[r].v10, self.tree[l].v11 + self.tree[r].v00)
        self.tree[x].v11 = max(self.tree[l].v10 + self.tree[r].v11, self.tree[l].v11 + self.tree[r].v01)


MOD = 1000000007

def maximum_sum_subsequence(nums, queries):
    n = len(nums)
    tree = SegTree(n)
    tree.init(nums)

    ans = 0
    for q in queries:
        tree.update(q[0], q[1])
        ans = (ans + tree.query()) % MOD
    return ans


if __name__ == "__main__":
    nums = [3, 5, 9]
    queries = [[1, -2], [0, -3]]
    result = maximum_sum_subsequence(nums, queries)
    print(result)

在这里插入图片描述

标签:nums,int,self,v11,tree,st,数组,queries,查询
From: https://www.cnblogs.com/moonfdd/p/18652001

相关文章

  • posggres 的聚合查询,记录数好奇怪:
    我的测试环境如何产生数据的:用sysbench生成和测试过!sysbench--db-driver=pgsql--pgsql-host=127.0.0.1--pgsql-port=5432--pgsql-user=test02--pgsql-password=test02--pgsql-db=postgres--oltp-table-size=200000--oltp-tables-count=10--rand-init=on--threads=10......
  • BUGAWAY算法小抄-差分数组
    BUGAWAY算法小抄-差分数组什么是差分数组?差分数组的思想是通过对原始数组进行处理,得到一个新的数组(差分数组),利用该数组来高效地进行区间更新操作。具体来说,差分数组记录的是相邻元素之间的差值,而不是原始数组的元素本身。差分数组的原理1.差分数组的构造:假设有一个数组A=......
  • 使用Docker部署简单实用的 IP查询工具箱 【复制成功,复制和转载请标注本文地址】
    通过这个工具,可以查看需要查询的公网IP地址、连通性、检查WebRTC连接、检查DNS、网速测试、MTR测试等。假如你不想部署到本地,也可以直接体验官方网址:https://ipcheck.ing下面就开始部署:只需要一行命令就可以一键安装到docker,端口自行修改。部署起来非常的简单,使用SSH终端......
  • 特殊数据类型的深度分析:JSON、数组和 HSTORE 的实用价值
    title:特殊数据类型的深度分析:JSON、数组和HSTORE的实用价值date:2025/1/4updated:2025/1/4author:cmdragonexcerpt:随着数据管理需求的多样化,许多现代数据库系统开始支持特殊数据类型,以满足更多复杂应用场景的需求。在PostgreSQL中,JSON、数组和HSTORE类型为开......
  • leetCode 33:搜索旋转排序数组
    题目:整数数组nums按升序排列,数组中的值互不相同。在传递给函数之前,nums在预先未知的某个下标k(0<=k<nums.length)上进行了旋转,使数组变为[nums[k],nums[k+1],...,nums[n-1],nums[0],nums[1],...,nums[k-1]](下标从0开始计数)。例如,[0,1,2,4,5,6,7]在下标3......
  • 在JS中哪些数组原生方法会改变原数组?
    在JavaScript中,一些数组的原生方法会改变原数组,而另一些则不会。以下是一些会改变原数组的常用原生方法:push():向数组的末尾添加一个或多个元素,并返回新的长度。pop():删除并返回数组的最后一个元素。shift():删除并返回数组的第一个元素。unshift():向数组的开头添加一个或多个......
  • 使用JS写一个方法计算嵌套数组的深度
    你可以使用递归函数来计算嵌套数组的深度。以下是一个使用JavaScript编写的示例方法,用于计算嵌套数组的深度:functiongetArrayDepth(arr){if(!Array.isArray(arr)){return0;//如果不是数组,返回深度0}letmaxDepth=0;for(leti=0;i<arr.length;i......
  • 写一个方法对数组对象的某几个key进行排序
    在前端开发中,JavaScript是一种常用的语言,我们可以使用其数组的sort()方法来对数组对象的特定key进行排序。以下是一个简单的示例,假设我们有一个对象数组,并且我们想要根据对象的agekey对其进行排序:functionsortByKey(array,key){returnarray.sort((a,b)=>(a[k......
  • 使用js写一个方法判断数组是否为等差数组
    等差数组是指数组中任意两个相邻元素的差值都相等的数组。下面是一个使用JavaScript编写的函数,该函数可以判断一个数组是否为等差数组:functionisArithmeticArray(arr){if(arr.length<2){//如果数组长度小于2,那么它不能被视为等差数组returnfalse;......
  • C++中的字符( char )、字符数组( char[] )、字符串( std::string )
    字符(char)定义:char是C++中的基本数据类型,用于表示单个字符。char在内存中通常占用一个字节(8位)。在ASCII编码系统中,每个字符都对应一个唯一的整数值,char类型可以存储这些值来表示相应的字符。charch='A';//存储字符'A'与其他类型的联系:字符本质上是一个小整数类......