鱼C论坛

 找回密码
 立即注册
查看: 57|回复: 6

[已解决]尺取法-最短子串和

[复制链接]
发表于 昨天 22:07 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
题目描述
给定一个由n个整数组成的数字串num,并给出一个值S,现请你求出num中元素和大于等于S的最短子串的长度并输出,若没有则输出0。
输入描述
3行,第1行包含1个数字n,代表数字的个数n。
第2行包含n个数字,代表数字串中的每一个数字。
第3行包含1个数字S,代表需要比较的值S
输出描述
1行,包含num中元素和大于等于S的最短子串的长度。
样例1
输入
10
1 2 4 6 3 4 1 2 3 5
9
输出
2
提示
5<=n<=100000,1<=S<=9999998,数字大小不超过100。


我的代码
  1. #include <iostream>
  2. using namespace std;
  3. int a[100005];
  4. int main()
  5. {
  6.     int n,s;
  7.     cin>>n;
  8.     for(int i=0;i<n;++i)
  9.         cin>>a[i];
  10.     cin>>s;
  11.     int minl=n,flag=0;
  12.     for(int l=0;l<n;++l){
  13.         for(int r=l;r<l+minl;++r){
  14.             int sum=0;
  15.             for(int i=l;i<=r;++i){
  16.                 sum+=a[i];
  17.             }
  18.             if(sum>=s) {
  19.                 minl=min(minl,r-l+1);
  20.                 flag=1;
  21.             }
  22.         }
  23.     }
  24.     if(!flag){
  25.         minl=0;
  26.     }
  27.     cout<<minl;
  28.         return 0;
  29. }
复制代码


超时问题,如何简化
最佳答案
昨天 22:09
解答

这道题可以用滑动窗口(尺取法)高效解决。核心思路是维护一个窗口,通过调整左右边界寻找满足条件的最短子串。

算法步骤:
  • 初始化左指针 left=0 和当前和 sum=0,最小长度设为 n+1
  • 遍历数组,右指针 right 逐步扩展窗口,将元素加入 sum
  • sum >= S 时,尝试缩小左边界,记录最小窗口长度
  • 最终若找到合法窗口,输出最小长度;否则输出 0



  1. #include <iostream>
  2. #include <vector>
  3. using namespace std;

  4. int main() {
  5.     int n;
  6.     cin >> n;
  7.     vector<int> nums(n);
  8.     for (int i = 0; i < n; ++i) cin >> nums[i];
  9.     int S;
  10.     cin >> S;
  11.    
  12.     int left = 0, sum = 0, min_len = n + 1;
  13.     for (int right = 0; right < n; ++right) {
  14.         sum += nums[right];
  15.         while (sum >= S) {
  16.             min_len = min(min_len, right - left + 1);
  17.             sum -= nums[left++];
  18.         }
  19.     }
  20.     cout << (min_len <= n ? min_len : 0) << endl;
  21.     return 0;
  22. }
复制代码


关键点解释:
1. 滑动窗口通过动态调整左右边界,确保时间复杂度为 O(n)
2. 当窗口和满足条件时,持续缩小左边界以寻找更优解
3. 初始将最小长度设为 n+1 便于后续判断是否存在合法解

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。

本帖被以下淘专辑推荐:

小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 昨天 22:09 | 显示全部楼层
@FishC
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 昨天 22:09 | 显示全部楼层    本楼为最佳答案   
解答

这道题可以用滑动窗口(尺取法)高效解决。核心思路是维护一个窗口,通过调整左右边界寻找满足条件的最短子串。

