题解 | #数组求和统计#

数组求和统计

http://www.nowcoder.com/practice/efc16ce46397436a8d1a0008c52093c1

题意整理:

整理题意既:给出两个大小为n的数组a, b,统计有多少数对(l, r)满足且a数组的[l,r]区间和等于b数组的的和

方法一:暴力枚举

核心思想:

可以对a进行二重枚举,既枚举l和r,然后计算对应值是否满足条件。第二维的枚举使用变量sum统计前面的数的和,使得对单个区间计算的复杂度为1

核心代码:

class Solution {
public:
    int countLR(vector<int>& a, vector<int>& b) {
        int n = a.size();
        int ans = 0;
        for(int i = 0; i < n; ++i) {//枚举l
            int sum = 0;//隐式前缀和
            for(int j = i; j < n; ++j) {//枚举j
                sum += a[j];
                if(sum == b[i] + b[j]) {//满足条件
                    ++ans;
                }
            }
        }
        return ans;
    }
};

复杂度分析:

时间复杂度:。使用了二重枚举,复杂度为
空间复杂度:,只使用了常数个变量

方法二:

核心思想:

方法一采用了隐式的前缀和,既sum变量,也可以通过显式的前缀和进行计算。前缀和数组不需要创建,可以通过直接对a进行修改完成。
前缀和
图片说明

核心代码:

class Solution {
public:
    int countLR(vector<int>& a, vector<int>& b) {
        int n = a.size();
        for(int i = 1; i < n; ++i) {
            a[i] += a[i - 1];//将其改为前缀和数组
        }
        int ans = 0;
        for(int i = 0; i < n; ++i) {
            for(int j = i; j < n; ++j) {
                //计算当前和
                int p = i == 0 ? a[j] : a[j] - a[i - 1];//对0特判
                if(p == b[i] + b[j]) {
                    ++ans;
                }
            }
        }
        return ans;
    }
};

复杂度分析:

时间复杂度:。使用了二重枚举,复杂度为
空间复杂度:,只使用了常数个变量

方法三:

核心思想:

方法二使用了前缀和,可是因为二重枚举使得时间复杂度仍为。考虑能否对其优化
观察方法二代码,发现实际上是在计算满足 的(l,r)对数量
改写上述等式为,等式左侧和右侧都只有一个数字,所以可以进行一维的枚举实现。
更具体的说,在计算出a的前缀数组sum后,既计算 ,
可以通过一个map记录之前的每个完成统计。

核心代码:

class Solution {
public:
    int countLR(vector<int>& a, vector<int>& b) {
        int n = a.size();
        int ans = 0;
        vector<int> sum(n);//前缀数组
        sum[0] = a[0];
        //计算前缀数组
        for(int i = 1; i < n; ++i) {
            sum[i] = sum[i - 1] + a[i];
        }
        unordered_map<int, int> mp;//统计之前的sum[l] - a[l] + b[l]
        for(int i = 0; i < n; ++i) {
            int p = sum[i] - a[i] + b[i];
            mp[p]++;//进行l统计
            if(mp.find(sum[i] - b[i]) != mp.end()) {
                //进行r求和
                ans += mp[sum[i] - b[i]];
            }
        }
        return ans;
    }
};

复杂度分析:

时间复杂度:。前缀和数组的计算为。统计时只进行一维枚举,对unordered_map单次操作的时间复杂度为,故总的时间复杂度为
空间复杂度:,既前缀数组和哈希表的开销,两者均为

全部评论

相关推荐

迷茫的大四🐶:自信一点,我认为你可以拿到50k,低于50k完全配不上你的能力,兄弟,不要被他们骗了,你可以的
点赞 评论 收藏
分享
避坑恶心到我了大家好,今天我想跟大家聊聊我在成都千子成智能科技有限公司(以下简称千子成)的求职经历,希望能给大家一些参考。千子成的母公司是“同创主悦”,主要经营各种产品,比如菜刀、POS机、电话卡等等。听起来是不是有点像地推销售公司?没错,就是那种类型的公司。我当时刚毕业,急需一份临时工作,所以在BOSS上看到了千子成的招聘信息。他们承诺无责底薪5000元,还包住宿,这吸引了我。面试的时候,HR也说了同样的话,感觉挺靠谱的。于是,我满怀期待地等待结果。结果出来后,我通过了面试,第二天就收到了试岗通知。试岗的内容就是地推销售,公司划定一个区域,然后你就得见人就问,问店铺、问路人,一直问到他们有意向为止。如果他们有兴趣,你就得摇同事帮忙推动,促进成交。说说一天的工作安排吧。工作时间是从早上8:30到晚上18:30。早上7点有人叫你起床,收拾后去公司,然后唱歌跳舞(销售公司都这样),7:55早课(类似宣誓),8:05同事间联系销售话术,8:15分享销售技巧,8:30经理训话。9:20左右从公司下市场,公交、地铁、自行车自费。到了市场大概10点左右,开始地推工作。中午吃饭时间大约是12:00,公司附近的路边盖饭面馆店自费AA,吃饭时间大约40分钟左右。吃完饭后继续地推工作,没有所谓的固定中午午休时间。下午6点下班后返回公司,不能直接下班,需要与同事交流话术,经理讲话洗脑。正常情况下9点下班。整个上班的一天中,早上到公司就是站着的,到晚上下班前都是站着。每天步数2万步以上。公司员工没有自己的工位,百来号人挤在一个20平方米的空间里听经理洗脑。白天就在市场上奔波,公司的投入成本几乎只有租金和工资,没有中央空调。早上2小时,晚上加班2小时,纯蒸桑拿。没有任何福利,节假日也没有3倍工资之类的。偶尔会有冲的酸梅汤和西瓜什么的。公司的晋升路径也很有意思:新人—组长—领队—主管—副经理—经理。要求是业绩和团队人数,类似传销模式,把人留下来。新人不能加微信、不能吐槽公司、不能有负面情绪、不能谈恋爱、不能说累。在公司没有任何坐的地方,不能依墙而坐。早上吃早饭在公司外面的安全通道,未到上班时间还会让你吃快些不能磨蹭。总之就是想榨干你。复试的时候,带你的师傅会给你营造一个钱多事少离家近的工作氛围,吹嘘工资有多高、还能吹自己毕业于好大学。然后让你早点来公司、无偿加班、抓住你可能不会走的心思进一步压榨你。总之,大家在找工作的时候一定要擦亮眼睛,避免踩坑!———来自网友
qq乃乃好喝到咩噗茶:不要做没有专业门槛的工作
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务