college/Spring-2024/CS-2124/Lectures.org

482 lines
17 KiB
Org Mode

* Lecture 1
[2024-01-16 Tue]
** Recommended Books
- Data Structures Using C and C++, by Langsam, Augenstein, and Tenenbaum
- Data Structures and Algorithm Analysis in C, Mark Allen Wiss
** Office Hours
- Tuesday & Thursdays, 3pm - 4pm
** Attendance
- Not mandatory
- Missing will cause a loss of /bonus/ points
** Late Work
- Point reduction for late work
** Why Study Data Structures
*** Application
- Big Data
- Data is everything and it must be managed to extract information
- Applications, websites must be optimized (Data access)
*** Student
- Fundamental
- Develops thinking for programming
- Improves solving problems with better time complexities (performance)
- Many self-taught programmer lack fundamentals of Computer Science
- Popular technologies change, Data Structures or Analysis of Algorithms remain the same
*** Textbook Definition
- Refers to a scheme for organizing related pieces of information
- Basic types of data structures include:
- Files / lists
- Arrays / Records
- Trees / Tables
- Graphs
** Types of Data Structures
*** Structures & Unions
- Stuctures
Contains ordered group of data objects, each data object in a structure is a /member/ or a
/field/.
- Union
Similar to a structure except that all of its members start at the same location in memory. A
union variable can represent the value of only one of its members at a time.
*** Graphs and Trees
- Graphs and Trees are linked abstract data structures composed of nodes.
- Each node contains a value and one or more pointers to other nodes arranged in a hierarchy.
- Graphs can be used to represent networks, while variants of trees can be used for sorting and
searching.
*** Data Structure Operations
1. Traversal
2. Searching
3. Insertion
4. Deletion
5. Sorting
6. Merging
*** Planning a Computer Program
- As a programmer you are not supposed to start directly by coding
- The most important part of programming is brain storming on how to solve the problem
- First step can be on paper
- Technically we term such steps as *Pseudocode*
- Some programmers also use Algorithm to solve the issue on paper, then start programming
*** Algorithm Specification
- An *algorithm* is a finite set of instructions that, if followed, accomplishes a particular
task.
- All Algorithms must satisfy the following criteria:
1. Input
2. Output 3. Defi
5. Effectiveness
*** How Programs Solve Problems
- Program Flow Control
- The order in which program statements are executed
- Heuristics
- Some problems are very complex or no algorithm exist to solve some problems, at such
conditions programmers rely on heuristics
- Intrusion Detection Systems can rely on heuristics to identify attacks
- Heuristics are basically identified patterns or elements to assist in creating a solution to
some problem
* Lecture 2
[2024-01-23 Tue]
- Time and Space Complexity
- Intro to Asymptotic Notations
- Big O Notation
- Searching
- Binary Search
- Linear Search
** Time and Space Complexity
*** Time Complexity
- Time taken by an algorithm for execution
- Process of determining how processing time increases as the /size of the problem/ (input size)
increases
- Generally time complexity is expressed by keeping only the /values which affects runtime most/.
- *For example*, if time complexity for a pgrogram needs to be calculated as a function of (i.e.
$n^4 + n^3 + n^2 + n = n^4$) as all terms are small and they have a lesser impact on overall
computation time when compared with $n^4$
*** Space Complexity
- Memory required by an algorithm to execute a program
- Space complexity is the total amount of /memory space used/ by an algorithm/pgroam including
the space of input values for execution
*** Time and Space Complexity Importance
- Intrusion Detection Systems must process gigabits of data or more with minimal latency, time
complexity is important for this
- Handling potentially petabytes of data, memory complexity is important here
*** Data Structures
- Data Structures are necessary for designing efficient algorithms
- Provides reusability and abstraction
- Appropriate data structures helps programmers save time and space
- Assists in optimizing data manipulation (i.e. add, remove, edit large amounts of data)
- For example: A tree based data structure is ideal for college course info storage in a program
- Determining what courses need to be taken, courses have "dependencies" on other courses that
could be more than a single course dependency
*** Intro to Asymptotic Notations
- Tells us how good an algorithm is when compared with another algorithm
- Parameters can play a part
- i.e. hardware used for implementation, Operating System, CPU model, processor generation,
etc.
- Therefore, we use Asymptotic analysis to compare space and time complexity
- Analyzes two algorithms based on changes in their performance concerning the increment or
decrement in the input size
- *Big O* ($O()$) describes the upper bound of complexity
- Worst case scenario
- Runtime usually depends on the size of the input:
- $T(n)$: /the time taken on an input of size $n$.
- Asymptotic analysis considers the growth of $T(n)$
*** Big O Notation
- Worst case scenario (analyzes algorithm's upper bound)
- Best-case scenario is not considered for use in a comparative analysis
- That's why we employ worst-case scenarios to get meaningful input
- Algorithm in data structure while programming code is critical
- Big O makes it easier to compare algorithms
- *Big O* notation, $O(g(n))$ is a collection of functions
- A function $f(n)$ is a member of that collection only if it fits the following criteria:
- Constant $c$ and $n_0$ exist where $f(n) <= c.g(n)$ for all $n >= n_0$ Where '$c$'
represents constant values
- $O(f(n))$ describes the upper bound of $f(n)$, /worst-case scenario/
- $\omega(f(n))$ describes the lower bound of $f(n)$, /best-case scenario/
**** Why Do We Need Big O?
- World we live in today consists of complicated apps & software, each running various devices
and each has different capabilities
- Some devices like desktops can run heavy machine learning software, but others like phones can
only run apps
- WHen you create an application, you'll need to optimize your code so that it runs smoothly
across devices to give you an edge over your competitors
**** Computing Big O Notation
- *Big-O* Asymptotic Notation gives up Upper Bound:
1. Determine what the input is and what '$n$' represents (i.e. $f(n)=O(g(n))$)
2. Identify maximum number of operations the algorithm performs in terms of '$n$'. (i.e.
addition of two numbers is just 1 operation)
3. Eliminate all excluding the highest order terms. (i.e. if you have $n^4$ and $n^3$ consider
only $n^4$)
4. Remove all constant factors. Constants will remain /constant/ regardless of user input
- Basically *Big-O* is used to measure and compare worst-case scenarios of algorithms
**** Example Big O Notations
| Big O Notation | Example |
|-------------------------|--------------------------------|
| Constant: $O(c)$ | $O(1)$ |
| Logarithmic $O(log(n))$ | $n=20$ means $log(20) = 2.996$ |
| Linear: $O(n)$ | $n=20$ means $20$ |
| Quadratic: $O(n^2)$ | $n=20$ means $20^2 = 400$ |
| Exponential: $O(2^n)$ | $n=20$ means $2^20$ = 1084576 |
| Factorial: $O(n!)$ | $N=20$ means $20!$ |
**** Example Program
#+begin_src c
#include <stdio.h>
int main() {
int n;
printf("N = ");
scanf("%d", &n);
printf("Got: %d\n", n);
int a[n];
for (int i = 0; i < n; i++)
printf("a[%d] = %d \n", i, a[n]=i-1)
}
#+end_src
- Big O of this is $O(n)$
*** Searching
- Searching in data structures refers to the process of finding the location of an element in a
list
- One of the important parts of many data structures algorithms, as one operation can be
performed on an element if and only if we find it
- We do not want searching to take '$n$' steps for searching an array of '$n$' number of
elements
- In some cases we are bound to take '$n$' steps
- Different algorithms try to minimize the number of steps to search an element
**** Binary Search
- Divide and conquer approach
- Requires the data to be sorted
- In sequential search, when we compare against the first item, there are at most more items to
look through if the first item is not what we are looking for
- Instead of searching the list in sequence, a /binary search/ will start by examining the
middle term
- If that term is the one we are searching for, we are done
- If it is not the correct term, we can use the ordered nature of the list to eliminate half
of the remaining items
- If the term we are searching for is greater then the middle item, we know that the entire
lower half of the list as well as the middle item can be eliminated from further
consideration
- The term, if it is the list, must be in the upper half
***** Algorithm Steps
#+begin_src python
def binary_search(arr: list[int], term: int, low: int, high: int) -> int:
mid = (low + high)/2
while low <= high:
if (low > high):
raise("Unable to find the search term!")
else if (arr[mid] < term):
low = mid + 1
else if (arr[mid] == term):
return mid
else
high = mid - 1;
mid = (low + high) / 2
binary_search([0, 1, 2, 3, 4], 0, 0, 5) # Outputs: 5
#+end_src
**** Linear Search
- The Linear Search (sequential search) algorithm starts at one end of a list and goes through
each element of a list until the desired element is found, otherwise the search continues till the
end of the data set
- Does not require data to be sorted
- Poor *Big-O* complexity: $O(n)$
#+begin_src python
def linear_search(arr: list[int], term: int) -> int:
for i in range(0, len(arr)):
if arr[i] == term:
return i
raise(f"Unable to find {term} in array!")
#+end_src
* Lecture 2
** Big O Notation
*** $O(log(n))$
- Divide and conquer
- If the base is not specified in CS, assume a base of $2$: $O(log_2(n))$
- Binary search is an algorithm that is of $O(log(n))$ complexity
** Sorting Algorithms
- Sorting refers to arranging data in a particular format
- Many search algorithms depend on sorted data, hence /sorting/
- In general there are *2 approaches* to sort an array of elements:
1. Some algorithms work by moving elements to their final position, one at a time. You sort an
array of size N, put 1 item in place, and continue sorting an array of size N - 1.
- Memory efficient
- Performance inefficient
2. Some algorithms put items into a temporary position, close(r) to their final position. YOu
rescan, moving items closer to the final position with each iteration.
- Memory inefficient
- Performance efficient
*** Complexity and Running Time
- Factors:
1. Algorithmic complexity
2. Additional space requirements
3. Use of recursion
- Have to be careful with recursion, can easily spiral out into an $b^n$ complexity
4. Worst-case behavior
- Worst-case behavior is important for real-time systems that need guaranteed performance
5. Behavior on already-sorted or nearly-sorted data
*** Stable vs Unstable Sorting
- A stable sort is one which preserves the original order of the input set
- Elements of same value will be in order
- An unstable sort does not preserve order of elements
- Ordering will be only on the sorted value, original order is not respected
*** In-place and Out-of-place Sorting
- An *In-place algorithm* modifies the inputs, which can be a list or an array, without using
any additional memory As the algorithm runs, the input is usally overwritten by the output, so no
additional space is required.
- In-place algorithms may take some memory, like using some variables for its operation
- Overall, it takes constant memory. Space complexity of $O(1)$
- An algorithm that is not in place is called a *not-place* or *out-of-place* algorithm. These
sorting algorithms use =extra space= for sorting, which depends upon the size of the input
*** Bubble Sort
- *Bubble Sort* works by repeatedly swapping adjacent elements if they are in the wrong order
- Not suitable for large data sets as its average and worst-case complexity is quite high
- Bubble sort is an In-place and Stable sorting algorithm
- Big O's:
- Time Complexity: $O(n^2)$
- Space Complexity: $O(n^2)$
- Steps:
1. Walk through the array n-times
2. As you walk through the array, check if the current element and it's next neighbor are out
of order
3. If they are out of order, swap them
*** Selection Sort
- *Selection Sort* is an *in-place* algorithm in which the list is divided into two parts
1. The sorted part at the left end
2. The unsorted part at the right end
- The smallest element is selected from the unsorted array and swapped with the leftmost
element, and that element becoems a part of the sorted array. This process continues moving
unsorted array boundary by one element to the right.
- Selection sort is generally preferred over Bubble Sort
- Big O's:
- Time Complexity: $O(n^2)$
- Space Complexity: $O(1)$
- Steps:
1. Set ~MIN~ to location $0$
2. Search the minimum element in the list
3. Swap with value at location ~MIN~
4. Increment ~MIN~ to point to next element
5. Repeat until list is sorted
*** Insertion Sort
# TODO: Finish this section out
- Insertion sort is a simple sorting algorithm that works similar to the way you sort playing
cards in your hands
- The array is virtually split into a sorted and an unsorted part
- Values from the unsorted part are picked and placed at the correct position in the sorted part
- Steps:
1. If it is the first element, it is already sorted. return 1;
2. Pick next element
3. Compare with all elements in the sorted sub-list
4. Shift all elements in the sorted sub-list that is greater than the value to be sorted
*** Algorithm Comparison
| Bubble Sort | Selection Sort | Insertion Sort |
|-------------------------------|--------------------------------------------------------|---------------------------------------|
| Simple Sorting Algorithm | Simple Sorting Algorithm | Simple Sorting Algorithm |
| Compares Neighboring Elements | Takes the smallest element and moves it into its place | Transfer one element at a time to its |
*** Merge Sort
- Divide and Conquer Algorithm
- Works by dividing an array into smaller subarrays, sorting each subarray, and then merging the
sorted subarrays back together
- Does not check if the data is already sorted.
- Steps:
1. Find middle index of the array: ~Middle = 1 + (last-first)/2~
2. Divide the array from the middle
3. Call merge sort for the first half of the array: ~MergeSort(array, first, middle)~
4. Call merge sort for the second half of the array: ~MergeSort(array, middle + 1, last)~
5. Merge the two sorted halves into a single sorted array
- Big O's:
- Time Complexity: $nlog(n)$
- Space Complexity: $O(n)$
*** Quick Sort
- Divide and Conquer Algorithm
- Picks an element as a pivot and partitions the given array around the picked pivot
- There are many different versions of Quick Sort that pick pivot in different ways:
1. Always pick the first element as a pivot
2. Always pick the last element as a pivot
3. Pick a random elemt as a pivot
4. Pick median as the pivot
- Big O's:
- Time Complexity: $O(n^2)$, average case is $O(nlog(n))$
- Steps:
1. Pick an element from the array as the pivot
2. Divide the unosrted array of elements in two arrays
a) Values less than the pivot come in the first sub array
b) Values greater than the pivot come in the second sub-array
3. Recursively repeat step ~2~ (until the sub-arrays are sorted)
* Lecture 4
** Methods for Reducing Complexity
- Goes along with the *Simplicity* Design Principle
- They are:
1. Abstraction
2. Modularity
3. Layering
4. Hierarchy
** Stack
- What is a stack
- A data structure.
- New items are inserted on the "top" and deleted or removed from the top. Top is the most
recent item inserted.
- A stack is a Last In First Out (LIFO) or First In Last Out (FILO) data structure. This means
that the last item to get stored on the stack (often called Push operation) is the first one
to get out of it (often called as Pop operation).
** Stack Operations
1. *push*: Adds an element to the top of the stack
2. *pop*: Removes the topmost element from the stack
3. *isEmpty*: Checks whether the stack is empty
4. *isFull*: checks whether the stack is full
5. *top*: Displays the topmost element of the stack
6. *Peek*: View the top element on the stack
- *Important* things to remember when wokring with stacks:
- Initially, a pointer (top) is set to keep track of the topmost itemi n the stack. The stack
is initialized to -1
- Then, a check is performed to determine if the stack is empty by comparing top to -1
- As elements are added to the stack, the position of the top is updated
- As soon as elements are popped or deleted, the topmost element is removed and the position of
top is updated
- There are *two ways to implement* a stack:
1. Using a =array=
2. Using a =linked list=