题目大意是传入一个字符串s和一个字符串数组words,其中words中的所有字符串均等长。要在s中找所有的索引index,使得以s[index]为起始字符的长为words中字符串总长的s的子串是由words中的所有字符串(每个出现一次)拼接而成。


这个题目有点恶趣味,而且也很难找到特别有效的优化方案。下面说说我的思路:

首先记s的长度为m,而words的长度为k,words中字符串的长度为n。显然当n*k>m时只需要返回一个空集即可,因此可以认为n*k<=m。

首先我们将所有的words中的元素插入到红黑树中,并将words[i]映射为words[i]被插入的次数(即words[i]在words中出现的个数)。这所花费的时间为插入比较次数*每次比较时间=O(log2(k))*n=O(nlog2(k))。现在我们所需的数据结构也就建立完毕了。

map = empty-red-black-tree

for(i = 0; i < words.length; i = i + 1)

  times = map.get(words[i])

  if times ==  NIL

    times = 0

  times = times + 1

  map.put(words[i], times)

接着我们需要利用一种有趣的想法。先以0~nk作为起始匹配子串,并每次向右移动k位,即下一次判断k~(n+1)k是否可行。同时用一个remain变量记录尚未被匹配的words中字符串的数目。一旦remain为0,则意味着当前扫描的子串出现了所有words中的字符串。并且次数map中记录的映射值应该表示当前扫描的字串中对于各个字符串所缺少出现的次数。

每次子串右移k位时,我们只需要做类似移除头部k个字符所代表的字符串并加入后续k个字符所代表字符串的工作。移除首部长为k的子串,需要修改map中的值,其费用为O(nlog2(k)),而插入尾部长为k的子串,同样需要修改map中的值,其费用同样为O(nlog2(k))。故一次右移动所花费的总时间为O(nlog2(k)),还有一些O(1)的简单操作,比如修改remain的值,以及判断当前子串是否满足要求。

在子串尾部触及s的结尾时,停止当前循环。并展开下一次循环,其以1~nk+1作为起始匹配子串。外部循环的结束条件是t~nk+t的起始情况都已经被考虑过了,其中0<=t<n。这样我们就扫描了所有有可能满足条件的s的子串,分别以0,1,...,m-nk作为起始。按照前面所说对map的操作发生的次数应该为m-nk,但是实际上对map的操作发生的次数应该是m次,不要忘记了对起始子串也存在着修正map和remain的工作,其次还存在着恢复map和remain的操作时间,为k次。因此上面的循环总共的时间复杂度为O(m+k)*O(nlog2(k))=O((m+k)nlog2(k))=O(mnlog2(k))。

remove(subs)  //从当前扫描子串中移除n长字符串subs

  times = map.get(subs)

  if times == NIL

    return

  times = times+1

  map.put(subs, times)

  if(times > 0)

    remain = remain+1

append(subs) //向当前扫描子串中加入n长字符串subs

  times = map.get(subs)

  if times == NIL

    return

  times = times-1

  map.put(subs, times)

  if(times >= 0)

    remain = remain+1

.............

result = empty-list

remain = n

for(i = 0; i < n; i = i+1)

  for(j = i; j < m; j = j+n)

    start = j - n*k

    end = j

    if(start >= 0)

      startStr = s.substring(start, start  + n)

      remove(startStr)

    if(end + k <= m)

      endStr = s.substring(end, end + n)

      append(endStr)

    if(remain == 0)

      result.add(begin+n)

  reset map values and remain

这就是上述想法的实现代码。综合初始化操作,总的时间复杂度为O(nlog2(k))+O(mnlog2(k))=O(mnlog2(k))<O(m^2)。

稍微说一下优化前面一些步骤的想法,首先从map中取值修改后插回map的操作,可以利用一个包装器包装整数,之后取回后只需修改包装器中的整数,而不需重新插回。还有就是substring的算法,也可以利用包装器来直接包装s,并限定其有效范围,这样就可以将创建子字符串的费用优化到O(1)。当然这些优化的都不是必须的,因为它们是否优化都无法改变整体的时间复杂度。而整体的时间复杂度取决于利用字符串向map取值操作上,利用散列以及缓存散列码的技术可以真正对上述过程产生优化。假如散列足够优化,即所有的字符串都会被散列到不同的槽中,那么为所有字符串计算散列码的时间复杂度为O(n*k)+O(m*n)=O(mn),而每次取值并插回的时间复杂度为O(n),而计算返回值时的双重循环共执行m次,故时间复杂度为O(n)*O(m)=O(nm),因此结果的时间复杂度为O(mn)+O(nm)=O(mn),当然这只是理想状态而已。

