Getting started with competitive programming
What this post is about
This post assumes that
- you are familiar with Codeforces and competitive programming
- you know the basics of a particular programming language (C++)
This post is
- more about getting a good setup (according to me)
- about setting and using a debug template
- some resources to practice CP.
Selecting the right editor
There are a few good choices for selecting an editor for CP. Here are some:
- CP editor
- Sublime Text
- VS Code
I’ll go over the best setup according to me, which is the CP editor with a few other extensions.
- First, go and download CP editor. If using linux, an AppImage would be fine. Just download the AppImage and give execute permissions.
- Download cf tool. Extract the executable and add it to a location which is in your PATH. As this is a command line tool, it should be in your PATH.
- If on a linux machine, you could
cd
into the folder where you extracted the cf tool and on the terminal type:sudo cp cf /usr/bin/
. This would copy the cf tool to/usr/bin/
. Note: You can use any other location which is in your$PATH
. - If using windows, have a look at: Add to the PATH on Windows 10.
- If on a linux machine, you could
Now, we need to configure the cf tool, which is where most of the trouble comes. So open the terminal and type cf
. If you are unable to see the usage instructions, then it means that your cf
isn’t in the PATH. Go above and and do the steps correctly.
- Now type
cf config
. Option0
for login (as we are using it for the first time). - After successfully logging in, we will configure our template, so that the tool gets ready for submitting to codeforces. So go ahead and type
cf config
option1
. Now select the language code. - Now enter the absolute path to your template (Look at the template section and come back, If you don’t know what this is).
The suffix of template above will be added by default.
- leave emptyTemplate's alias (e.g. "cpp" "py"):
- In my case I used cppBefore script
- leave emptyScript
- just add the path to your default template as empty isn’t allowed.After script
- leave emptyMake it default (y/n)?
-y
- Add competitive companion to your browser.
- Go to the options of the extension and add
10045
toCustom ports
. This is the port whichCP editor
will listen to for requests. - This extension is able to parse the testcases and send them to a port for a large variety of competitive programming websites.
- Go to the options of the extension and add
With the setup almost done, just open CP editor
and check if the Messages
section is clean. If there is some error, it means that some tool isn’t configured correctly. If there is an issue with the compiler, then make sure that g++
is in your $PATH
. Open the preferences
tab and edit your preferences. (preference help can be found on their site too)
To test if everything works well, go to any problem on codeforces, and click on the green + icon of your competitive companion extension. Now after opening the CP editor
, you should be able to see a new file loaded with the default template and the sample testcases of the problem. Write the code and press compile and run
to see the automatic checking of testcases. You can even edit/add testcases and change the type of checker. After you feel that all cases work out, just hit the submit button, to directly submit the problem from within the editor without leaving it!! If you get any error in messages
, that means that cf tool
hasn’t been configured correctly.
Making your default template
To speed up the coding, a default template would be nice to have. Here is my default c++
template (as of now. I do make changes to it):
Template
#pragma GCC target ("avx2")
#pragma GCC optimization ("O3")
#pragma GCC optimization ("unroll-loops")
#include "bits/stdc++.h"
/*#include <ext/pb_ds/assoc_container.hpp>
#include <ext/pb_ds/tree_policy.hpp>*/
using namespace std;
//using namespace __gnu_pbds;
typedef long long ll;
typedef long double ld;
typedef pair<ll,ll> pll;
typedef vector<bool> vb;
typedef vector<int> vi;
typedef vector<ll> vll;
typedef vector<vi> vvi;
typedef vector<vb> vvb;
typedef vector<vll> vvll;
typedef vector<pll> vpll;
typedef vector<string> vs;
typedef unordered_map<ll,ll> umll;
template<class T> using pq = priority_queue<T, vector<T>, greater<T>>;
/*template <typename num_t>
using ordered_set = tree<num_t, null_type, less<num_t>, rb_tree_tag, tree_order_statistics_node_update>;*/
// find_by_order(k): iterator to the kth largest(0 indexed). order_of_key(k): no. of items < k
#define GET_MACRO(_1,_2,_3,_4,NAME,...) NAME
#define FOR3(i,a,b) for(long long i=a;i<b;i++)
#define FOR4(i,a,b,step) for(long long i=a;i<b;i=i+step)
#define REV3(i,a,b) for(long long i=a;i>=b;i--)
#define REV4(i,a,b,step) for(long long i=a;i>=b;i=i-step)
#define FOR(...) GET_MACRO(__VA_ARGS__, FOR4, FOR3)(__VA_ARGS__)
#define REV(...) GET_MACRO(__VA_ARGS__, REV4, REV3)(__VA_ARGS__)
#define F first
#define S second
#define pb push_back
#define ub upper_bound
#define lb lower_bound
#define all(v) v.begin(),v.end()
#define rall(v) v.rbegin(),v.rend()
#define tc ll tests;cin>>tests;while(tests--)
#define io ios_base::sync_with_stdio(false);cin.tie(nullptr);
#define coutv(v) for(auto it: (v))cout<<it<<" ";newl;
#define cout2d(v) for(auto it: (v)) {for(auto j:it) cout<<j<<" ";newl;}
#define cinv(v,n) vll (v)(n);FOR(i,0,(n)){cin>>v[i];}
#define cinvg(v,n) (v).resize(n);FOR(i,0,(n)){cin>>v[i];}
#define cin2d(v,n,m) vvll (v)(n,vll(m,0));FOR(i,0,n){FOR(j,0,m){cin>>v[i][j];}}
#define cin2dg(v,n,m) (v).resize(n,vll(m));FOR(i,0,n){FOR(j,0,m){cin>>v[i][j];}}
#define pyes(CONDITION) cout << (CONDITION ? "YES" : "NO") << '\n';
#define newl cout<<"\n"
#define MOD 1000000007
#define INF LLONG_MAX/2
#define m1(x) template<class T, class... U> void x(T&& a, U&&... b)
#define m2(x) (int[]){(x forward<U>(b),0)...}
m1(pr) { cout << forward<T>(a); m2(cout << " " <<); cout << "\n"; }
m1(re) { cin >> forward<T>(a); m2(cin >>); }
template <class ...Args>
auto &readd(Args &...args) { return (cin >> ... >> args); }
#define re(...) __VA_ARGS__; readd(__VA_ARGS__)
const ll dx[4] = {0,1,0,-1}, dy[4] = {1,0,-1,0};
// const ll dx[8] = {-2,-1,1,2,2,1,-1,-2}, dy[8] = {1,2,2,1,-1,-2,-2,-1}; //knight moves
// ************************DEBUG START********************************
#ifndef ONLINE_JUDGE
//#define cerr cout
#include "myprettyprint.hpp"
#else
#define dbg(...)
#endif
// ************************DEBUG END**********************************
constexpr ll pct(ll x) { return __builtin_popcountll(x); } // # of bits set
constexpr ll bits(ll x) {return x == 0LL ? 0LL : 63LL-__builtin_clzll(x); } // floor(log2(x))
constexpr ll p2(ll x) { return 1LL<<x; }
constexpr ll msk2(ll x) { return p2(x)-1LL; }
mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
void test()
{
}
int main()
{
io;
ll tests=1;
cin>>tests;
while(tests--)
{
test();
}
return 0;
}
Lets have a look at how to use the template and some features:
typedefs
are used to define a new type from existing data types. Have a look at this GFG article for a better understanding.#define
defines aliases to certain things.io
:ios_base::sync_with_stdio(false);cin.tie(nullptr);
for fast input output.GCC pragmas
: these are compiler specific, and may optimise certain solutions. There is no guarantee, but it may even work.#ifndef ONLINE_JUDGE //#define cerr cout #include "myprettyprint.hpp" #else #define dbg(...) #endif
By default the ONLINE_JUDGE constant is defined when submitting code in most online judges ex: codeforces or codechef. It helps the code to determine whether the code is being run on an online judge or on a local system machine.
So how can we take advantage of this?
With a#ifndef
we can have various things like custom header files and other definitions locally.#ifndef
means ‘if not defined’.
Now anelse
would mean when it is defined (so on the judge). In this case, we can simply set our custom definitions to "" (empty string), so that they are ignored in the code which the online judge compiles. The#ifndef
directive ends withendif
.Now lets have a look at how to use the template :
Loops
// normal code: // using template: for(long long i=0;i<n;i++) {} FOR(i,0,n) {} for(long long i=0;i<n;i+=5) {} FOR(i,0,n,5) {} // we can use different parameters!! for(long long i=n;i>=0;i--) {} REV(i,n,0) {} for(long long i=n-2;i>=1;i-=2) {} REV(i,n-2,1,2) {}
Inputs
// normal code: // using template: ll i, j, k; ll re(i,j,k); cin>>i>>j>>k; // accepts multiple parameters and takes input of the type specified, declaring the variables locally. // If you have different datatypes/have globally declared variables, use: ll n; string s; main() { re(n,s); } // input vectors: cinv(a, n) // note: semi-colon not needed cinvg(a, n) // a is already globally declared cin2d(g, n, m) // 2d vector of size n*m cin2dg(g, n, m) // 2d global vector of size n*m newl; // newline pyes(boolean) // print YES if true, else NO coutv(a, n) // print vector contents space separated and endline cout2d(g, n, m) // print 2d vector
STL
// normal code: // using template: p.first p.F p.second p.S sort(a.begin(), a.end()); sort(all(a)); v.push_back(5); v.pb(5); auto it=upper_bound(a.begin(),a.end(),7) auto it=ub(all(a),5) v.rbegin(),v.rend() rall(v) priority_queue<ll,vector<ll>,greater<ll>> pq<ll>
myprettyprint.hpp
myprettyprint.hpp: Read the debugging section to know more!!
There are several other possibilites that you can add too.
Debugging
This is a very important part in programming. Everyone faces bugs and we should know how to find them quickly.
Here are a few ways:
- Writing print statements to print the value of variables.
- Using a debugger and debugging using breakpoints.
Both of the above methods have shortcomings.The first one is very slow and while submitting any code, we need to remove the print statement. The second one is not easy for a beginner and requires some other setup and knowledge. So, we need to find something better and smarter. We use our own debugging library.
Firstly, one should use good compiler flags to compile the code, so that several things like - Integer Overflow
, Array out of bounds
and several other errors can be caught and pointed out by the compiler.
Here is the compile command which catches several errors and is really useful:
g++ -std=c++1z -D_GLIBCXX_DEBUG -D_GLIBCXX_DEBUG_PEDANTIC -g -fsanitize=undefined
(if you want to know more: link)
You can add this to your c++ compile command in the preferences of CP editor.
Now, coming to the main part. The debugging template:
myprettyprint.hpp
template <class T1, class T2>
ostream &operator<<(ostream &os, const pair<T1, T2> &p) {
return os << '{' << p.first << ", " << p.second << '}';
}
template <class T, class = decay_t<decltype(*begin(declval<T>()))>,
class = enable_if_t<!is_same<T, string>::value>>
ostream &operator<<(ostream &os, const T &c) {
os << '[';
for (auto it = c.begin(); it != c.end(); ++it)
os << &", "[2 * (it == c.begin())] << *it;
return os << ']';
}
//support up to 5 args
#define _NTH_ARG(_1, _2, _3, _4, _5, _6, N, ...) N
#define _FE_0(_CALL, ...)
#define _FE_1(_CALL, x) _CALL(x)
#define _FE_2(_CALL, x, ...) _CALL(x) _FE_1(_CALL, __VA_ARGS__)
#define _FE_3(_CALL, x, ...) _CALL(x) _FE_2(_CALL, __VA_ARGS__)
#define _FE_4(_CALL, x, ...) _CALL(x) _FE_3(_CALL, __VA_ARGS__)
#define _FE_5(_CALL, x, ...) _CALL(x) _FE_4(_CALL, __VA_ARGS__)
#define FOR_EACH_MACRO(MACRO, ...) \
_NTH_ARG(dummy, ##__VA_ARGS__, _FE_5, _FE_4, _FE_3, _FE_2, _FE_1, _FE_0) \
(MACRO, ##__VA_ARGS__)
//Change output format here
#define out(x) #x " = " << x << "; "
#define dbg(...) \
cerr << "Line " << __LINE__ << ": " FOR_EACH_MACRO(out, __VA_ARGS__) << "\n"
This might seem scary to look at, but there’s no need to get scared as we only need to know how to use it. In fact, we can smartly put this in a header myprettyprint.hpp
and simply include it using ifndef ONLINE_JUDGE
, as I have done in my template.
For the include to work, myprettyprint.hpp
should be placed in a directory so that it is in the search path when compiling.
on linux, you can place it here: /usr/include/c++/9/myprettyprint.hpp
.
If you don’t know where the compiler looks for header files, then see: gcc Search Path.
How to use:
debug examples
ll my_func(ll a, ll b)
{
return a*b;
}
void test()
{
int x=5, y=4;
string s="Hello";
vector<int> v={1,2,3,4,5};
set<int> st={34,45};
map<int, vector<int> > m;
for(int i=0;i<3;i++)
{
for(int j=0;j<4;j++)
{
m[i].push_back(i+j);
}
}
dbg(x, y, s, v, st);
dbg(m, my_func(3,7));
}
output
Line 43: x = 5; y = 4; s = Hello; v = [1, 2, 3, 4, 5]; st = [34, 45];
Line 44: m = [{0, [0, 1, 2, 3]}, {1, [1, 2, 3, 4]}, {2, [2, 3, 4, 5]}]; my_func(3,7) = 21;
So this debug template is able to print out a variety of stl containers
including the ones in pbds
(policy based data structures) too.
So while submitting the problem, we need not even remove the dbg()
statements, as they get automatically converted to empty strings. (else dbg(...)
)
If you are printing your output to a terminal which supports ANSI based colours
, you could even use this debug template:
colour debug template
// colout print
template <class T1, class T2>
ostream &operator<<(ostream &os, const pair<T1, T2> &p) {
return os << '{' << p.first << ", " << p.second << '}';
}
template <class T, class = decltype(begin(declval<T>())),
class = enable_if_t<!is_same<T, string>::value>>
ostream &operator<<(ostream &os, const T &c) {
os << '[';
for (auto it = begin(c); it != end(c); ++it)
os << (it == begin(c) ? "" : ", ") << *it;
return os << ']';
}
#define _NTH_ARG(_1, _2, _3, _4, _5, _6, N, ...) N
#define _FE_1(_CALL, x) _CALL(x)
#define _FE_2(_CALL, x, ...) _CALL(x) _FE_1(_CALL, __VA_ARGS__)
#define _FE_3(_CALL, x, ...) _CALL(x) _FE_2(_CALL, __VA_ARGS__)
#define _FE_4(_CALL, x, ...) _CALL(x) _FE_3(_CALL, __VA_ARGS__)
#define _FE_5(_CALL, x, ...) _CALL(x) _FE_4(_CALL, __VA_ARGS__)
#define _FE_6(_CALL, x, ...) _CALL(x) _FE_5(_CALL, __VA_ARGS__)
#define FOR_EACH_MACRO(MACRO, ...) \
_NTH_ARG(__VA_ARGS__, _FE_6, _FE_5, _FE_4, _FE_3, _FE_2, _FE_1) \
(MACRO, __VA_ARGS__)
#define watch(x) cerr << "\033[1;32m" #x " = \033[1;34m" << (x) << "\033[0m; ";
#define dbg(...) \
cerr << "\033[2;31mLine " << __LINE__ << ": \033[0;m"; \
FOR_EACH_MACRO(watch, __VA_ARGS__) \
cerr << "\n"
output
With all this, debgging should become very easy and quick now. Just print out whatever you want.
Precompiling header files
This section is mostly for those who face long compile times (>2s-3s)
.
One way to reduce compile time is to precompile the header files using the same compile command that you use to compile your code.
- navigate to the header file
bits/stdc++.h
on your terminal (/usr/include/x86_64-linux-gnu/c++/9/bits
on ubuntu) and type:sudo g++ -std=c++1z -D_GLIBCXX_DEBUG -D_GLIBCXX_DEBUG_PEDANTIC -g -fsanitize=undefined stdc++.h
- You could also copy the above file and place it in the directory having your code. Precompiling here will also generate the
.gch
file. All you need to change in your code is#include <bits/stdc++.h>
to#include "<bits/stdc++.h>"
. This will tell the compiler to search for a local header in thebits
subdirectory and namestdc++.h
first, and then the default search path. When the compiler finds a.gch
file, it uses it instead of the.h
file.
CP_files
│
│ probA.cpp
│
└───bits
│ stdc++.h
| stdc++.gch
This should reduce your compile time a little.
Resources
PRACTICE PRACTICE PRACTICE. The only thing that can help you get better at CP.
Here are some resources which are best for practice and learning. Note that these are just some good ones according to me. There are several others which you can easily find, but the point to remember is to focus on actual practice and not to get overwhelmed by the mammoth content out there.
Practice and Learning
- USACO guide : This is one of the finest resources written by topcoders. It has everything. Right from guidance to practice problems.
- hackerearth : this site contains good articles and practice problems.
- cp-algorithms : one of the best references for theory and implementation of a large variety of algorithms in c++.
- codeforces : One of the best sites to practice and improve. It contains severals problems and contests. Just sort the problems based on a particular rating (typically +200-300 of your current) and start practicing. You can also host mashups and practice.
- codechef : This site has good problems and contest too. The DSA learning series is also amazing to learn a particular topic.
- atcoder : This site hosts good quality problems. the
ABC
beginner contests are resourceful to a beginner. - a2oj : This has currently been discontinued, but with this extension, you can pick a ladder and start solving.
- Blogs
Youtube channels
- Code N Code: A good channel which explains several topics.
- Kartik Arora : He is a CM himself and explains several topics.
- Priyansh Agarwal : He is a CM himself and explains several topics and does screencasts.
- Utkarsh Gupta: A GM himself, has explained and done several screen casts.
- Colin Galen: A GM who regularly does screen casts of contests.
- Neal Wu: A legend’s channel!
Browser extensions
- codeforces customizer: An extension made by me!! Check the blog to see what it does. Use the
all submissions
tab to see others’ submissions and improve! (also do not forget to upvote the blog and rate a 5 star 🌟 on the chrome web store 😁) - keep problems: use this to mark favourite problems/add notes/export etc.