Skip to content

jgmmendes/Data-Structures

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

99 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Summary

  1. Data Structures
    1. Array
    2. Matrix
    3. List
      1. Singly Linked List
      2. Doubly Linked List
      3. Circular Doubly Linked List
    4. Stack
      1. Static Stack
      2. Dynamic Stack
    5. Queue
      1. Static Queue
      2. Dynamic Queue
    6. Graph
      1. Undirected Weighted Graph
        1. Adjacency Matrix Version
        2. Adjacency Linked List Version
  2. Helpers
    1. Exception Handler
    2. Node
    3. Vertex
  3. Build Tool Guide
    1. Creating
      1. New Data Structure
      2. New Helper
    2. Adding Dependency
      1. To Data Structure
      2. To Helper
      3. Get Dependencies Already Added
      4. Access Modifier
      5. .info File
    3. Testing
      1. Just one Data Structure Or Helper
      2. Test Suite
        1. test
        2. test-debug
        3. check
    4. Building
    5. Packing
  4. Remarks
    1. TYPE Specific Functions
    2. Write Code To Test Implementation
  5. Exercises
    1. Self-Organizing List
    2. ADT - Abstract Data Type
  6. Some used Versions

Data Structures

A Data Structure is a way of organizing and storing data in a computer system. It provides a systematic and efficient way of accessing and manipulating data. Data structures are designed to suit different types of data and various operations that need to be performed on that data.

Array

An Array is a fundamental Data Structure that allows storing a fixed-size sequence of elements of the same type. It provides a way to organize and access data in a contiguous block of memory. Each element in an array is identified by its index, which represents its position within the Array.

Click here to see its Spec.

Click here to see its implementation.

Matrix

A Matrix is a two-dimensional Array (like an Array of Arrays) organized in rows and columns. A Matrix is made up of rows and columns. The number of rows and columns in a Matrix determines its size or dimensions. For example, an "m x n" Matrix has "m" rows and "n" columns.

Click here to see its Spec.

Click here to see its implementation.

List

A List is a Data Structure that stores a collection of elements in a specific order. It is a fundamental data structure in computer science and is commonly used to organize and manipulate data.

Singly Linked List

A Singly Linked List is a Data Structure used to store a collection of elements. It consists of a sequence of nodes, where each node contains both the element and a reference (or pointer) to the next node in the List. The last node in the List has a reference to null, indicating the end of the List.

Click here to see its Spec.

Click here to see its implementation.

Doubly Linked List

A Doubly Linked List is a Data Structure that extends the functionality of a Singly Linked List by providing links or references to both the previous and next nodes in the List. Each node in a Doubly Linked List contains three components: the data or element to be stored, a reference to the previous node, and a reference to the next node.

Click here to see its Spec.

Click here to see its implementation.

Circular Doubly Linked List

A Circular Doubly Linked List is a variation of a Doubly Linked List where the last node in the List has a reference to the first node, creating a circular structure. This means that the next node of the last node points back to the first node, and the previous node of the first node points to the last node.

Click here to see its Spec.

Click here to see its implementation.

Stack

A Stack is a fundamental Data Structure in computer science that follows the LIFO (Last-In, First-Out) principle. It models a collection of elements with two primary operations: push and pop.

Static Stack

A Static Stack is a Stack Data Structure that is implemented using a fixed-size Array or a static memory allocation. Unlike a Dynamic Stack, where the size can grow or shrink dynamically as elements are pushed or popped, a Static Stack has a predetermined size that is allocated at the time of its creation.

Click here to see its Spec.

Click here to see its implementation.

Dynamic Stack

A Dynamic Stack is a Stack Data Structure that can grow or shrink in size dynamically as elements are pushed or popped. Unlike a Static Stack, which has a fixed size allocated at the time of creation, a Dynamic Stack can adjust its capacity based on the number of elements it currently holds.

Click here to see its Spec.

Click here to see its implementation.

Queue

A Queue is a Data Structure that follows the FIFO (First-In, First-Out) principle, where elements are added at the rear and removed from the front. It allows for efficient insertion and deletion operations.

Static Queue

