每日算法 -【Swift 算法】查找字符串数组中的最长公共前缀

Swift 实现:查找字符串数组中的最长公共前缀

在面试或实际开发中,我们常常会遇到需要在一组字符串中找出它们的最长公共前缀的问题。

例如,输入:

swift 复制代码
["flower", "flow", "flight"]

输出应为:

swift 复制代码
"fl"

如果不存在公共前缀,则应返回空字符串 ""


🧠 解题思路

最长公共前缀的本质是找出所有字符串从左开始一致的部分。思路上有三种主流方式:

  1. 逐步比较法(横向扫描)
  2. 逐字符比较法(纵向扫描)
  3. 排序后只比较首尾字符串

1️⃣ 方法一:逐步比较法(横向扫描)

以第一个字符串为初始前缀,逐个与后面的字符串比较并修剪前缀,直到不再能修剪为止。

swift 复制代码
/// 方法一:横向扫描
/// - Parameter strs: 字符串数组
/// - Returns: 最长公共前缀
func longestCommonPrefix_Horizontal(_ strs: [String]) -> String {
    // 如果数组为空,直接返回空字符串
    guard let first = strs.first else { return "" }

    // 初始前缀设为第一个字符串
    var prefix = first

    // 遍历剩下的字符串
    for str in strs.dropFirst() {
        // 如果当前字符串不以 prefix 开头,就逐步去掉 prefix 的最后一个字符
        while !str.hasPrefix(prefix) {
            prefix = String(prefix.dropLast())
            // 如果前缀被减到空字符串,说明没有公共前缀
            if prefix.isEmpty {
                return ""
            }
        }
    }
    return prefix
}

🔍 适合场景:数据量较小,或字符串之间前缀相差较大时。


2️⃣ 方法二:逐字符比较法(纵向扫描)

逐列检查每个字符串相同位置的字符,一旦发现不同或越界,即可确定前缀。

swift 复制代码
/// 方法二:纵向扫描
/// - Parameter strs: 字符串数组
/// - Returns: 最长公共前缀
func longestCommonPrefix_Vertical(_ strs: [String]) -> String {
    // 如果数组为空,直接返回空字符串
    guard let first = strs.first else { return "" }

    // 遍历第一个字符串的每个字符(按下标索引)
    for i in first.indices {
        let char = first[i]
        // 遍历其他所有字符串
        for str in strs {
            // 如果当前字符串已越界或字符不一致,返回当前索引之前的前缀
            if i >= str.endIndex || str[i] != char {
                return String(first[..<i])
            }
        }
    }
    return first
}

🔍 适合场景:前缀较短时效率较高,因为可以提早中止。


3️⃣ 方法三:排序法

对数组排序,最长公共前缀一定存在于排序后的第一个和最后一个字符串之间。

swift 复制代码
/// 方法三:排序法
/// - Parameter strs: 字符串数组
/// - Returns: 最长公共前缀
func longestCommonPrefix_Sorting(_ strs: [String]) -> String {
    // 空数组直接返回
    guard !strs.isEmpty else { return "" }

    // 字符串排序后,前缀只需比较首尾两个字符串
    let sortedStrs = strs.sorted()
    let first = sortedStrs.first!
    let last = sortedStrs.last!

    var i = first.startIndex
    // 从头开始逐个字符比较
    while i < first.endIndex && first[i] == last[i] {
        i = first.index(after: i)
    }

    // 返回首尾相同的前缀部分
    return String(first[..<i])
}

🔍 适合场景:当字符串数量不多、排序成本可接受时非常简洁有效。


📈 时间复杂度对比

方法 时间复杂度 空间复杂度 特点
横向扫描法 O(N * M) O(1) 简单直观,易于理解
纵向扫描法 O(N * M) O(1) 提前结束判断更高效
排序法 O(N log N + M) O(log N) 代码最简洁,依赖排序开销

其中:

  • N 为字符串数组的长度
  • M 为最短字符串的长度

✅ 总结

策略名称 适合场景 优点 缺点
横向扫描 字符串之间差异较大 实现简单,逻辑清晰 前缀缩减较慢,效率一般
纵向扫描 公共前缀较短或数据量大时 可提早中止,更高效 代码略复杂
排序法 字符串数量不多且排序代价可接受 实现简洁,只比较首尾两个 有排序开销

💬 如果你喜欢本文

可以点赞、收藏或分享给朋友,我们一起写更好的 Swift 代码!

相关推荐
情缘晓梦.2 分钟前
C语言数据存储
c语言·开发语言
xunyan62343 分钟前
第九章 JAVA常用类
java·开发语言
Aaron15883 分钟前
基于VU13P在人工智能高速接口传输上的应用浅析
人工智能·算法·fpga开发·硬件架构·信息与通信·信号处理·基带工程
予枫的编程笔记5 分钟前
【论文解读】DLF:以语言为核心的多模态情感分析新范式 (AAAI 2025)
人工智能·python·算法·机器学习
IOT-Power11 分钟前
QT 对话框(QDialog)中 accept、reject、exec、open的使用
开发语言·qt
froginwe1113 分钟前
ASP Session
开发语言
im_AMBER13 分钟前
Leetcode 99 删除排序链表中的重复元素 | 合并两个链表
数据结构·笔记·学习·算法·leetcode·链表
lbb 小魔仙21 分钟前
【Python】零基础学 Python 爬虫:从原理到反爬,构建企业级爬虫系统
开发语言·爬虫·python
Swift社区22 分钟前
ArkTS Web 组件里,如何通过 javaScriptProxy 让 JS 同步调用原生方法
开发语言·前端·javascript
Q741_14723 分钟前
海致星图招聘 数据库内核研发实习生 一轮笔试 总结复盘(1) 作答语言:C/C++ 链表 二叉树
开发语言·c++·经验分享·面试·笔试