1. import java.util.*;
  2. /**
  3. * Source : https://oj.leetcode.com/problems/substring-with-concatenation-of-all-words/
  4. *
  5. * Created by lverpeng on 2017/7/13.
  6. *
  7. * You are given a string, S, and a list of words, L, that are all of the same length.
  8. * Find all starting indices of substring(s) in S that is a concatenation of each word
  9. * in L exactly once and without any intervening characters.
  10. *
  11. * For example, given:
  12. * S: "barfoothefoobarman"
  13. * L: ["foo", "bar"]
  14. *
  15. * You should return the indices: [0,9].
  16. * (order does not matter).
  17. *
  18. *
  19. * 找出L中单词连接成的子串在字符串S中出现所有位置
  20. *
  21. */
  22. public class SubstringWithConcatenationOfAllWords {
  23. /**
  24. *
  25. * 将strArr中的所有单词放在hash表中,单词为key,相同单词出现的次数为value
  26. *
  27. * strArr中所有单词长度一致,S重要出现所有单词的连接成为的字符串,那么S的长度一定要大于strArr中所有单词的总长度,
  28. * 也就是起始字符在0-(S.length - strArr.length * strArr[0].length)之间,
  29. * 以上面的位置为起始位置,依次判断接下来的strArr.length个单词是否正好是上面hash表中的单词,并判断相同单词出现次数,
  30. * 如果没有出现则退出循环,如果出现的次数大于hash表中单词的次数也break
  31. * 一轮循环完成之后判断循环的次数是否正好是strArr.length,如果是,说明S包含strArr连接的字符串,记录此时的起始位置到结果中
  32. *
  33. * @param S
  34. * @param strArr
  35. * @return
  36. */
  37. public int[] findSubstring (String S, String[] strArr) {
  38. if (S.length() < 1 | strArr.length < 1) {
  39. return new int[]{};
  40. }
  41. Map<String, Integer> wordMap = new HashMap<String, Integer>(); // 存放strArr单词的哈希表
  42. for (String str : strArr) {
  43. if (wordMap.keySet().contains(str)) {
  44. wordMap.put(str, wordMap.get(str) + 1);
  45. } else {
  46. wordMap.put(str, 1);
  47. }
  48. }
  49. int loopCount = 0;
  50. int arrLen = strArr.length;
  51. int wordLen = strArr[0].length();
  52. int arrStrLen = arrLen * wordLen;
  53. List<Integer> result = new ArrayList<Integer>();
  54. for (int i = 0; i < S.length() - arrStrLen; i++) {
  55. int j = 0;
  56. Map<String, Integer> subStrMap = new HashMap<String, Integer>();
  57. for (j = 0; j < arrLen; j++) {
  58. loopCount ++;
  59. String subStr = S.substring(i + j * wordLen, i + j * wordLen + wordLen);
  60. if (!wordMap.keySet().contains(subStr)) {
  61. break;
  62. } else {
  63. if (subStrMap.keySet().contains(subStr)) {
  64. subStrMap.put(subStr, subStrMap.get(subStr) + 1);
  65. } else {
  66. subStrMap.put(subStr, 1);
  67. }
  68. }
  69. if (subStrMap.get(subStr) > wordMap.get(subStr)) {
  70. break;
  71. }
  72. }
  73. if (j == arrLen) {
  74. result.add(i);
  75. }
  76. }
  77. System.out.println("loopCount------->" + loopCount);
  78. int[] res = new int[result.size()];
  79. for (int i = 0; i < result.size(); i++) {
  80. res[i] = result.get(i);
  81. }
  82. return res;
  83. }
  84. /**
  85. * 上面是以步长为1进行循环,下面以步长为word长度进行循环
  86. *
  87. * @param S
  88. * @param strArr
  89. * @return
  90. */
  91. public int[] findSubstring1 (String S, String[] strArr) {
  92. if (S.length() < 1 || strArr.length < 1) {
  93. return new int[]{};
  94. }
  95. Map<String, Integer> wordMap = new HashMap<String, Integer>();
  96. for (int i = 0; i < strArr.length; i++) {
  97. if (wordMap.keySet().contains(strArr[i])) {
  98. wordMap.put(strArr[i], wordMap.get(strArr[i]));
  99. } else {
  100. wordMap.put(strArr[i], 1);
  101. }
  102. }
  103. List<Integer> result = new ArrayList<Integer>();
  104. int wordLen = strArr[0].length();
  105. Map<String, Integer> subStrMap = new HashMap<String, Integer>();
  106. int loopCount = 0;
  107. for (int i = 0; i < wordLen; i++) {
  108. int count = 0;
  109. int left = i; // 记录待匹配子串起始位置
  110. for (int j = i; j < S.length() - wordLen; j += wordLen) {
  111. loopCount ++;
  112. String subStr = S.substring(j, j + wordLen);
  113. if (wordMap.keySet().contains(subStr)) {
  114. if (subStrMap.keySet().contains(subStr)) {
  115. subStrMap.put(subStr, subStrMap.get(subStr) + 1);
  116. } else {
  117. subStrMap.put(subStr, 1);
  118. }
  119. count ++;
  120. if (subStrMap.get(subStr) <= wordMap.get(subStr)) {
  121. count ++;
  122. } else {
  123. // 说明当前开始位置不匹配
  124. while (subStrMap.get(subStr) > wordMap.get(subStr)) {
  125. //
  126. String startWord = S.substring(left, left + wordLen);
  127. subStrMap.put(startWord, subStrMap.get(startWord) - 1);
  128. left += wordLen;
  129. count --;
  130. }
  131. }
  132. if (count == strArr.length) {
  133. // 找到了
  134. result.add(left);
  135. // 向后移动一个单词
  136. count --;
  137. String startWord = S.substring(left, left + wordLen);
  138. subStrMap.put(startWord, subStrMap.get(startWord) - 1);
  139. left += wordLen;
  140. }
  141. } else {
  142. // 清空变量,重新开始查找
  143. left = j + wordLen;
  144. subStrMap.clear();
  145. count = 0;
  146. }
  147. }
  148. }
  149. System.out.println("loopCount------->" + loopCount);
  150. int[] res = new int[result.size()];
  151. for (int i = 0; i < result.size(); i++) {
  152. res[i] = result.get(i);
  153. }
  154. return res;
  155. }
  156. public static void main(String[] args) {
  157. SubstringWithConcatenationOfAllWords substringWithConcatenationOfAllWords = new SubstringWithConcatenationOfAllWords();
  158. String S = "barfoothefoobarman";
  159. String[] strArr = new String[]{"foo", "bar"};
  160. System.out.println(Arrays.toString(substringWithConcatenationOfAllWords.findSubstring(S, strArr)));
  161. System.out.println(Arrays.toString(substringWithConcatenationOfAllWords.findSubstring1(S, strArr)));
  162. }
  163. }

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分析和实现

    题目大意是传入一个字符串s和一个字符串数组words,其中words中的所有字符串均等长.要在s中找所有的索引index,使得以s[index]为起始字符的长为words中字符串总长的s的子串是由wo ...

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

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

  9. 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 ...

  10. 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. redis使用规范文档 20170522版

    运维redis很久了,一直是口头给rd说各种要求,尝试把这些规范总结成文档 摘选一些可能比较通用的规则如下: 强制:所有的key设置过期时间(最长可设置过期时间10天,如有特殊要求,联系dba说明原因 ...

  2. mysql启动错误,提示crash 错误

    :: mysqld_safe Starting mysqld daemon with databases from /data/mysql_data -- :: [Note] Plugin 'FEDE ...

  3. AutoCAD开发2--添加带属性的点

    Private Sub CommandButton11_Click() Dim pPoint As AcadPoint Dim DataType(0 To 1) As Integer Dim Data ...

  4. IoGetRelatedDeviceObject学习

    PDEVICE_OBJECT IoGetRelatedDeviceObject( IN PFILE_OBJECT FileObject ) /*++ Routine Description: This ...

  5. JAVA中内部类(匿名内部类)访问的局部变量为什么要用final修饰?

    本文主要记录:在JAVA中,(局部)内部类访问某个局部变量,为什么这个局部变量一定需要用final 关键字修饰? 首先,什么是局部变量?这里的局部是:在方法里面定义的变量. 因此,内部类能够访问某局部 ...

  6. Hbase 性能改进

    第一种性能改进方式:

  7. IT资产管理—采购与合同管理功能

  8. SGU 176 Flow construction (有源有汇有上下界最小流)

    题意:给定 n 个点,m 条有向边,如果有向边的标号是1的话,就表示该边的上界下界都为容量 ,如果有向边的标号为0的哈,表示该边的下界为0,上界为容量 ,现在问,从 1 到 n 的最小流是多少,并输出 ...

  9. std::string的拷贝赋值研究

    说明:以下涉及的std::string的源代码摘自4.8.2版本.结论:std::string的拷贝复制是基于引用计数的浅拷贝,因此它们指向相同的数据地址. // std::string类定义type ...

  10. Pycharm2018的激活方法或破解方法

    1.授权服务器激活 优点:方便快捷 缺点:激活的人数多了就容易被封杀,所以可能经常需要去激活 选择License server激活,然后填入: idea.qmanga.com或http://xidea ...