最大字段求和

算法1:三重循环暴力求解法

#include <stdio.h>
int maximum_segment_sum(int array[],int n)
{
    int this_sum = 0;
    int max_sum = 0;
    int i,j,k;
    for(i = 0;i < n;i++)
    {//分别以第一个第二个元素开始
        for(j = i;j < n;j++)
        {//不停的在这个串上多加几个元素
            this_sum = 0;
            for(k = i;k <= j;k++)
            {//遍历求和
                this_sum += array[k];
            }
            if(this_sum > max_sum)
            {
                max_sum = this_sum;
            }
        }
    }
    return max_sum;
}
int main() {
    int array[7] = {1,-5,3,4,-2,-3,10};
    int result = 0;
    result = maximum_segment_sum(array,7);
    printf("%d",result);
    return 0;
}

算法2:二重循环求解

#include <stdio.h>
int maximum_segment_sum(int array[],int n)
{
    int this_sum = 0;
    int max_sum = 0;
    int i,j,k;
    for(i = 0;i < n;i++)
    {
        this_sum = 0;
        for(j = i;j < n;j++)
        {
            this_sum += array[j];

            if(this_sum > max_sum)
            {
                max_sum = this_sum;
            }
        }

    }
    return max_sum;
}
int main() {
    int array[7] = {1,-5,3,4,-2,-3,10};
    int result = 0;
    result = maximum_segment_sum(array,7);
    printf("%d",result);
    return 0;
}

算法3:分治法

最关键的两个问题是:

我们要维护区间的哪些信息呢?lsum rsum msum isum
我们如何合并这些信息呢?通过pushup函数

#include <stdio.h>

/*结构体,里面包含一个字段的最重要的信息*/
struct Status
{
    int lsum;//lsum为左字段的最大和
    int rsum;//rsum为右子段的最大和
    int msum;//msum为该段的最大和
    int isum;//isum为该段的和
};

/*两个数求最大函数*/
int fmax(int a,int b)
{
    if(a >= b)
    {
        return a;
    }
    else if(b > a)
    {
        return b;
    }

}

/*上升操作*/
struct Status push_up(struct Status l,struct Status r)
{   //其中 lr分别为左右子段
    int isum = l.isum + r.isum;
    int lsum = fmax(l.lsum,l.isum + r.lsum);
    int rsum = fmax(r.rsum,r.isum + l.rsum);
    int msum = fmax(fmax(l.msum,r.msum),l.rsum+r.lsum);//看是否需要跨越中间,如果跨越一定说明左子段的rsum+右子段的lsum是最大的
    return (struct Status){lsum,rsum,msum,isum};
}

/*获取左边最大右边最大的信息*/
struct Status getnum(int array[] ,int left,int right)
{
    int middle = (left + right)/2;
    
    if(left == right)
        return (struct Status){array[left],array[left],array[left],array[left]};

    struct Status lsub = getnum(array,left,middle);//分别获取左右子段的信息
    struct Status rsub = getnum(array,middle + 1,right);
    
    return push_up(lsub,rsub);//相当于分治算法中的合成步骤

}

/*求最大字段和函数*/
int max_array(int array[],int left,int right)
{
     return getnum(array,left,right).msum;
}

int main()
{
    int array[7] = {1,-5,3,4,-2,-3,10};
    int result = 0;
    result = max_array(array,0,6);
    printf("%d",result);
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Stream流对某个字段求和的方法是使用Java 8中的stream的summingInt或summingDouble方法来实现。这些方法可以将流中的元素映射为一个整数或双精度数,并返回它们的总和。 举个例子,假设我们有一个包含若干对象的List,每个对象都有一个数字字段,我们想要对这些数字字段进行求和。可以按照以下步骤进行操作: 1. 导入所需的类: import java.util.List; import java.util.stream.Collectors; 2. 假设我们有一个包含对象的List,其中每个对象都有一个数字字段: List<SomeObject> list = ... 3. 使用stream方法获取流,并使用mapToInt方法将每个对象的数字字段映射为整数流: int sum = list.stream() .mapToInt(SomeObject::getNumberField) .sum(); 其中,SomeObject是我们对象的类名,getNumberField是获取数字字段的方法名。 这样,我们就可以得到数字字段的总和。注意,如果数字字段是浮点数类型,可以使用mapToDouble和summingDouble方法来进行相同的操作。 综上所述,使用stream的summingInt或summingDouble方法可以对某个字段进行求和。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Java8 stream 中利用 groupingBy 进行多字段分组求和案例](https://download.csdn.net/download/weixin_38692928/12821045)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [Java8用Stream流一行代码实现数据分组统计,排序,最大值、最小值、平均值、总数、合计](https://blog.csdn.net/xiaoheihai666/article/details/128152182)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值