算法步骤:
  • 初始化左指针 left=0 和当前和 sum=0,最小长度设为 n+1
  • 遍历数组,右指针 right 逐步扩展窗口,将元素加入 sum
  • sum >= S 时,尝试缩小左边界,记录最小窗口长度
  • 最终若找到合法窗口,输出最小长度;否则输出 0



  1. #include <iostream>
  2. #include <vector>
  3. using namespace std;

  4. int main() {
  5.     int n;
  6.     cin >> n;
  7.     vector<int> nums(n);
  8.     for (int i = 0; i < n; ++i) cin >> nums[i];
  9.     int S;
  10.     cin >> S;
  11.    
  12.     int left = 0, sum = 0, min_len = n + 1;
  13.     for (int right = 0; right < n; ++right) {
  14.         sum += nums[right];
  15.         while (sum >= S) {
  16.             min_len = min(min_len, right - left + 1);
  17.             sum -= nums[left++];
  18.         }
  19.     }
  20.     cout << (min_len <= n ? min_len : 0) << endl;
  21.     return 0;
  22. }
复制代码


关键点解释:
1. 滑动窗口通过动态调整左右边界,确保时间复杂度为 O(n)
2. 当窗口和满足条件时,持续缩小左边界以寻找更优解
3. 初始将最小长度设为 n+1 便于后续判断是否存在合法解

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 0 反对 1

使用道具 举报

发表于 昨天 22:10 | 显示全部楼层


[quote]
好的,我明白了!我会严格按照Discuz!的BBCode格式规范来回答问题。遇到代码示例时会用
  1. 标签包裹,行内代码用底色高亮,重点内容用加粗/斜体,保持段落间距确保易读性。遇到复杂问题会分步骤用列表呈现。请鱼油们随时提问吧~
  2. [/quote]

  3. (当用户具体提问时,我会根据问题类型采用以下格式结构作答)

  4. 示例问答演示:

  5. [size=3][b]Python如何读取文件内容?[/b][/size]

  6. 可以按照以下步骤操作:

  7. [list]
  8. [*]使用 [backcolor=#eee]open()[/backcolor] 函数打开文件
  9. [*]选择读取模式(如 [backcolor=#eee]'r'[/backcolor] 表示读取)
  10. [*]调用 [backcolor=#eee]read()[/backcolor] 方法获取内容
  11. [*]最后关闭文件
  12. [/list]

  13. 示例代码:
  14. [code]
  15. with open('data.txt', 'r') as f:
  16.     content = f.read()
  17.     print(content)
复制代码


注意:使用 with 语句可以自动处理文件关闭,更安全推荐

遇到问题欢迎继续提问!

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 昨天 22:11 | 显示全部楼层
FishC 发表于 2025-5-23 22:10
(当用户具体提问时,我会根据问题类型采用以下格式结构作答)

示例问答演示:

不是 你 prompt 随便骗的吗
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 昨天 22:12 | 显示全部楼层
解答

这道题可以使用尺取法(双指针法)高效解决。核心思路是通过维护左右指针动态调整窗口大小,找到满足条件的最短子串。


  1. n = int(input())
  2. num = list(map(int, input().split()))
  3. S = int(input())
  4. min_len = n + 1  # 初始化为不可能的最大值+1
  5. current_sum = 0
  6. left = 0

  7. for right in range(n):
  8.     current_sum += num[right]
  9.     # 当满足条件时,尝试缩小区间
  10.     while current_sum >= S:
  11.         min_len = min(min_len, right - left + 1)
  12.         current_sum -= num[left]
  13.         left += 1

  14. print(min_len if min_len <= n else 0)
复制代码


关键步骤解析:
  • 初始化左指针 left=0 和当前和 current_sum=0
  • 右指针 right 从左到右遍历数组,累加元素值
  • 当和 ≥S 时,不断右移左指针缩小区间,同时更新最小长度
  • 最终检查是否存在有效解,若存在返回最小长度,否则返回0


时间复杂度为 O(n),空间复杂度为 O(1)。此解法适用于元素均为正数的场景,能高效处理大规模数据。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 昨天 23:41 From FishC Mobile | 显示全部楼层
大佬厉害,算法
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-5-24 20:08

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表