news 2026/5/26 2:25:58

算法题 字符的最短距离

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
算法题 字符的最短距离

字符的最短距离

问题描述

给你一个字符串s和一个字符c,其中cs中至少出现一次。

返回一个整数数组answer,其中answer[i]是字符串s中下标i处的字符到最近的字符c的最短距离。

两个下标ij之间的距离为abs(i - j)

示例

输入: s = "loveleetcode", c = "e" 输出: [3,2,1,0,1,0,0,1,2,2,1,0] 解释: 字符 'e' 在下标 3、5、6、11 处出现。 对于下标 0,最近的 'e' 在下标 3,距离为 |0-3| = 3。 对于下标 1,最近的 'e' 在下标 3,距离为 |1-3| = 2。 ...

算法思路

核心

  1. 最近距离:对于任意位置 i,最近的字符 c 要么在 i 的左边,要么在 i 的右边
  2. 两次遍历
    • 第一次从左到右:记录每个位置到左边最近 c 的距离
    • 第二次从右到左:记录每个位置到右边最近 c 的距离
    • 取两次遍历结果的最小值

方法

  • 两次遍历:时间复杂度 O(n),空间复杂度 O(1)
  • 预处理位置:先找到所有 c 的位置,然后对每个位置二分查找最近的 c
  • 暴力:对每个位置遍历整个字符串找最近的 c(效率低)

代码实现

方法一:两次遍历

