Skip to content

SAgarwal-24/CP

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Custom Comparator in priority_queue (In general)

Syntax

priority_queue<data_type, container, comparator> ds;

1. data_type (mandatory) 
2. container (optional) 
3. comparator (optional) 

The Characteristics of the heap data structure are:

- O(1) or constant time retrieval of min/max in the list.
- O(log N) time for insertion and deletion.
class Compare {
    public:
       bool operator()(T a, T b){
           if(cond){
               return true;
           }
           return false;
      }
};
  1. You should declare a class Compare and overload operator() function.
  2. When true is returned, it means the order is NOT correct and swapping of elements takes place.
  3. When false is returned, it means the order is correct and NO swapping of elements takes place.

Priorty_Queue_Custom_Comparator_User_Defined_Class

/*
How to make a min-heap of user-defined class? 
Let us consider below example where we build a min-heap of 2 D points ordered by X-axis.
*/

// C++ program to use priority_queue to implement Min Heap 
// for user defined class 
#include <bits/stdc++.h> 
using namespace std; 

// User defined class, Point 
class Point 
{ 
	int x; 
	int y; 
public: 
	Point(int x, int y) 
	{ 
		this->x = x; 
		this->y = y; 
	}

	int getX() const { return x; } 
	int getY() const { return y; } 
}; 

// To compare two points 
class myComparator 
{ 
public: 
	int operator() (const Point& p1, const Point& p2) 
	{ 
		return p1.getX() > p2.getX(); 
	} 
}; 

// Driver code 
int main () 
{ 
	// Creates a Min heap of points (order by x coordinate) 
	priority_queue <Point, vector<Point>, myComparator > pq; 

	// Insert points into the min heap 
	pq.push(Point(10, 2)); 
	pq.push(Point(2, 1)); 
	pq.push(Point(1, 5)); 

	// One by one extract items from min heap 
	while (pq.empty() == false) 
	{ 
		Point p = pq.top(); 
		cout << "(" << p.getX() << ", " << p.getY() << ")"; 
		cout << endl; 
		pq.pop(); 
	} 

	return 0; 
} 


What is Dijkstra's Algorithm

 Dijkstra's Algorithm is -> an algorithm to find the shortest path between two nodes in a graph or two cells in a matrix like our problem.
It's part of a family of algorithms designed to uncover the shortest path between nodes or cells like BFS, Bellman-Ford Algorithm, Floyd-Warsahll Algorithm. Each algorithm has its unique applications.

Let's focus on Dijkstra's Algorithm:
  Nature: It's a single-source, multiple-destination algorithm. In simpler terms, it pinpoints the shortest path from a starting node to any other node in our graph.

  Constraints: Dijkstra's Algorithm is suitable for graphs or problems featuring positive weights. If we encounter negative weights, we'd choose alternative algorithms such as Bellman-Ford.

https://leetcode.com/problems/path-with-minimum-effort/discuss/4049576/99.92-oror-Dijkstra's-Algorithm-oror-Binary-Search-oror-Commented-Code

DSU MY TEMPLATE

class tree {

    int num_components;
    int nodes;
    int *par, *rank; //rank -> is size of component

 public:
    tree(int n){
        this->nodes = n;
        num_components = n;

        par = new int[n+1];
        rank = new int[n+1];

        for(int i=1; i<=n; i++)
         par[i] = -1,  rank[i] = 1;
    }

    

    int findset(int u){

        if(par[u]<0)
            return u;

        return par[u] = findset(par[u]); //path compression
    }

    void union_set(int u, int v){

        u = findset(u);
        v = findset(v);

        if(u==v)return;

        if(rank[u]>rank[v]){ //merge smaller into bigger
            par[v] = u; 
            rank[u]+=rank[v];
            rank[v]=0;
        }
        else{
            par[u] = v;
            rank[v] += rank[u];
            rank[u] = 0;
        }
    }

    int getSize(int u){  //to find size of a component 

        return rank[findset(u)];
    }

     void print(){

        for(int i=1; i<=nodes; i++){
            cout<<par[i]<<" - "<<i<<endl;
        }

        cout<<"----\n";
    }

    void print_rank(){

        for(int i=1; i<=nodes; i++){
            cout<<rank[i]<<" ";
        }

        cout<<"\n----\n";
    }
};

(Concept of Modulo Multiplicative Inverse and Fermat little Theorem)

To calculate C(n,r) modulo 10^9+7

-> C(n,r) % M -> [n!/((n-r)! . (r!))] % M

we want something like below , but it is mathematically wrong -> (n! % M) / ([(n-r)! % M]. [(r!) % M] %M )

so we will do -> C(n,r) % M = [ numerator * inv(denominator) ] % M
-> C(n,r) % M = [ (numerator % M) * (inv(denominator) % M) ] % M

//Using fermat little theo -> inv(n) = power(n, mod-2); -> Find using bin expo

int pow(int n, int i){
    int a=n,ans=1;
    while(i>0){
        if(i&1){
            ans=(ans*1LL*a)%MOD;
        }
        a=(a*1LL*a)%MOD;
        i=i>>1;
    }
    return ans;
}

int inv(int n){
    return pow(n, MOD-2);
}
int nCr(int n, int r){
    if(r>n)return 0;
    else{
        int res = f[n];
        res = (res*1LL*inv(f[r]))%MOD;
        res = (res*1LL*inv(f[n-r]))%MOD;
        return res;
    }
}

alter for C(n,r) modulo MOD

const int MOD = 998244353;

int add(int x, int y)
{
    x += y;
    while(x >= MOD) x -= MOD;
    while(x < 0) x += MOD;
    return x;
}

 vector<vector<int>> C(n + 1);
    for(int i = 0; i <= n; i++)
    {
        C[i].resize(i + 1);
        C[i][0] = C[i][i] = 1;
        for(int j = 1; j < i; j++)
            C[i][j] = add(C[i - 1][j], C[i - 1][j - 1]);
    }
    

To calculate C(n,r) efficiently without modulo

int ncr(int n, int r){

    if(n<r)
        return 0;

    if(r>n-r)
        r = n-r;

    /*  if(r==0)                (can also sepreately handle  this case )
            return 1;           (notice if r==0 then we get ans=1 )
    */

    int nr=1, den=1; 

    while(r){

        nr *= n;
        den *= r;

        int g = gcd(nr,den);

        nr /= g;
        den /= g;

        n--;
        r--;
    }

  
    return nr/den;       // or simply->  return nr; 
}

General template for Binary Search on real values

      long double ans=0.0, l=0, r=10000000;
    
    //change no of iterations acc. to requirements
    for(int iterations=0;iterations<100; iterations++){  //Alter for-loop can be while-loop->  while(r-l>1e-4) 

        long double mid = l+(r-l)/2;
      
        if(check_logic(mid)){
            ans = mid;
            l = mid;
        }else{
            r = mid ;
        }  
    }
 
  cout<<setprecision(20)<<ans<<endl;

Print all permutations of a given string using backtracking:

(https://www.geeksforgeeks.org/write-a-c-program-to-print-all-permutations-of-a-given-string/)


void permute(string& a, int l, int r)
{
	// Base case
	if (l == r)
		cout << a << endl;
	else {
		// Permutations made
		for (int i = l; i <= r; i++) {

			// Swapping done
			swap(a[l], a[i]);

			// Recursion called
			permute(a, l + 1, r);

			// backtrack
			swap(a[l], a[i]);
		}
	}
}

// Driver Code
int main()
{
	string str = "ABC";
	int n = str.size();

	// Function call
	permute(str, 0, n - 1);
	return 0;
}