Picture Source: theatlantic
Algorithms form the backbone of modern computing, enabling efficient problem-solving and optimization in various domains. Understanding the performance characteristics of algorithms is crucial for software developers and computer scientists. This article delves into the fascinating realm of Analysis of Algorithms, exploring its significance and highlighting a comprehensive project that sheds light on solving problems using Python, Java, and MATLAB.
Analysis of Algorithms is a field of study that aims to evaluate and compare the efficiency and resource utilization of different algorithms. It involves quantifying an algorithm's time complexity, space complexity, and other factors that impact its performance. By analyzing algorithms, developers can make informed decisions about choosing the most suitable approach for solving specific problems. For deeper research and sample questions, please get the Analysis of Algorithms book. You should definitely check out this book.
In short, in computer science, analysis of algorithms is the process of figuring out the computational complexity of algorithms - the amount of time, storage or other resources required to execute them. In this repository, the solutions of the problems given in the Analysis of Algorithms course will be shown. Problems will be solved through various programming languages. Please indicate if there are any missing, incorrect or excess parts.
Recurrences play a crucial role in the analysis of algorithms, particularly when analyzing the time complexity of algorithms that involve recursive or divide-and-conquer strategies. In simple terms, a recurrence relation describes the running time or resource usage of an algorithm in terms of its own previous iterations or subproblems.
When analyzing algorithms with recurrence relations, the goal is to determine an expression or formula that represents the relationship between the input size and the running time of the algorithm. This allows us to estimate the algorithm's efficiency and understand its behavior as the input size grows.
1 - Suppose that
Solution of the problem on Java:
public class Programme {
public static void main(String[] args) {
int tam = 1000, n;
float[] an = new float[tam];
an[0] = 0;
an[1] = 0;
an[2] = 0;
for (n = 3; n < tam; n++) {
an[n] = (((n - 3) * an[n - 1]) / n) + 1;
System.out.println(an[n]);
}
}
}
Console output:
250
Solution of the problem on Python 3.9:
import numpy as np
tam = 1000
n = 3
an=np.zeros(tam)
an[0] = 0
an[1] = 0
an[2] = 0
while n < tam:
an[n] = (((n - 3) * an[n - 1]) / n) + 1
print(an[n])
n+=1
print(f"Answer of the question : {an[999]}")
Console output:
Answer of the question : 250.0
Solution of the problem on MATLAB:
tam = 1000;
n = 2;
an = zeros(tam, 1);
an(1) = 0;
an(2) = 0;
an(3) = 0;
while n <= tam
an(n) = ((((n-3) * an(n-1))/ n) + 1);
n=n+1;
end
plot(an)
xlabel('n');
ylabel('an');
title('recurrences');
an(999)
Console output:
250
2 - Plot the periodic part of the solution to the recurrence
Solution of the problem on Java (it does not include plotting):
public class Programme {
public static void main(String[] args) {
int tam = 972, n;
float[] an = new float[tam];
an[0] = 1;
an[1] = 1;
an[2] = 1;
for (n=4; n < tam; n++) {
an[n] = (9 * an[n / 3]) + n;
}
for (n=0; n < tam; n++) {
System.out.println("a["+n+"] = " + an[n]);
}
}
}
Console output:
a[971] = 110231.0
Solution of the problem on Python 3.9:
import numpy as np
tam = 972
n = 4
an=np.zeros(tam)
an[0] = 1
an[1] = 1
an[2] = 1
while n < tam:
an[n] = (9 * an[n // 3]) + n
n+=1
tam = 972
n = 0
an_arr = []
while n < tam:
if n >= 0 and n <=972:
print(f"a[{n}] = ", an[n])
an_arr.append(an[n])
n+=1
import matplotlib.pyplot as plt
plt.plot(an_arr)
plt.grid()
plt.xlabel('n')
plt.ylabel('$a_n$')
plt.title('Recurrence Periodic')
plt.savefig('recurrence_periodic.png')
plt.show()
Console output:
a[971] = 110231.0
3 - Suppose that
Solution of the problem on Java:
public class Programme {
public static void main(String[] args) {
int tam = 100, n;
float[] an = new float[tam];
an[0] = 0;
for (n = 1; n < tam; n++) {
an[n] = an[n - 1] - ((2 * an[n - 1]) / n) + (2 * (1 - (2 * an[n - 1])/ n));
System.out.println(an[n]);
}
System.out.println("a[99] = " + an[99]);
}
}
Console output:
a[99] = 28.571438
Solution of the problem on Python 3.9:
import numpy as np
tam = 100
n = 1
an=np.zeros(tam)
an[0] = 0
while n < tam:
an[n] = an[n - 1] - ((2 * an[n - 1]) / n) + (2 * (1 - (2 * an[n - 1])/ n))
print(an[n])
n+=1
print(f"\nAnswer of the question : {an[99]}")
Console output:
Answer of the question : 28.57142857142856
Solution of the problem on MATLAB:
format long
tam = 100;
n = 2;
an = zeros(tam, 1);
an(1) = 0;
while n <= tam
an(n) = an(n - 1) - ((2 * an(n - 1)) / n) + (2 * (1 - (2 * an(n - 1))/ n))
n=n+1;
end
Console output:
2.885714285714284e+01
Generating functions are powerful mathematical tools used in the analysis of algorithms. They provide a systematic way to encode a sequence or set of values into a formal series representation. By manipulating and operating on these series, generating functions allow us to extract valuable information about the underlying sequence, such as its growth rate, coefficients, and other properties.
In the context of analyzing algorithms, generating functions are primarily used to study the behavior of sequences that arise from counting problems or recursive algorithms. They can help us understand the time complexity of algorithms, the number of solutions to a problem, or the distribution of certain parameters.
Suppose that
Solution of the problem on Java:
public class Programme {
public static void main(String[] args) {
int tam = 100, n;
float[] an = new float[tam];
float aux;
an[0] = 0;
an[1] = 1;
for (n = 2; n < tam; n++) {
an[n] = (9 * an[n - 1]) - (20 * an[n - 2]);
if (n>=2 && n <= 100) {
System.out.println("a["+n+"] = " + an[n]);
}
}
for (n = 2; n < tam; n++) {
aux=(float)an[n]/an[n-1];
if (n>=2 && n <= 100) {
System.out.println("aux["+n+"] = " + aux);
}
}
}
}
Console output:
aux[99] = 5.000000000318287 which is
Solution of the problem on Python 3.9:
import numpy as np
tam = 100
n = 2
an=np.zeros(tam)
an[0] = 0
an[1] = 1
while n < tam:
an[n] = (9 * an[n - 1]) - (20 * an[n - 2])
if n >= 2 and n <=100:
print(f"a[{n}] = ", an[n])
n+=1
print("\n ====== \n")
tam = 100
n = 2
while n < tam:
aux = an[n]/an[n - 1]
aux = float(aux)
if n >= 2 and n <= 100:
print(f"aux[{n}] = ", aux)
n+=1
Console output:
aux[99] = 5.000000000318287 which is
Solution of the problem on MATLAB:
format long
tam = 100;
n = 3;
an = zeros(tam, 1);
an(1) = 0;
an(2) = 1;
while n <= tam
an(n) = (9 * an(n-1)) - (20 * an(n-2))
n=n+1;
end
tam = 100;
n = 3;
while n <= tam
aux = an(n)/an(n-1)
n=n+1;
end
Console output:
aux[100] = 5.000000000318287 which is
In the context of algorithm analysis, "words" and "mappings" refer to concepts related to combinatorial enumeration and counting problems. They are used to analyze the complexity and properties of algorithms that deal with generating or counting certain structures or arrangements.
- Words: In the analysis of algorithms, a "word" typically refers to a sequence of symbols or elements chosen from a given set. The length of a word is the number of symbols it contains. Words are often used to represent various structures or combinations, such as strings, permutations, subsets, or graphs.
Analyzing words involves studying their properties, counting the number of possible words of a given length or satisfying certain constraints, or generating all possible words within a specific set of rules. Algorithms that deal with word generation or word-related operations often rely on combinatorial techniques and mathematical tools to understand their complexity and behavior.
- Mappings: "Mappings" in algorithm analysis refer to functions that establish relationships between elements of different sets. In the context of combinatorial enumeration, mappings are used to describe how elements from one set correspond to elements in another set.
Mappings are particularly relevant when analyzing algorithms that involve counting or generating structures with specific properties. For example, counting the number of mappings from one set to another may involve determining how many ways elements can be assigned or related in a specific manner.
By studying the properties and counting the number of possible mappings, algorithm analysts gain insights into the complexity of algorithms that deal with structural arrangements, assignment problems, or graph-related tasks.
How many people asked before finding two with the same birthday?
Solution of the problem on Java:
import java.lang.Math;
public class Programme {
public static void main(String[] args) {
int M = 365;
double result = Math.sqrt(Math.PI * M / 2);
System.out.println("Result of the problem: " + result);
}
}
Console output:
Result of the problem: 23.944532972687885
Solution of the problem on Python 3.9:
from math import sqrt, pi
M = 365
result = sqrt(pi * M / 2)
print(f"Result of the problem: {result}")
Console output:
Result of the problem: 23.944532972687885
Solution of the problem on MATLAB:
format long
M = 365;
result = sqrt(pi * M / 2)
Console output:
result = 23.94453297268788
Analysis of Algorithms is a vital discipline for understanding the efficiency and effectiveness of different algorithmic approaches. This project serves as a comprehensive guide, showcasing problem-solving techniques in Python, Java, and MATLAB, and offering a comparative analysis of their performance characteristics. By delving into this project's findings, developers and computer scientists can gain a deeper understanding of algorithmic analysis and make informed choices when it comes to implementing algorithms in their own projects.
By delving into this project's findings, developers and computer scientists can gain a deeper understanding of algorithmic analysis and make informed choices when it comes to implementing algorithms in their own projects.
If you have something to say to me please contact me:
- Twitter: Doguilmak
- Mail address: doguilmak@gmail.com