classSolution{/** * 使用两次遍历计算每个字符到最近目标字符的最短距离 * * @param s 输入字符串 * @param c 目标字符 * @return 每个位置到最近c的最短距离数组 */publicint[]shortestToChar(Strings,charc){intn=s.length();int[]result=newint[n];// 初始化为一个很大的值(大于可能的最大距离)// 最大距离不会超过n,所以用n作为初始值for(inti=0;i<n;i++){result[i]=n;}// 第一次遍历:从左到右,计算到左边最近c的距离intprev=-n;// 记录上一个c的位置,初始化为-n确保第一次更新for(inti=0;i<n;i++){if(s.charAt(i)==c){prev=i;}result[i]=Math.min(result[i],i-prev);}// 第二次遍历:从右到左,计算到右边最近c的距离prev=2*n;// 记录下一个c的位置,初始化为2*n确保第一次更新for(inti=n-1;i>=0;i--){if(s.charAt(i)==c){prev=i;}result[i]=Math.min(result[i],prev-i);}returnresult;}}

方法二:预处理位置 + 二分查找

importjava.util.*;classSolution{/** * 先预处理所有目标字符的位置,然后对每个位置二分查找最近的位置 */publicint[]shortestToChar(Strings,charc){// 预处理:找到所有c的位置List<Integer>positions=newArrayList<>();for(inti=0;i<s.length();i++){if(s.charAt(i)==c){positions.add(i);}}int[]result=newint[s.length()];// 对每个位置,找到最近的cfor(inti=0;i<s.length();i++){result[i]=findMinDistance(positions,i);}returnresult;}/** * 在有序位置列表中找到距离target最近的位置 */privateintfindMinDistance(List<Integer>positions,inttarget){// 二分查找插入位置intleft=0,right=positions.size();while(left<right){intmid=left+(right-left)/2;if(positions.get(mid)<target){left=mid+1;}else{right=mid;}}intminDist=Integer.MAX_VALUE;// 检查left位置(第一个>=target的位置)if(left<positions.size()){minDist=Math.min(minDist,positions.get(left)-target);}// 检查left-1位置(最后一个<target的位置)if(left>0){minDist=Math.min(minDist,target-positions.get(left-1));}returnminDist;}}

方法三:暴力

classSolution{/** * 暴力:对每个位置遍历整个字符串 */publicint[]shortestToChar(Strings,charc){intn=s.length();int[]result=newint[n];for(inti=0;i<n;i++){intminDist=n;// 最大可能距离for(intj=0;j<n;j++){if(s.charAt(j)==c){minDist=Math.min(minDist,Math.abs(i-j));}}result[i]=minDist;}returnresult;}}

算法分析

  • 时间复杂度

    • 两次遍历:O(n) - 三次线性扫描(初始化+两次遍历)
    • 预处理+二分查找:O(n log k) - k是字符c的出现次数
    • 暴力:O(n²)
  • 空间复杂度

    • 两次遍历:O(1) - 只使用常数额外空间
    • 预处理+二分查找:O(k) - 存储c的位置
    • 暴力:O(1)

算法过程

1:s = “loveleetcode”, c = “e”

字符e的位置:[3, 5, 6, 11]

两次遍历

第一次遍历(左到右)

  • i=0: prev=-12, dist=0-(-12)=12
  • i=1: prev=-12, dist=13
  • i=2: prev=-12, dist=14
  • i=3: s[3]==‘e’, prev=3, dist=0
  • i=4: prev=3, dist=1
  • i=5: s[5]==‘e’, prev=5, dist=0
  • i=6: s[6]==‘e’, prev=6, dist=0
  • i=7: prev=6, dist=1
  • …继续到i=11

中间结果:[12,13,14,0,1,0,0,1,2,3,4,0]

第二次遍历(右到左)

  • i=11: s[11]==‘e’, prev=11, dist=min(0, 0)=0
  • i=10: prev=11, dist=min(4, 1)=1
  • i=9: prev=11, dist=min(3, 2)=2
  • i=8: prev=11, dist=min(2, 3)=2
  • i=7: prev=11, dist=min(1, 4)=1
  • i=6: s[6]==‘e’, prev=6, dist=min(0, 0)=0
  • i=5: s[5]==‘e’, prev=5, dist=min(0, 0)=0
  • i=4: prev=5, dist=min(1, 1)=1
  • i=3: s[3]==‘e’, prev=3, dist=min(0, 0)=0
  • i=2: prev=3, dist=min(14, 1)=1
  • i=1: prev=3, dist=min(13, 2)=2
  • i=0: prev=3, dist=min(12, 3)=3

最终结果:[3,2,1,0,1,0,0,1,2,2,1,0]

测试用例

publicstaticvoidmain(String[]args){Solutionsolution=newSolution();// 测试用例1:标准示例int[]result1=solution.shortestToChar("loveleetcode",'e');System.out.println("Test 1: "+Arrays.toString(result1));// [3,2,1,0,1,0,0,1,2,2,1,0]// 测试用例2:目标字符在开头int[]result2=solution.shortestToChar("aaab",'b');System.out.println("Test 2: "+Arrays.toString(result2));// [3,2,1,0]// 测试用例3:目标字符在结尾int[]result3=solution.shortestToChar("baaa",'b');System.out.println("Test 3: "+Arrays.toString(result3));// [0,1,2,3]// 测试用例4:所有字符都是目标字符int[]result4=solution.shortestToChar("eeee",'e');System.out.println("Test 4: "+Arrays.toString(result4));// [0,0,0,0]// 测试用例5:只有一个目标字符int[]result5=solution.shortestToChar("abcdefg",'d');System.out.println("Test 5: "+Arrays.toString(result5));// [3,2,1,0,1,2,3]// 测试用例6:单字符字符串int[]result6=solution.shortestToChar("a",'a');System.out.println("Test 6: "+Arrays.toString(result6));// [0]// 测试用例7:两个字符int[]result7=solution.shortestToChar("ab",'b');System.out.println("Test 7: "+Arrays.toString(result7));// [1,0]// 测试用例8:目标字符多次出现int[]result8=solution.shortestToChar("abccba",'c');System.out.println("Test 8: "+Arrays.toString(result8));// [2,1,0,0,1,2]// 测试用例9:长字符串int[]result9=solution.shortestToChar("abcdefghijklmnopqrstuvwxyz",'m');System.out.println("Test 9: "+Arrays.toString(result9));// [12,11,10,9,8,7,6,5,4,3,2,1,0,1,2,3,4,5,6,7,8,9,10,11,12,13]// 测试用例10:目标字符在中间int[]result10=solution.shortestToChar("abcde",'c');System.out.println("Test 10: "+Arrays.toString(result10));// [2,1,0,1,2]}

关键点

  1. 两次遍历

    • 单次遍历无法同时考虑左右两边的最近字符
    • 两次遍历分别处理左边界和右边界情况
  2. 初始化

    • 使用足够大的初始值确保正确更新
    • 避免使用Integer.MAX_VALUE防止溢出
  3. 距离计算

    • 左到右:i - prev(prev ≤ i)
    • 右到左:prev - i(prev ≥ i)
    • 保证距离为正数
  4. 边界处理

    • 字符串开头:只有右边的字符可选
    • 字符串结尾:只有左边的字符可选

常见问题

  1. 为什么不能用一次遍历?
    • 一次遍历只能知道已遍历部分的信息
    • 无法预知右边是否有更近的字符
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/26 3:25:11

系统学习USB Burning Tool的PC端工作模式切换

深入掌握USB Burning Tool的PC端模式切换&#xff1a;从“变砖”恢复到自动化烧录实战你有没有遇到过这样的场景&#xff1f;一台基于Amlogic芯片的设备&#xff0c;刷机后彻底无法启动——电源灯亮着&#xff0c;但屏幕黑屏、ADB连不上、串口无输出。你以为它“变砖”了&#…

作者头像 李华
网站建设 2026/5/26 7:14:03

WE Learn平台高效学习解决方案:5大核心功能深度解析

WE Learn平台高效学习解决方案&#xff1a;5大核心功能深度解析 【免费下载链接】WELearnHelper 显示WE Learn随行课堂题目答案&#xff1b;支持班级测试&#xff1b;自动答题&#xff1b;刷时长&#xff1b;基于生成式AI(ChatGPT)的答案生成 项目地址: https://gitcode.com/…

作者头像 李华
网站建设 2026/5/26 6:11:58

RePKG工具全面解析:轻松解锁Wallpaper Engine壁纸资源

RePKG作为一款专业的Wallpaper Engine资源处理工具&#xff0c;能够高效解包PKG格式文件并自动转换TEX纹理为常见图片格式。无论你是壁纸创作者还是技术爱好者&#xff0c;这款工具都能帮你深度挖掘壁纸资源的价值。 【免费下载链接】repkg Wallpaper engine PKG extractor/TEX…

作者头像 李华
网站建设 2026/5/25 8:34:18

RePKG终极指南:5步精通Wallpaper Engine资源管理与自定义

RePKG终极指南&#xff1a;5步精通Wallpaper Engine资源管理与自定义 【免费下载链接】repkg Wallpaper engine PKG extractor/TEX to image converter 项目地址: https://gitcode.com/gh_mirrors/re/repkg 想要深度定制你的Wallpaper Engine动态壁纸&#xff1f;RePKG作…

作者头像 李华
网站建设 2026/5/25 15:25:38

Blender3MF插件使用指南:5步掌握3D打印文件处理

Blender3MF插件使用指南&#xff1a;5步掌握3D打印文件处理 【免费下载链接】Blender3mfFormat Blender add-on to import/export 3MF files 项目地址: https://gitcode.com/gh_mirrors/bl/Blender3mfFormat 想要在Blender中轻松处理3D打印文件&#xff1f;Blender3mfFo…

作者头像 李华
网站建设 2026/5/25 10:46:31

Bypass Paywalls Clean:3分钟突破付费墙的终极解决方案

Bypass Paywalls Clean&#xff1a;3分钟突破付费墙的终极解决方案 【免费下载链接】bypass-paywalls-chrome-clean 项目地址: https://gitcode.com/GitHub_Trending/by/bypass-paywalls-chrome-clean 你是否曾经在深夜想要查阅一篇重要的技术文档&#xff0c;却被付费…

作者头像 李华