散列表中有一种变形,称为完全散列,其在最坏情况下依旧有着O(1)的查询时间复杂度。有兴趣的童鞋可以去自己去找找资料,利用完全散列就可以保证不同的字符串被散列到不同的槽中。上面所提及的优化也就有可能实现了。


最后提供AC代码:

 package cn.dalt.leetcode;

 import org.hibernate.internal.util.ValueHolder;

 import java.util.*;

 /**
  * Created by dalt on 2017/6/22.
  */
 public class SubstringwithConcatenationofAllWords {
     private static final class Substring {
         private char[] data;
         private int from;
         private int length;

         public Substring(char[] data, int from, int length) {
             this.data = data;
             this.from = from;
             this.length = length;
         }

         public Substring substring(int from, int length) {
             return new Substring(data, this.from + from, length);
         }

         Integer cachedHashCode;

         @Override
         public int hashCode() {
             if (cachedHashCode == cachedHashCode) {
                 int value = 0;
                 for (int i = from, bound = from + length; i < bound; i++) {
                     value = (value << 5) - value + data[i];
                 }
                 cachedHashCode = Integer.valueOf(value);
             }
             return cachedHashCode.intValue();
         }

         public char charAt(int i) {
             return data[i + from];
         }

         public int size() {
             return length;
         }

         @Override
         public boolean equals(Object obj) {
             if (obj == null)
                 return false;
             if (obj.getClass() != Substring.class)
                 return false;
             Substring other = (Substring) obj;
             if (hashCode() != other.hashCode() || length != other.length)
                 return false;
             for (int i = 0; i < length; i++) {
                 if (charAt(i) != other.charAt(i))
                     return false;
             }
             return true;
         }

         @Override
         public String toString() {
             return String.valueOf(data, from, length);
         }
     }

     private static final class IntHolder {
         private int value;
         private int storedValue;

         public IntHolder(int initValue) {
             value = initValue;
         }

         public void inc() {
             value++;
         }

         public void dec() {
             value--;
         }

         public void store() {
             storedValue = value;
         }

         public void restore() {
             value = storedValue;
         }

         public int getValue() {
             return value;
         }

         @Override
         public int hashCode() {
             return value;
         }

         @Override
         public String toString() {
             return value + "(" + storedValue + ")";
         }

         @Override
         public boolean equals(Object obj) {
             if (obj == null)
                 return false;
             if (obj.getClass() == IntHolder.class) {
                 return ((IntHolder) obj).value == value;
             }
             return false;
         }
     }

     public List<Integer> findSubstring(String s, String[] words) {
         if (words.length == 0) {
             List<Integer> result = new ArrayList<>(s.length());
             for (int i = 0, bound = s.length(); i < bound; i++) {
                 result.add(Integer.valueOf(i));
             }
             return result;
         }
         int m = s.length();
         int n = words[0].length();
         int k = words.length;

         Map<Substring, IntHolder> map = new HashMap<>(k);
         for (String word : words) {
             Substring pack = new Substring(word.toCharArray(), 0, word.length());
             IntHolder holder = map.get(pack);
             if (holder == null) {
                 holder = new IntHolder(0);
                 map.put(pack, holder);
             }
             holder.inc();
         }

         List<IntHolder> holders = new ArrayList<IntHolder>(map.values());
         for (IntHolder holder : holders) {
             holder.store();
         }
         List<Integer> result = new LinkedList<>();
         char[] sarray = s.toCharArray();
         for (int i = 0; i < n; i++) {
             for (IntHolder holder : holders) {
                 holder.restore();
             }
             int remain = words.length;
             for (int j = i; j < m; j = j + n) {
                 int start = j - n * k;
                 int end = j;
                 if (start >= 0) {
                     Substring sub = new Substring(sarray, start, n);
                     IntHolder times = map.get(sub);
                     if (times != null) {
                         times.inc();
                         if (times.getValue() > 0) {
                             remain++;
                         }
                     }
                 }
                 if (end + n <= m) {
                     Substring sub = new Substring(sarray, end, n);
                     IntHolder times = map.get(sub);
                     if (times != null) {
                         times.dec();
                         if (times.getValue() >= 0) {
                             remain--;
                         }
                     }
                 }
                 if (remain == 0) {
                     result.add(start + n);
                 }
             }
         }
         return result;
     }
 }

