zoukankan      html  css  js  c++  java
  • 字符串包含

    String 1:ABCD

    String 2:BAD

    答案是true,即String2里的字母在String1里也都有,或者说String2是String1的真子集。

    如果是下面两个字符串:

    String 1:ABCD

    String 2:BCE

    答案是false,因为字符串String2里的E字母不在字符串String1里。

    同时,如果string1:ABCD,string 2:AA,同样返回true。

    解法一 暴力求解

    对string2的每个字符都到string1里查找是否存在.复杂度是O(n*m)

    解法二 先排序再扫描

    如果允许排序的话,我们可以考虑下排序。比如可先对这两个字符串的字母进行排序,然后再同时对两个字串依次轮询。两个字串的排序需要(常规情况)O(m log m) + O(n log n)次操作,之后的线性扫描需要O(m+n)次操作。

    解法三  素数相乘

    假设有一个仅由字母组成字串,让每个字母与一个素数对应,从2开始,往后类推,A对应2,B对应3,C对应5,......。遍历第一个字串,把每个字母对应素数相乘。最终会得到一个整数。

    利用上面字母和素数的对应关系,对应第二个字符串中的字母,然后轮询,用每个字母对应的素数除前面得到的整数。如果结果有余数,说明结果为false。如果整个过程中没有余数,则说明第二个字符串是第一个的子集了(判断是不是真子集,可以比较两个字符串对应的素数乘积,若相等则不是真子集)。

    思路总结如下:

    1. 按照从小到大的顺序,用26个素数分别与字符'A'到'Z'一一对应。
    2. 遍历长字符串,求得每个字符对应素数的乘积。
    3. 遍历短字符串,判断乘积能否被短字符串中的字符对应的素数整除。
    4. 输出结果。

    如前所述,算法的时间复杂度为O(m+n)的最好的情况为O(n)(遍历短的字符串的第一个数,与长字符串素数的乘积相除,即出现余数,便可退出程序,返回false),n为长字串的长度,空间复杂度为O(1)。

    此种素数相乘的方法看似完美,但缺点是素数相乘的结果容易导致整数溢出。

    解法四 hash

    如果面试官继续追问,还有没有更好的办法呢?计数排序?除了计数排序呢?

    事实上,可以先把长字符串a中的所有字符都放入一个Hashtable里,然后轮询短字符串b,看短字符串b的每个字符是否都在Hashtable里,如果都存在,说明长字符串a包含短字符串b,否则,说明不包含。

    再进一步,我们可以对字符串A,用位运算(26bit整数表示)计算出一个“签名”,再用B中的字符到A里面进行查找。

    这里july用的是类似hash的做法:

    bool StringContain(string &a,string &b)
    {
        int hash = 0;
        for (int i = 0; i < a.length(); ++i)
        {
            hash |= (1 << (a[i] - 'A'));
        }
        for (int i = 0; i < b.length(); ++i)
        {
            if ((hash & (1 << (b[i] - 'A'))) == 0)
            {
                return false;
            }
        }
        return true;
    }

    这个方法的实质是用一个整数代替了hashtable,空间复杂度为O(1),时间复杂度还是O(n + m)。

    举一反三

    1、变位词

    • 如果两个字符串的字符一样,但是顺序不一样,被认为是兄弟字符串,比如bad和adb即为兄弟字符串,现提供一个字符串,如何在字典中迅速找到它的兄弟字符串,请描述数据结构和查询过程。

    题目来源:https://github.com/julycoding/The-Art-Of-Programming-By-July/blob/master/ebook/zh/01.02.md

  • 相关阅读:
    JavaScript面向对象之闭包的理解
    JavaScript面向对象之函数构造器的理解
    记录jq控制select 选中状态
    JavaScript面向对象之创建类和方法
    获取对象属相 点运算符 和方括号 运算符
    parseFloat 和 Number isNaN 转换
    Js数据类型和运算符
    break , continue 和 标签 跳出循环
    javaScript--循环语句
    三元运算符
  • 原文地址:https://www.cnblogs.com/ddddddwwwxx/p/5520701.html
Copyright © 2011-2022 走看看