本题 题目链接

题目描述


我的题解

方法三双100%, 方法一 适合范围广

方法一:双指针(也叫 滑动窗口)

思路分析

  • 用两个指针i和表示当前枚举到的以i为起点,j为终点的区间,sum表示[i,j]的区间和:

    • 当sum < tiarget,j指针向前移动,扩大区间,增大区间和。即:j++,sum+=j ;
    • 当sum > target,i指针向前移动,收缩区间,减小区间和。即:sum-=i,i++;
    • 当sum == target,i指针向前移动2个单位,j向前移动一个单位。即sum = sum-i-(i+1),i-=2,j++,sum+=j;

      (当sum=target时,无论是 i 先向前移动,还是 j 先向前移动,另一个指针下一步都会向前一步,此时,一定有sum>target(剔除一个小的加进来一个大的数)。

      故而可以直接把 i++和 j++合成一步。又因sum>target,i 又会向前一步。故最终可以再把这两步合成一步,直接令sum=target时,i 向前移动两个单位而j向前移动1个单位)
  • (对于求区间和sum也可以用数学公式求啦:(连续序列和=(首项+末项)*项数/2))

代码如下

    public int[][] findContinuousSequence(int target) {
ArrayList<int[]> resLists = new ArrayList<>();
int sum = 1;
for (int i = 1,j = 1; j <= (target >>1) + 1; ) {
if (sum < target) {
j++;
sum += j;
} else if (sum > target) {
sum -= i;
i++;
} else {
resLists.add(getArray(i,j));
j++;
sum = sum + j - i - (i + 1);
i = i+2;
}
}
return resLists.toArray(new int[0][]);
} private int[] getArray(int i, int j) {
if (i == j) return new int[0]; // 题目要求至少含2个数
int[] arr = new int[(j - i + 1)];
for (int k = 0; k <= j - i; k++) arr[k] = k + i; return arr;
}

方法二:求根公式

(官方的题解,我就不写了,直接截图附上C++代码)

代码如下(C++)

class Solution {
public:
vector<vector<int>> findContinuousSequence(int target) {
vector<vector<int>> vec;
vector<int> res;
int sum = 0, limit = (target - 1) / 2; // (target - 1) / 2 等效于 target / 2 下取整
for (int x = 1; x <= limit; ++x) {
long long delta = 1 - 4 * (x - 1ll * x * x - 2 * target);
if (delta < 0) continue;
int delta_sqrt = (int)sqrt(delta + 0.5);
if (1ll * delta_sqrt * delta_sqrt == delta && (delta_sqrt - 1) % 2 == 0){
int y = (-1 + delta_sqrt) / 2; // 另一个解(-1-delta_sqrt)/2必然小于0,不用考虑
if (x < y) {
res.clear();
for (int i = x; i <= y; ++i) res.emplace_back(i);
vec.emplace_back(res);
}
}
}
return vec;
}
};

方法三:等差数列特殊性质,奇偶讨论,双100

(参考了一位大佬的)

思路分析

  • 实际上某个序列可以按其包含奇数/偶数个元素来讨论

    • 当某序列含有奇数个(2k+1)元素时,Sequence : i, i+1, i+2 ... i+2k

      其序列和为中间元素的2k+1倍,即:Sum(Sequence) = (2k+1) * (i+k)
    • 当某序列含有偶数个(2k)元素时, Sequence : i, i+1, i+2 ... i+2k-1

      其序列和为中间两个元素的k倍,即:Sum(Sequence) = k * [(i+k-1) + (i+k)]
  • 因此枚举所有可能的序列长度len(从2开始,题目要求至少2个连续的数):

    • 奇数时直接判断长度len是否整除target,整除则符合题意,找到序列
    • 偶数时判断 2k*(mid1+mid2) = target,k为正整数,mid1和mid2为序列中间的两个数,因连续,故和为大于1的奇数。

      成立,则找到序列
  • 由于枚举按照序列长度递增顺序,因此输出时将结果逆序输出

  • 对于枚举长度,len上界:

代码如下


public int[][] findContinuousSequence(int target) {
ArrayList<int[]> resLists = new ArrayList<>(); int len = 2;
while (len * (1 + len) < (target << 1)) {
if (len % 2 == 1) { // 长度为奇数
if (target % len == 0) { // 找到序列(中间数为 target/len)
int[] arr = new int[len];
for (int i = 0, val = target / len - len / 2; i < len; i++) // 存数组
arr[i] = val++;
resLists.add(arr);
}
} else { // 长度为偶数
int k = len / 2;
// 符合式子: 2k*(mid1+mid2) = target,k为正整数,mid1和mid2为连续的两个数,故和为大于1的奇数
if (target % k == 0 && target / k % 2 == 1) { // 找到序列(中间两个数的和为 target/k)
int[] arr = new int[len];
for (int i = 0, val = target / k / 2 - len / 2 + 1; i < len; i++)
arr[i] = val++;
resLists.add(arr);
}
} len++;
}
Collections.reverse(resLists);
return resLists.toArray(new int[0][]);
}