Leetcode:Substring with Concatenation of All Words分析和实现的更多相关文章

  1. LeetCode: Substring with Concatenation of All Words 解题报告

    Substring with Concatenation of All Words You are given a string, S, and a list of words, L, that ar ...

  2. [LeetCode] Substring with Concatenation of All Words 串联所有单词的子串

    You are given a string, s, and a list of words, words, that are all of the same length. Find all sta ...

  3. LeetCode:Substring with Concatenation of All Words (summarize)

    题目链接 You are given a string, S, and a list of words, L, that are all of the same length. Find all st ...

  4. [leetcode]Substring with Concatenation of All Words @ Python

    原题地址:https://oj.leetcode.com/problems/substring-with-concatenation-of-all-words/ 题意: You are given a ...

  5. Leetcode Substring with Concatenation of All Words

    You are given a string, S, and a list of words, L, that are all of the same length. Find all startin ...

  6. [LeetCode] Substring with Concatenation of All Words(good)

    You are given a string, S, and a list of words, L, that are all of the same length. Find all startin ...

  7. LeetCode()Substring with Concatenation of All Words 为什么我的超时呢?找不到原因了!!!

    超时代码 class Solution { public: vector<int> findSubstring(string s, vector<string>& wo ...

  8. LeetCode HashTable 30 Substring with Concatenation of All Words

    You are given a string, s, and a list of words, words, that are all of the same length. Find all sta ...

  9. leetcode面试准备: Substring with Concatenation of All Words

    leetcode面试准备: Substring with Concatenation of All Words 1 题目 You are given a string, s, and a list o ...

随机推荐

  1. 神经网络中的Softmax激活函数

    Softmax回归模型是logistic回归模型在多分类问题上的推广,适用于多分类问题中,且类别之间互斥的场合. Softmax将多个神经元的输出,映射到(0,1)区间内,可以看成是当前输出是属于各个 ...

  2. I.MX6 Android 5.1 纯Linux、U-Boot编译

    /***************************************************************************** * I.MX6 Android 5.1 纯 ...

  3. 21天学通C++_Day2

    继续学习,今天满课,相对学习内容较少,下面罗列内容: 0.常量 ▪字面常量: ▪使用关键字const声明的常量,const double Pi = 22.0/7; //后面有分号,跟定义变量一样 ▪使 ...

  4. Sublime Text 3下Emmet使用技巧

    链接:http://jingyan.baidu.com/article/92255446a87900851648f4d6.html

  5. postman安装Postman Interceptor 插件

    做后端开发避免不了进行接口调试,但是一般的项目都是前后端分离的,如果把前端代码下到本地,较为费事,这个时候就需要一个可以进行接口调试的工具.Postman就是一个不错的选择. Postman是什么? ...

  6. 剑指offer-第五章优化时间和空间效率(两个链表的第一个公共节点)

    思路1:要求的是两个链表的第一个公共节点,首先想到的是用栈来存放两个链表,然后依次从栈中抛出,直到最后一个相同的节点为止.但是要用到两个栈,空间复杂度为O(n): 思路2:从头到尾分别遍历两个链表得到 ...

  7. 初识ADO.NET

    摘要 作为.NET框架最重要的组件之一,ADO.NET扮演着应用程序与数据交互的重要的角色.本文将从宏观的角度来探讨ADO.NET,和大家一起了解ADO.NET来龙去脉以及ADO.NET的主要组成部分 ...

  8. sqlserver sql语句查看分区记录数、查看记录所在分区

    select count(1) ,$PARTITION.WorkDatePFN(workdate) from imgfile group by $PARTITION.WorkDatePFN(workd ...

  9. int 和 Integer 有什么区别

    原文地址:https://blog.csdn.net/chenliguan/article/details/53888018 1 int与Integer的基本使用对比 (1)Integer是int的包 ...

  10. OpenCL™ 2.0 – Pipes

    copy from http://developer.amd.com/community/blog/2014/10/31/opencl-2-0-pipes/ OpenCL™ 2.0 – Pipes I ...