A Static Queue, if referring to a fixed-size Queue, would be implemented using a fixed-size array or a static memory allocation. The capacity of the Static Queue would be predetermined and remain constant throughout its lifetime.

Click here to see its Spec.

Click here to see its implementation.

Dynamic Queue

A Dynamic Queue is a Queue Data Structure that can grow or shrink dynamically as elements are enqueued (added) or dequeued (removed). Unlike a Static Queue, which has a fixed size allocated at the time of creation, a Dynamic Queue can adjust its capacity based on the number of elements it currently holds.

Click here to see its Spec.

Click here to see its implementation.

Graph

A Graph is a Data Structure that represents a collection of interconnected nodes or vertices, often referred to as "points", and the connections between them, known as "edges." Graphs are used to model relationships between objects or entities.

Undirected Weighted Graph

An undirected weighted graph is a type of graph in graph theory that consists of a set of vertices and a set of edges.

The edges between vertices have no direction, which means they do not have a specific starting or ending point. If there's an edge between vertex A and vertex B, it implies that you can traverse it from A to B or from B to A with the same ease. This property reflects a symmetric relationship between connected vertices.

Each edge in the graph has an associated weight, which is a numerical value. These weights represent some measure or value associated with the relationship between the vertices connected by the edge.

Adjacency Matrix Version

Here Undirected Weighted Graph were implemented with an adjacency matrix and Array of Vertex, where the rows and columns correspond to the vertices of the graph, and the values in the matrix represent the weights of the edges between those vertices.

Click here to see its Spec.

Click here to see its implementation.

Adjacency Linked List Version

Helpers

In this context, Helpers means "more primitives Data Structures", that does not allow dependency from other complex Data Structure.

Exception Handler

The Exception Handler is a special "class" designed to help with exception handling in the development of Data Structures.

Click here to see its Spec.

Click here to see its implementation.

Node

A Node is an individual element or item in the data structure. Each Node contains both data and references to the previous and next nodes.

Click here to see its Spec.

Click here to see its implementation.

Vertex

A Vertex is a “point” on a Graph that represents an entity or element. Here it was implemented with data and valency property (which means how many connections the Vertex has).

Click here to see its Spec.

Click here to see its implementation.

Build Tool Guide

To abstract some repetitive steps in the process of implementing new Data Structures or Helpers, adding dependencies, running tests, compiling, building the library, among other things, a Build Tool was created using Makefile and Shell scripts.

Creating

Is possible to create a new Data Structure or a new Helper directly in CLI instead create each directory manually.

New Data Structure

To create a new Data Structure, in the root of the project you can run one of the commands below:

make create-ds
make create datastructure

and enter the name of new Data Structure.

New Helper

To create a new Helper, in the root of the project you can run one of the commands below:

make create-h
make create helper

and enter the name of new Helper.

Adding Dependency

You can also add local dependencies to a Data Structure or a Helper already created.

What does means local dependencies? It means that you can add only Data Structures and Helpers created in this project.

Note that, like Helpers does not allow dependency from complex Data Structure, you can add only other Helpers as its dependencies.

To Data Structure

If you want to add other Data Structure as dependency:

make install_ds

or

make install ds

If you want to add a Helper as dependency:

make install_helper

or

make install h

To Helper

If you want to add other Helper as dependency:

make install_helper

or

make install h

Get Dependencies Already added

To get headers of dependencies already added on a or Data Structure on a Helper, on its directory run:

make install 

Access Modifier

Each dependency has an access modifier that rule where it will be added during Adding a new dependency. There are 3 access modifiers: Public, Protected and Private.

  1. Public (pub): The dependency include will be on header of the parent, it means that everyone with access on header can see how this dependency is used.
  2. Protected (ptd): The dependency include will be on source code of parent, it means that only who can access source code is able to see how the dependency is used.
  3. Private (pvt): The dependency include will not be put anywhere, just on .info file. It means that it is not a direct dependency and will be used only during compile time.

.info File

The .info File is list of the Dependencies of a Data Structure or Helper during. It is used to verify if the dependency is already added during adding a new Dependency. The pattern of element of this list is: modifier_type_NameOfDependency, or modifier is the Access Modifier, type is about being a Data Structure (ds) or Helper (hp).

