kaiosilveira / hacker-rank-challenges

An aggregator of my completed code challenges in Hacker Rank, containing detailed explanation, benchmarking, time complexity analysis, and thorough testing

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

Continuous Integration

Problem-solving

This repository is an aggregator of my solutions to code challenges. Inside each solution you will find:

  • a main file for the challenge, which contains the samples and the challenge execution (index.rb)
  • the challenge wrapper, with constraint validations (challenge.rb + challenge.spec.rb)
  • the underlying algorithm used to solve the challenge (algorithm.rb + algorithm.spec.rb)
  • a benchmark of the results (benchmarking.rb + some CSVs with benchmark output)
  • a README.md file with a detailed explanation of the code implemented

Programming language choice πŸ‘¨πŸ½β€πŸ’»

Because of its simplicity, elegance and API completeness, Ruby was the chosen programming language to implement the code for the challenges.

Available tasks βš™οΈ

  • rake test: run all unit tests from all files
  • rake create_challenge'[challenge_name]': create a new directory with the default structure for a challenge

Testing & Continuous integration πŸ§ͺ

Each challenge contains a unit test suite covering the constraints described by the challenge itself, at least one happy path, and sometimes some interesting particular cases.

The built-in test/unit test framework was used to implement the test suites, as no external dependency will be needed.

Continuous integration

A simple CI pipeline was put in place to execute the tests for each directory inside this repo. It sets up Ruby and runs the tests. This happens for every push to main. The config is:

name: Continuous Integration

on:
  push:
    branches: ["main"]

jobs:
  test:
    runs-on: ubuntu-latest
    strategy:
      matrix:
        ruby-version: ["2.6"]

    steps:
      - uses: actions/checkout@v3
      - name: Set up Ruby
        uses: ruby/setup-ruby@v1
        with:
          ruby-version: ${{ matrix.ruby-version }}
          bundler: "Gemfile.lock"
          bundler-cache: true
      - name: Lock bundle
        run: bundle lock --add-platform x86_64-linux

      - name: Run tests
        run: bundle exec rake

See ruby.yml for the actual file.

Benchmarking ⏰

To benchmark the code execution, the benchmark gem was used. A benchmarking.rb module was created to abstract the setup for the benchmarking, allowing the client code to call it passing a reference to the function to be executed and some more configuration parameters. The Benchmarking module also aggregates the value of n and the time it took to run the function for each execution and gives it back to the client code, so it can store the results and perform the graphical analysis later. See below an example of the benchmark in action:

➜ ruby ./left-rotation/benchmarking.rb
n = 0  0.000011   0.000003   0.000014 (0.000011)
n = 1  0.000003   0.000001   0.000004 (0.000003)
n = 2  0.000004   0.000001   0.000005 (0.000005)
n = 3  0.000003   0.000000   0.000003 (0.000004)
n = 4  0.000003   0.000000   0.000003 (0.000003)
n = 5  0.000003   0.000001   0.000004 (0.000003)
n = 6  0.000005   0.000001   0.000006 (0.000005)
n = 7  0.000003   0.000000   0.000003 (0.000004)
n = 8  0.000004   0.000002   0.000006 (0.000004)
n = 9  0.000003   0.000001   0.000004 (0.000003)
n = 10  0.000005   0.000001   0.000006 (0.000005)

Time complexity πŸ“ˆ

The time complexity is calculated for each challenge, using the Big O notation. Common time complexities are:

  • Constant: $O(1)$
  • Linear: $O(n)$
  • Quadratic: $O(n^2)$

The experiment to calculate the time complexity for each challenge is an exercise of using all possible values described in the Constraints section of the challenge at Hacker Rank, always with an eye on the term that varies the most and could cause the biggest impact on the code execution. The analysis is performed on the dump generated by the benchmark process described above.

For each challenge, a chart was created to allow us to visually identify the time complexity for the solution. The YouPlot gem was used to create the charts from the command line. See below some examples of generated charts:

  • Constant time complexity $O(1)$:
➜ cat ./cat-and-mouse/results.csv | uplot line -d, -w 50 -h 15 -t Results --canvas ascii --xlabel n --ylabel "T(n)"
                                   Results
             β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
        0.02 β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
   T(n)      β”‚                                   |              β”‚
             β”‚                          .        |              β”‚
             β”‚                          |        |              β”‚
             β”‚                          |    .   |              β”‚
             β”‚                          |    |   |              β”‚
             β”‚ .        ,.              |    |   |              β”‚
             β”‚ | .    , ||              |    |   |              β”‚
           0 β”‚@@_11___]_L1__1___________]____]___L_________]____β”‚
             β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
             0                                            1000000
                                      n
  • Linear time complexity $O(n)$:
➜ cat ./left-rotation/results.csv | uplot line -d, -w 50 -h 15 -t Results --canvas ascii --xlabel n --ylabel "T(n)"
                                   Results
             β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
        0.02 β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                  β”‚
             β”‚                                                , β”‚
             β”‚                                                /\β”‚
             β”‚                                               . |β”‚
   T(n)      β”‚                                         .r./""`  β”‚
             β”‚                                     __/"`        β”‚
             β”‚                            ._.--"`-/             β”‚
             β”‚                        _.--`                     β”‚
             β”‚                   r/"""                          β”‚
             β”‚             .r""/"                               β”‚
             β”‚     _.__--""`                                    β”‚
           0 β”‚__--" `                                           β”‚
             β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
             0                                             100000
                                      n
  • Quadratic time complexity $O(n^{2})$:
➜ cat ./electronic-shops/results.csv | uplot line -d, -w 50 -h 15 -t Results --canvas ascii --xlabel n --ylabel "T(n)"
                                Results
          β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
        5 β”‚                                            ./    β”‚
          β”‚                                          .r`     β”‚
          β”‚                                         .`       β”‚
          β”‚                                       ./         β”‚
          β”‚                                     ./           β”‚
          β”‚                                   ./`            β”‚
          β”‚                                 ./`              β”‚
   T(n)   β”‚                              .r"                 β”‚
          β”‚                            ./`                   β”‚
          β”‚                          ./`                     β”‚
          β”‚                       _-"                        β”‚
          β”‚                    _-"                           β”‚
          β”‚                _r/"                              β”‚
          β”‚          ._r-/"                                  β”‚
        0 β”‚_____.--/"`                                       β”‚
          β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
          0                                              10000

About

An aggregator of my completed code challenges in Hacker Rank, containing detailed explanation, benchmarking, time complexity analysis, and thorough testing


Languages

Language:Ruby 100.0%