剑指 Offer 57 - II. 和为s的连续正数序列的更多相关文章

  1. 剑指 Offer 57 - II. 和为s的连续正数序列 + 双指针 + 数论

    剑指 Offer 57 - II. 和为s的连续正数序列 Offer_57_2 题目描述 方法一:暴力枚举 package com.walegarrett.offer; /** * @Author W ...

  2. 力扣 - 剑指 Offer 57 - II. 和为s的连续正数序列

    题目 剑指 Offer 57 - II. 和为s的连续正数序列 思路1(双指针/滑动窗口) 所谓滑动窗口,就是需要我们从一个序列中找到某些连续的子序列,我们可以使用两个for循环来遍历查找,但是未免效 ...

  3. 剑指offer四十一之和为S的连续正数序列

    一.题目   题目描述:小明很喜欢数学,有一天他在做数学作业时,要求计算出9~16的和,他马上就写出了正确答案是100.但是他并不满足于此,他在想究竟有多少种连续的正数序列的和为100(至少包括两个数 ...

  4. LeetCode 题解 | 面试题57 - II. 和为s的连续正数序列

    题目描述 面试题57 - II. 和为s的连续正数序列 难度简单37收藏分享切换为英文关注反馈 输入一个正整数 target ,输出所有和为 target 的连续正整数序列(至少含有两个数). 序列内 ...

  5. 《剑指offer》面试题57 - II. 和为s的连续正数序列

    问题描述 输入一个正整数 target ,输出所有和为 target 的连续正整数序列(至少含有两个数). 序列内的数字由小到大排列,不同序列按照首个数字从小到大排列. 示例 1: 输入:target ...

  6. 剑指offer41:所有和为S的连续正数序列,例如,有多少种连续的正数序列的和为100

    1 题目描述 小明很喜欢数学,有一天他在做数学作业时,要求计算出9~16的和,他马上就写出了正确答案是100.但是他并不满足于此,他在想究竟有多少种连续的正数序列的和为100(至少包括两个数).没多久 ...

  7. [简单-剑指 Offer 53 - II. 0~n-1中缺失的数字]

    [简单-剑指 Offer 53 - II. 0-n-1中缺失的数字] 一个长度为n-1的递增排序数组中的所有数字都是唯一的,并且每个数字都在范围0-n-1之内.在范围0-n-1内的n个数字中有且只有一 ...

  8. 剑指 Offer 68 - II. 二叉树的最近公共祖先 + 最近公共祖先(LCA)

    剑指 Offer 68 - II. 二叉树的最近公共祖先 Offer_68_2 题目详情 题解分析 java代码 package com.walegarrett.offer; /** * @Autho ...

  9. 剑指 Offer 58 - II. 左旋转字符串 + 简单题

    剑指 Offer 58 - II. 左旋转字符串 Offer_58_2 题目描述 java代码 package com.walegarrett.offer; /** * @Author WaleGar ...

随机推荐

  1. VMware实现宿主机和虚拟机处于同一网段

    打开虚拟网络编辑器 选择VMnet0桥接模式,在VMnet信息中,选择可以选择的网卡,然后保存. 打开虚拟机设置,在“硬件”选项卡的网络适配器中选择桥接模式即可.

  2. (一)ansible 安装配置

    CentOS 7.5 一,安装 yum -y install ansible 二,配置hosts文件 /etc/ansible/hosts s1 ansible_ssh_port= ansible_s ...

  3. Blazor带我重玩前端(三)

    写在前面 需要升级VS2019以及.NET Core到最新版(具体的最低支持,我已经忘了,总是越新支持的就越好),以更好的支持自己开发Blazor项目. WebAssembly 搜索Blazor模板 ...

  4. 如何在同一台电脑上部署多个tomcat实现多个tomcat在同一台电脑上同时启动

    有时候我们在开发的过程中难免会遇到需要在同一台电脑部署多个tomcat,且还要他们能够都单独同时启动不会对其他的tomcat造成影响 本文就简单记录一下,如何来实现这个骚操作 1. 下载tomcat的 ...

  5. Scala 面向对象(三):package 包 (二)

    1 包对象 基本介绍:包可以包含类.对象和特质trait,但不能包含函数/方法或变量的定义.这是Java虚拟机的局限.为了弥补这一点不足,scala提供了包对象的概念来解决这个问题. package ...

  6. 数据可视化之powerBI基础(十七)掌握PowerBI按列排序,再也不用担心顺序错乱了

    https://zhuanlan.zhihu.com/p/64421933 经常有朋友问,当把文本字段放到坐标轴或者切片器上,显示的顺序都乱了,完全不是自己想要的,就像下面这些, ↑月份坐标轴 ↑星期 ...

  7. 机器学习实战基础(二十三):sklearn中的降维算法PCA和SVD(四) PCA与SVD 之 PCA中的SVD

    PCA中的SVD 1 PCA中的SVD哪里来? 细心的小伙伴可能注意到了,svd_solver是奇异值分解器的意思,为什么PCA算法下面会有有关奇异值分解的参数?不是两种算法么?我们之前曾经提到过,P ...

  8. Spring cloud系列教程第十篇- Spring cloud整合Eureka总结篇

    Spring cloud系列教程第十篇- Spring cloud整合Eureka总结篇 本文主要内容: 1:spring cloud整合Eureka总结 本文是由凯哥(凯哥Java:kagejava ...

  9. 重学c#系列——c# 托管和非托管资源(三)

    前言 c# 托管和非托管比较重要,因为这涉及到资源的释放. 现在只要在计算机上运行的,无论玩出什么花来,整个什么概念,逃不过输入数据修改数据输出数据(计算机本质),这里面有个数据的输入,那么我们的内存 ...

  10. MySQL 三万字精华总结 + 面试100 问,吊打面试官绰绰有余(收藏系列)

    写在之前:不建议那种上来就是各种面试题罗列,然后背书式的去记忆,对技术的提升帮助很小,对正经面试也没什么帮助,有点东西的面试官深挖下就懵逼了. 个人建议把面试题看作是费曼学习法中的回顾.简化的环节,准 ...