For example:

  1. Adding the Exception Handler as a public dependency: pub_hp_ExceptionHandler
  2. Adding the Undirected Weighted Graph version 1 as a protected dependency: ptd_ds_UndirectedWeightedGraph1
  3. Adding the Array as a private dependency: pvt_ds_Array

Testing

Just one Data Structure or Helper

To test any of Listed Data Structures or Helpers, you may navigate to their respective directory and run the following command in the terminal:

make run_tests

For example, if I want to run Unit Tests for Array, then:

cd main/Array && make run_tests

Test Suite

To maintain a high level of quality, there are a lot of unit test for each Data Structure and Helper. If one want to run all tests at once, there are 3 modes and one can choose which is better for each situation.

make test

It runs quicly and silent. During execution, it shows only which Data Structure or Helper are executing, time spent of executed Data Structure and Helper. At final of execution, it shows how many test was executed, how many tests passes or fail and total time of execution.

One can execute test suite in this mode by running the following command in the root of project:

make test

make test-debug

This mode is used to debug the test suite, and it runs slowly when compared with make test. It shows the same as when execute make test, in addiction it shows each test executed (description and result).

One can execute test suite in this mode by running the following command in the root of project:

make test-debug

make check

The third is a more restrictive mode and always runs before make build or make pack, ensuring that if any test fails the entire build or pack process is interrupted.

One can execute test suite in this mode by running the following command in the root of project:

make check

Building

The Build Command is used to generate and compile the Library headers .h and Static Lib libds.a when you run in the terminal:

make build

or

make b

it runs check command and if all tests passes, it will create some directories:

  • LIBDS
  • LIBDS/include
  • LIBDS/lib

The headers are in LIBDS/include and the Static Liblibds.a in LIBDS/lib.

Packing

The Pack Command goes one step further than the Build Command, as its life cycle ends after packing the contents of the LIDBDS directory as libds.tar.gz and deleting the LIDBDS directory. Since the Pack Command depends on the Build Command, when executed in the terminal:

make pack

or

make p

if any tests fail, the process is interrupted.

Remarks

TYPE specific functions

It is said at the beginning of every header of a Data Structure in this library, but it is worth mentioning here that, as the implementation followed a concept of generics, to use it with any data type, it is necessary to implement at least 3 functions:

  • TYPE printing function: To print data correctly.
    • like: void (*TYPE_print_function)(void *data)
    • example:
void int_print_function(void *data) {
    printf("%d", int_convert_function(data));
}
  • TYPE convert function: As some functions returns void*, one must use a function to convert void* to TYPE*.
    • like: TYPE (*TYPE_convert_function)(void *data)
    • example:
int int_convert_function(void *data) {
    return *((int *) data);
}
  • TYPE comparison function: To compare correctly data.
    • like: int (*TYPE_compare_function)(void *data1, void *data2)
    • example:
int int_compare_function(void *data1, void *data2) {
    int d1 = int_convert_function(data1), d2 = int_convert_function(data2);
    return d2 - d1;
}

Code to test implementation

You can write some code, just to test some features while developing (you may consider write unit tests instead), that implements the created Data Structures. Just write it in the file (app_X.c) inside the /main/apps directory and in its Data Structure directory run:

make run_apps

Exercises

Self-Organizing List

A Self-Organizing List is a Data Structure that reorganizes its elements based on the frequency of access or some other criteria in order to optimize future access patterns. It dynamically adjusts the order of elements to improve the efficiency of common operations, such as searching or accessing frequently used elements.

make run_apps
  • Can run tests in the terminal:
make run_tests

ADT - Abstract Data Type

ADT is a high-level description of a Data Structure that defines the behavior of a data type independently of its implementation details. ADTs provide a logical representation of data and operations that can be performed on that data, without specifying how those operations are implemented.

Some used versions

  • OS: Ubuntu 20.04.6 LTS (Focal Fossa)
  • GCC: 9.4.0
  • C: C17 (201710L)
  • Unity Framework: 2.5.2
  • Make: 4.2.1