面试算法117:相似的字符串

题目

如果交换字符串X中的两个字符就能得到字符串Y,那么两个字符串X和Y相似。例如,字符串"tars"和"rats"相似(交换下标为0和2的两个字符)、字符串"rats"和"arts"相似(交换下标为0和1的字符),但字符串"star"和"tars"不相似。

输入一个字符串数组,根据字符串的相似性分组,请问能把输入数组分成几组?如果一个字符串至少和一组字符串中的一个相似,那么它就可以放到该组中。假设输入数组中的所有字符串的长度相同并且两两互为变位词。例如,输入数组为["tars","rats","arts","star"],可以分成两组,一组为{"tars","rats","arts"},另一组为{"star"}。

分析

把输入数组中的每个字符串看成图中的一个节点。如果两个字符串相似,那么它们对应的节点之间有一条边相连,也就属于同一个子图。例如,字符串["tars","rats","arts","star"]根据相似性分别属于两个子图

java 复制代码
public class Test {
    public static void main(String[] args) {
        String[] A = {"tars", "rats", "arts", "star"};
        int result = numSimilarGroups(A);
        System.out.println(result);
    }

    public static int numSimilarGroups(String[] A) {
        int[] fathers = new int[A.length];
        for (int i = 0; i < fathers.length; i++) {
            fathers[i] = i;
        }

        int groups = A.length;
        for (int i = 0; i < A.length; i++) {
            for (int j = i + 1; j < A.length; j++) {
                if (similar(A[i], A[j]) && union(fathers, i, j)) {
                    groups--;
                }
            }
        }

        return groups;
    }

    private static boolean similar(String str1, String str2) {
        int diffCount = 0;
        for (int i = 0; i < str1.length(); i++) {
            if (str1.charAt(i) != str2.charAt(i)) {
                diffCount++;
            }
        }

        return diffCount <= 2;
    }

    private static boolean union(int[] fathers, int i, int j) {
        int fatherOfI = findFather(fathers, i);
        int fatherOfJ = findFather(fathers, j);
        if (fatherOfI != fatherOfJ) {
            fathers[fatherOfI] = fatherOfJ;
            return true;
        }

        return false;
    }

    private static int findFather(int[] fathers, int i) {
        if (fathers[i] != i) {
            fathers[i] = findFather(fathers, fathers[i]);
        }

        return fathers[i];
    }

}
相关推荐
金融小师妹20 分钟前
AI量化模型解析黄金3300关口博弈:市场聚焦“非农数据”的GRU-RNN混合架构推演
大数据·人工智能·算法
金融小师妹25 分钟前
基于LSTM-GRU混合网络的动态解析:美联储维稳政策与黄金单日跌1.5%的非线性关联
大数据·人工智能·算法
拾光拾趣录26 分钟前
ES6到HTTPS全链路连环拷问,99%人第3题就翻车?
前端·面试
白日梦想家-K1 小时前
题单【模拟与高精度】
开发语言·c++·算法
岁忧1 小时前
(LeetCode 面试经典 150 题) 138. 随机链表的复制 (哈希表)
java·c++·leetcode·链表·面试·go
重生之我是Java开发战士1 小时前
【C语言】内存函数与数据在内存中的存储
c语言·开发语言·算法
roman_日积跬步-终至千里2 小时前
【机器学习】“回归“算法模型的三个评估指标:MAE(衡量预测准确性)、MSE(放大大误差)、R²(说明模型解释能力)
算法·机器学习·回归
只会蓝桥杯能算acmer吗3 小时前
面试小总结
面试·职场和发展
小指纹4 小时前
图论-最短路Dijkstra算法
数据结构·c++·算法·深度优先·图论
赴3355 小时前
逻辑回归 银行贷款资格判断案列优化 交叉验证,调整阈值,下采样与过采样方法
算法·机器学习·逻辑回归·下采样·交叉验证·过采样·阈值