mirror of
https://github.com/TheAlgorithms/Java.git
synced 2025-07-10 21:43:15 +08:00
Added new File WordBoggle.java in Java/Misc (#2007)
* Added New Java file in Java/Misc * Update largestRange.java * Update largestRange.java * Update largestRange.java * Update largestRange.java * Revert "Largest range" * Added WordBoggle.java * Update WordBoggle.java * Added RangeInSortedArray.java in Java/Misc
This commit is contained in:
82
Misc/RangeInSortedArray.java
Normal file
82
Misc/RangeInSortedArray.java
Normal file
@ -0,0 +1,82 @@
|
||||
package Misc;
|
||||
|
||||
import java.util.*;
|
||||
|
||||
public class RangeInSortedArray {
|
||||
|
||||
public static void main(String[] args) {
|
||||
// Testcases
|
||||
assert Arrays.equals(sortedRange(new int[] {1, 2, 3, 3, 3, 4, 5}, 3), new int[] {2, 4});
|
||||
assert Arrays.equals(sortedRange(new int[] {1, 2, 3, 3, 3, 4, 5}, 4), new int[] {5, 5});
|
||||
assert Arrays.equals(sortedRange(new int[] {0, 1, 2}, 3), new int[] {-1, -1});
|
||||
}
|
||||
|
||||
// Get the 1st and last occurrence index of a number 'key' in a non-decreasing array 'nums'
|
||||
// Gives [-1, -1] in case element doesn't exist in array
|
||||
public static int[] sortedRange(int[] nums, int key) {
|
||||
int[] range = new int[] {-1, -1};
|
||||
alteredBinSearchIter(nums, key, 0, nums.length - 1, range, true);
|
||||
alteredBinSearchIter(nums, key, 0, nums.length - 1, range, false);
|
||||
return range;
|
||||
}
|
||||
|
||||
// Recursive altered binary search which searches for leftmost as well as rightmost occurrence of
|
||||
// 'key'
|
||||
public static void alteredBinSearch(
|
||||
int[] nums, int key, int left, int right, int[] range, boolean goLeft) {
|
||||
if (left > right) return;
|
||||
int mid = (left + right) / 2;
|
||||
if (nums[mid] > key) alteredBinSearch(nums, key, left, mid - 1, range, goLeft);
|
||||
else if (nums[mid] < key) alteredBinSearch(nums, key, mid + 1, right, range, goLeft);
|
||||
else {
|
||||
if (goLeft) {
|
||||
if (mid == 0 || nums[mid - 1] != key) range[0] = mid;
|
||||
else alteredBinSearch(nums, key, left, mid - 1, range, goLeft);
|
||||
} else {
|
||||
if (mid == nums.length - 1 || nums[mid + 1] != key) range[1] = mid;
|
||||
else alteredBinSearch(nums, key, mid + 1, right, range, goLeft);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Iterative altered binary search which searches for leftmost as well as rightmost occurrence of
|
||||
// 'key'
|
||||
public static void alteredBinSearchIter(
|
||||
int[] nums, int key, int left, int right, int[] range, boolean goLeft) {
|
||||
while (left <= right) {
|
||||
int mid = (left + right) / 2;
|
||||
if (nums[mid] > key) right = mid - 1;
|
||||
else if (nums[mid] < key) left = mid + 1;
|
||||
else {
|
||||
if (goLeft) {
|
||||
if (mid == 0 || nums[mid - 1] != key) {
|
||||
range[0] = mid;
|
||||
return;
|
||||
} else right = mid - 1;
|
||||
} else {
|
||||
if (mid == nums.length - 1 || nums[mid + 1] != key) {
|
||||
range[1] = mid;
|
||||
return;
|
||||
} else left = mid + 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static int getCountLessThan(int[] nums, int key) {
|
||||
return getLessThan(nums, key, 0, nums.length - 1);
|
||||
}
|
||||
|
||||
public static int getLessThan(int[] nums, int key, int left, int right) {
|
||||
int count = 0;
|
||||
while (left <= right) {
|
||||
int mid = (left + right) / 2;
|
||||
if (nums[mid] > key) right = mid - 1;
|
||||
else if (nums[mid] <= key) {
|
||||
count = mid + 1; // Atleast mid+1 elements exist which are <= key
|
||||
left = mid + 1;
|
||||
}
|
||||
}
|
||||
return count;
|
||||
}
|
||||
}
|
130
Misc/WordBoggle.java
Normal file
130
Misc/WordBoggle.java
Normal file
@ -0,0 +1,130 @@
|
||||
import java.util.*;
|
||||
|
||||
public class WordBoggle {
|
||||
|
||||
/**
|
||||
* O(nm * 8^s + ws) time where n=width of boggle board, m=height of boggle board, s=length of
|
||||
* longest word in string array, w= length of string array, 8 is due to 8 explorable neighbours
|
||||
* O(nm + ws) space.
|
||||
*/
|
||||
public static List<String> boggleBoard(char[][] board, String[] words) {
|
||||
Trie trie = new Trie();
|
||||
for (String word : words) trie.add(word);
|
||||
Set<String> finalWords = new HashSet<>();
|
||||
boolean[][] visited = new boolean[board.length][board.length];
|
||||
for (int i = 0; i < board.length; i++)
|
||||
for (int j = 0; j < board[i].length; j++)
|
||||
explore(i, j, board, trie.root, visited, finalWords);
|
||||
return new ArrayList<>(finalWords);
|
||||
}
|
||||
|
||||
public static void main(String[] args) {
|
||||
// Testcase
|
||||
List<String> ans =
|
||||
new ArrayList<>(
|
||||
Arrays.asList("a", "boggle", "this", "NOTRE_PEATED", "is", "simple", "board"));
|
||||
assert (boggleBoard(
|
||||
new char[][] {
|
||||
{'t', 'h', 'i', 's', 'i', 's', 'a'},
|
||||
{'s', 'i', 'm', 'p', 'l', 'e', 'x'},
|
||||
{'b', 'x', 'x', 'x', 'x', 'e', 'b'},
|
||||
{'x', 'o', 'g', 'g', 'l', 'x', 'o'},
|
||||
{'x', 'x', 'x', 'D', 'T', 'r', 'a'},
|
||||
{'R', 'E', 'P', 'E', 'A', 'd', 'x'},
|
||||
{'x', 'x', 'x', 'x', 'x', 'x', 'x'},
|
||||
{'N', 'O', 'T', 'R', 'E', '_', 'P'},
|
||||
{'x', 'x', 'D', 'E', 'T', 'A', 'E'},
|
||||
},
|
||||
new String[] {
|
||||
"this",
|
||||
"is",
|
||||
"not",
|
||||
"a",
|
||||
"simple",
|
||||
"test",
|
||||
"boggle",
|
||||
"board",
|
||||
"REPEATED",
|
||||
"NOTRE_PEATED",
|
||||
})
|
||||
.equals(ans));
|
||||
}
|
||||
|
||||
public static void explore(
|
||||
int i,
|
||||
int j,
|
||||
char[][] board,
|
||||
TrieNode trieNode,
|
||||
boolean[][] visited,
|
||||
Set<String> finalWords) {
|
||||
if (visited[i][j]) return;
|
||||
|
||||
char letter = board[i][j];
|
||||
if (!trieNode.children.containsKey(letter)) {
|
||||
return;
|
||||
}
|
||||
visited[i][j] = true;
|
||||
trieNode = trieNode.children.get(letter);
|
||||
if (trieNode.children.containsKey('*')) finalWords.add(trieNode.word);
|
||||
|
||||
List<Integer[]> neighbors = getNeighbors(i, j, board);
|
||||
for (Integer[] neighbor : neighbors)
|
||||
explore(neighbor[0], neighbor[1], board, trieNode, visited, finalWords);
|
||||
|
||||
visited[i][j] = false;
|
||||
}
|
||||
|
||||
public static List<Integer[]> getNeighbors(int i, int j, char[][] board) {
|
||||
List<Integer[]> neighbors = new ArrayList<>();
|
||||
if (i > 0 && j > 0) neighbors.add(new Integer[] {i - 1, j - 1});
|
||||
|
||||
if (i > 0 && j < board[0].length - 1) neighbors.add(new Integer[] {i - 1, j + 1});
|
||||
|
||||
if (i < board.length - 1 && j < board[0].length - 1)
|
||||
neighbors.add(new Integer[] {i + 1, j + 1});
|
||||
|
||||
if (i < board.length - 1 && j > 0) neighbors.add(new Integer[] {i + 1, j - 1});
|
||||
|
||||
if (i > 0) neighbors.add(new Integer[] {i - 1, j});
|
||||
|
||||
if (i < board.length - 1) neighbors.add(new Integer[] {i + 1, j});
|
||||
|
||||
if (j > 0) neighbors.add(new Integer[] {i, j - 1});
|
||||
|
||||
if (j < board[0].length - 1) neighbors.add(new Integer[] {i, j + 1});
|
||||
|
||||
return neighbors;
|
||||
}
|
||||
}
|
||||
|
||||
// Trie used to optimize string search
|
||||
class TrieNode {
|
||||
|
||||
Map<Character, TrieNode> children = new HashMap<>();
|
||||
String word = "";
|
||||
}
|
||||
|
||||
class Trie {
|
||||
|
||||
TrieNode root;
|
||||
char endSymbol;
|
||||
|
||||
public Trie() {
|
||||
this.root = new TrieNode();
|
||||
this.endSymbol = '*';
|
||||
}
|
||||
|
||||
public void add(String str) {
|
||||
TrieNode node = this.root;
|
||||
for (int i = 0; i < str.length(); i++) {
|
||||
char letter = str.charAt(i);
|
||||
if (!node.children.containsKey(letter)) {
|
||||
TrieNode newNode = new TrieNode();
|
||||
node.children.put(letter, newNode);
|
||||
}
|
||||
node = node.children.get(letter);
|
||||
}
|
||||
node.children.put(this.endSymbol, null);
|
||||
node.word = str;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user