-
Notifications
You must be signed in to change notification settings - Fork 369
/
Copy pathExtended_Euclidean_Algorithm.java
151 lines (116 loc) · 6.28 KB
/
Extended_Euclidean_Algorithm.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
/*
------------------------------------------------- PREREQUISITE ---------------------------------------------------------
--> Modular Multiplicative inverse exist when two numbers are CO-PRIME to each other in other words the two numbers gcd = 1
--> Modular Multiplicative inverse changes with respective modulo value and not remain same for the number.
---------------------------------------- Modulo Multiplicative Inverse(MMI) --------------------------------------------
Inverse of a number (A) will be = A^-1 or we can say that --> A * A^-1 will be equal to 1
but incase of modular multiplicative inverse of a number (A) is when the expression below is true
(A*A^-1) mod (C) = 1
from the expression we can say that A^-1 will be the inverse of A.
Example:-
We have to find the Modulo Multiplicative inverse of 2 whose mod is 5
From the formula
(A*A^-1) mod (C) = 1 now will be --> (2*A^-1) mod (5) = 1
we have to find the A^-1 value
simply we can use a loop to iterate upto the mod value and find the inverse by the below method
(2*A^-1) mod (5) = 1 now,
--> (2*1)%5 = 2 which is not equal to 1 so increment
--> (2*2)%5 = 4 which is not equal to 1 so increment
--> (2*1)%5 = 1 which is equal to 1 so,
for the number 2 the modular multiplicative inverse will be 3 when modulo is 5. If modulo changes the modulo multiplicative
inverse changes.
------------------------------------------- The Problem Statement ------------------------------------------------------
Given two numbers, the first one is the number which we want to find the modulo multiplicative inverse and
the second one is the modulo value
INPUT :- 2 <-- number which we want to find modulo multiplicative inverse
5 <-- the respective modulo value
OUTPUT :- 3
---------------------------------------- The Problem of normal approach ------------------------------------------------
To find mmi we can simply use a for loop starting from 0 to value of mod in-between the mmi exist a BigO(n) approach,
it is good for small numbers,the problem starts when the modulo is increased lets say the value of modulo is 21783,
in this case the loop runs for 21783 times to find the mmi in worst case which takes a huge time to compute for that
we go for an algorithm called the EXTENDED EUCLIDEAN ALGORITHM to find the mmi which takes BigO(log(n)) time which is
a significant reduction in time complexity, log(21783) = 4.338 so the loop at worst case itself runs 4 times only.
------------------------------------------ Extended Euclidean Algorithm ------------------------------------------------
a.x + b.y = gcd --(1)
And x1 and y1 are results for inputs b%a and a
(b%a).x1 + a.y1 = gcd
When we put b%a = (b – ([b/a]).a) in above,
we get following. Note that [b/a] is floor(b/a)
(b – ([b/a]).a).x1 + a.y1 = gcd
Above equation can also be written as below
b.x1 + a.(y1 – ([b/a]).x1) = gcd --(2)
After comparing coefficients of ‘a’ and ‘b’ in (1) and (2), we get following,
x = y1 – (b/a) * x1
y = x1
----------------------------------------------- Complexities -----------------------------------------------------------
Time Complexity --> BigO(log(n))
Space Complexity --> BigO(1)
Space Complexity --> BigO(n) --> In case of recursion.
*/
//Importing Scanner Class to get input from the user.
import java.util.Scanner;
public class Extended_Euclidean_Algorithm
{
// Declaring variables as global and static so that we can use anywhere else in the program.
static int t1=0;
static int t2=1;
static int quotient=0;
static int remainder =0;
public static void main(String[] args)
{
// Telling the uses that the number should be co-prime
System.out.println("\nTO FIND MMI THE TWO NUMBERS SHOULD BE CO-PRIME TO EACH OTHER OR THE GCD OF TWO NUMBERS SHOULD BE EQUAL TO 1");
System.out.println("OTHERWISE THE INVERSE DOESN'T EXIST\n");
// Initializing the Scanner Class
Scanner sc = new Scanner(System.in);
// Reading the number which user want to find inverse
System.out.print("Enter a number you want to find inverse = ");
int number = sc.nextInt();
// Reading the modulo value from the user
System.out.print("Enter the modulo value = ");
int modulo = sc.nextInt();
// Calling the recursive function.
int gcd = mmi(number,modulo);
// Checking whether the inverse exist or not
// gcd!=1 --> If gcd is not equal to 1 then inverse doesn't exist
if(gcd !=1)
{
System.out.println("\nInverse doesn't exist because they are not co-primes to each other");
}
else
{
// Computing the value of Modular multiplicative inverse
// t1 is negative --> t1 = t1+modulo
// t1 is positive --> print t1 no problem
// Since it takes another if statement inside this else statement we can use the below operation to combine the results.
int res = (t1%modulo + modulo)%modulo;
System.out.println("\nThe inverse of the number " + number + " is " + res + " when modulo is " + modulo);
}
}
// Function which performs the Modular Multiplicative inverse
// To keep the function safe and accessible inside this class only, I declared it as private you can declare it as public too.
private static int mmi(int number,int modulo)
{
// BASE CASE of the Recursion
if(number ==0)
{
t1=0;
t2 = 1;
return modulo;
}
// Finding the remainder of the number with modulo
remainder = modulo%number;
// Recursively calling the function to find gcd so we can say that the mmi exist or not.
int gcd = mmi(remainder,number);
// Finding the quotient of the number with modulo
quotient = modulo/number;
// The Extended Euclidean Algorithm
// To find the value of MMI
int temp = t1;
t1 = t2-quotient*t1;
t2 = temp;
// Returning the gcd to check whether the inverse exist or not
return gcd;
}
}