Update Anagrams and add unit test (#3002)

* Add test for Anagrams
* Update Anagrams.java

Co-authored-by: Yang Libin <contact@yanglibin.info>
This commit is contained in:
Aldo Telese 2022-04-04 10:34:27 +02:00 committed by GitHub
parent 140f6ec6e3
commit 8d099ee7d1
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 78 additions and 81 deletions

View File

@ -1,21 +1,23 @@
/** Author : Siddhant Swarup Mallick
* Github : https://github.com/siddhant2002
*/
/** PROBLEM DESCRIPTION :
* An anagram is a word or phrase formed by rearranging the letters of a different word or phrase, typically using all the original letters exactly once.[1] For example, the word anagram itself can be rearranged into nag a ram, also the word binary into brainy and the word adobe into abode. Reference from https://en.wikipedia.org/wiki/Anagram
*/
package com.thealgorithms.strings; package com.thealgorithms.strings;
import java.util.*;
public class Anagrams import java.util.Arrays;
{ import java.util.HashMap;
/**
* An anagram is a word or phrase formed by rearranging the letters of a different word or phrase,
* typically using all the original letters exactly once.[1]
* For example, the word anagram itself can be rearranged into nag a ram,
* also the word binary into brainy and the word adobe into abode.
* Reference from https://en.wikipedia.org/wiki/Anagram
*/
public class Anagrams {
// 4 approaches are provided for anagram checking. approach 2 and approach 3 are similar but differ in running time. // 4 approaches are provided for anagram checking. approach 2 and approach 3 are similar but differ in running time.
public static void main(String args[]) { public static void main(String[] args) {
String first = "deal"; String first = "deal";
String second = "lead"; String second = "lead";
// All the below methods takes input but doesn't return any output to the main method. // All the below methods takes input but doesn't return any output to the main method.
Anagrams nm=new Anagrams(); Anagrams nm = new Anagrams();
System.out.println(nm.approach2(first, second)); /* To activate methods for different approaches*/ System.out.println(nm.approach2(first, second)); /* To activate methods for different approaches*/
System.out.println(nm.approach1(first, second)); /* To activate methods for different approaches*/ System.out.println(nm.approach1(first, second)); /* To activate methods for different approaches*/
System.out.println(nm.approach3(first, second)); /* To activate methods for different approaches*/ System.out.println(nm.approach3(first, second)); /* To activate methods for different approaches*/
@ -37,52 +39,38 @@ public class Anagrams
*/ */
} }
boolean approach1(String s, String t) boolean approach1(String s, String t) {
{ if (s.length() != t.length()) {
if (s.length() != t.length())
{
return false; return false;
} } else {
else
{
char c[] = s.toCharArray(); char c[] = s.toCharArray();
char d[] = t.toCharArray(); char d[] = t.toCharArray();
Arrays.sort(c); Arrays.sort(c);
Arrays.sort(d); /* In this approach the strings are stored in the character arrays and both the arrays are sorted. After that both the arrays are compared for checking anangram */ Arrays.sort(d); /* In this approach the strings are stored in the character arrays and both the arrays are sorted. After that both the arrays are compared for checking anangram */
if (Arrays.equals(c, d)) if (Arrays.equals(c, d)) {
{
return true; return true;
} else } else {
{
return false; return false;
} }
} }
} }
boolean approach2(String a, String b) boolean approach2(String a, String b) {
{ if (a.length() != b.length()) {
if(a.length()!=b.length())
{
return false; return false;
} } else {
else int m[] = new int[26];
{ int n[] = new int[26];
int m[]=new int[26]; for (char c : a.toCharArray()) {
int n[]=new int[26]; m[c - 'a']++;
for(char c: a.toCharArray())
{
m[c-'a']++;
} }
// In this approach the frequency of both the strings are stored and after that the frequencies are iterated from 0 to 26(from 'a' to 'z' ). If the frequencies match then anagram message is displayed in the form of boolean format // In this approach the frequency of both the strings are stored and after that the frequencies are iterated from 0 to 26(from 'a' to 'z' ). If the frequencies match then anagram message is displayed in the form of boolean format
// Running time and space complexity of this algo is less as compared to others // Running time and space complexity of this algo is less as compared to others
for(char c:b.toCharArray()) for (char c : b.toCharArray()) {
{ n[c - 'a']++;
n[c-'a']++;
} }
for(int i=0;i<26;i++) for (int i = 0; i < 26; i++) {
{ if (m[i] != n[i]) {
if(m[i]!=n[i])
{
return false; return false;
} }
} }
@ -90,57 +78,45 @@ public class Anagrams
} }
} }
boolean approach3(String s, String t) boolean approach3(String s, String t) {
{ if (s.length() != t.length()) {
if(s.length()!=t.length())
{
return false; return false;
} }
// this is similar to approach number 2 but here the string is not converted to character array // this is similar to approach number 2 but here the string is not converted to character array
else else {
{ int a[] = new int[26];
int a[]=new int[26]; int b[] = new int[26];
int b[]=new int[26]; int k = s.length();
int k=s.length(); for (int i = 0; i < k; i++) {
for(int i=0;i<k;i++) a[s.charAt(i) - 'a']++;
{ b[t.charAt(i) - 'a']++;
a[s.charAt(i)-'a']++;
b[t.charAt(i)-'a']++;
} }
for(int i=0;i<26;i++) for (int i = 0; i < 26; i++) {
{ if (a[i] != b[i])
if(a[i]!=b[i])
return false; return false;
} }
return true; return true;
} }
} }
boolean approach4(String s, String t) boolean approach4(String s, String t) {
{ if (s.length() != t.length()) {
if(s.length()!=t.length()) return false;
{
return false;
} }
// This approach is done using hashmap where frequencies are stored and checked iteratively and if all the frequencies of first string match with the second string then anagram message is displayed in boolean format // This approach is done using hashmap where frequencies are stored and checked iteratively and if all the frequencies of first string match with the second string then anagram message is displayed in boolean format
else else {
{ HashMap<Character, Integer> nm = new HashMap<>();
HashMap<Character,Integer> nm=new HashMap<>(); HashMap<Character, Integer> kk = new HashMap<>();
HashMap<Character,Integer> kk=new HashMap<>(); for (char c : s.toCharArray()) {
for(char c: s.toCharArray()) nm.put(c, nm.getOrDefault(c, 0) + 1);
{
nm.put(c, nm.getOrDefault(c,0)+1);
} }
for(char c: t.toCharArray()) for (char c : t.toCharArray()) {
{
kk.put(c, kk.getOrDefault(c,0)+1); kk.put(c, kk.getOrDefault(c, 0) + 1);
} }
// It checks for equal frequencies // It checks for equal frequencies
for(char c:nm.keySet()) for (char c : nm.keySet()) {
{ if (!nm.get(c).equals(kk.get(c))) {
if(!nm.get(c).equals(kk.get(c)))
{
return false; return false;
} }
} }

View File

@ -0,0 +1,21 @@
package com.thealgorithms.strings;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
public class AnagramsTest {
@Test
public void isAlphabetical() {
String input1 = "late";
Anagrams anagrams = new Anagrams();
assertTrue(anagrams.approach1(input1, "tale"));
assertTrue(anagrams.approach1(input1, "teal"));
assertTrue(anagrams.approach2(input1, "tale"));
assertTrue(anagrams.approach2(input1, "teal"));
assertTrue(anagrams.approach3(input1, "tale"));
assertTrue(anagrams.approach3(input1, "teal"));
assertTrue(anagrams.approach4(input1, "tale"));
assertTrue(anagrams.approach4(input1, "teal"));
}
}