#include <bits/stdc++.h>
#define ff first
#define ss second
#define pb push_back
#define mp make_pair
#define ins insert
#define sz(a) ((int)(a.size()))
#define mask(n) (1LL << (n))
#define bit(n, i) ((n) >> (i) & 1)
using namespace std;
using ll = long long;
using ld = long double;
const int N = 200005;
struct pt {
ll x, y, r;
pt(): x(), y(), r() {}
pt(ll _x, ll _y, ll _r): x(_x), y(_y), r(_r) {}
int operator<(const pt& o) {
if (x == o.x) {
return y < o.y;
}
return x < o.x;
}
};
ll n;
pt a[N];
namespace sub1 {
int check() {
return n <= 20;
}
int vis[N];
vector<int> g[N];
ll disqr(pt a, pt b) {
return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
}
void dfs(int u) {
vis[u] = 1;
for (int v: g[u]) {
if (!vis[v]) dfs(v);
}
}
void solve() {
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= n; ++j) {
if (i == j) continue;
if (disqr(a[i], a[j]) <= a[i].r * a[i].r) {
g[i].pb(j);
}
}
}
ll ans = (ll)(1e18);
for (int msk = 0; msk < mask(n); ++msk) {
for (int i = 1; i <= n; ++i) {
vis[i] = 0;
}
ll sum = 0;
for (int i = 0; i < n; ++i) {
if (bit(msk, i) && !vis[i+1]) {
dfs(i+1);
++sum;
}
}
int ok = 1;
for (int i = 1; i <= n; ++i) {
if (!vis[i]) {
ok = 0;
break;
}
}
if (!ok) continue;
ans = min(ans, sum);
}
cout << ans << "\n";
}
}
namespace sub2 {
int check() {
return n <= 1000;
}
ll id, cnt, on[N], in[N], low[N], num[N], pat[N];
stack<int> st;
vector<int> g[N];
ll disqr(pt a, pt b) {
return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
}
void scc(int u) {
on[u] = 1;
st.push(u);
low[u] = num[u] = ++id;
for (int v: g[u]) {
if (!num[v]) {
scc(v);
low[u] = min(low[u], low[v]);
}
if (on[v]) {
low[u] = min(low[u], num[v]);
}
}
if (low[u] == num[u]) {
++cnt;
int v = -1;
while (v != u) {
v = st.top();
st.pop();
on[v] = 0;
pat[v] = cnt;
}
}
}
void solve() {
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= n; ++j) {
if (i == j) continue;
if (disqr(a[i], a[j]) <= a[i].r * a[i].r) {
g[i].pb(j);
}
}
}
cnt = id = 0;
for (int u = 1; u <= n; ++u) {
if (!num[u]) scc(u);
}
for (int u = 1; u <= n; ++u) {
for (int v: g[u]) {
if (pat[u] != pat[v]) {
++in[pat[v]];
}
}
}
ll ans = 0;
for (int u = 1; u <= cnt; ++u) {
ans += (in[u] == 0);
}
cout << ans << "\n";
}
}
namespace subfull {
int check() {
return 1;
}
const int M = 1e6 + 5;
ll id, cnt, on[N], in[N], low[N], num[N], pat[N];
stack<int> st;
pt b[N];
vector<pair<pt, int>> vts[3*M];
map<pair<ll, ll>, ll> rep;
vector<int> g[N];
ll disqr(pt a, pt b) {
return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
}
ll bin(ll x, ll v) {
ll l = 0, r = sz(vts[x+M]) - 1, res = r+1;
while (l <= r) {
ll m = l + (r - l) / 2;
if (vts[x+M][m].ff.y >= v) {
res = m;
r = m - 1;
}
else {
l = m + 1;
}
}
return res;
}
void scc(int u) {
on[u] = 1;
st.push(u);
low[u] = num[u] = ++id;
for (int v: g[u]) {
if (!num[v]) {
scc(v);
low[u] = min(low[u], low[v]);
}
if (on[v]) {
low[u] = min(low[u], num[v]);
}
}
if (low[u] == num[u]) {
++cnt;
int v = -1;
while (v != u) {
v = st.top();
st.pop();
on[v] = 0;
pat[v] = cnt;
}
}
}
void solve() {
int m = 0;
for (int i = 1; i <= n; ++i) {
if (!rep.count(mp(a[i].x, a[i].y))) {
rep[mp(a[i].x, a[i].y)] = a[i].r;
b[++m] = a[i];
}
else {
rep[mp(a[i].x, a[i].y)] = max(rep[mp(a[i].x, a[i].y)], a[i].r);
}
}
n = m;
for (int i = 1; i <= n; ++i) {
a[i] = pt(b[i].x, b[i].y, rep[mp(b[i].x, b[i].y)]);
}
sort(a+1, a+n+1);
for (int i = 1; i <= n; ++i) {
vts[a[i].x+M].pb(mp(a[i], i));
}
int edg = 0;
for (int i = 1; i <= n; ++i) {
for (ll x = a[i].x-a[i].r; x <= a[i].x+a[i].r; ++x) {
ll p = bin(x, a[i].y-a[i].r);
while (p < sz(vts[x+M]) && vts[x+M][p].ff.y <= a[i].y + a[i].r) {
if (disqr(vts[x+M][p].ff, a[i]) <= a[i].r * a[i].r) {
g[i].pb(vts[x+M][p].ss);
if (++edg >= (int)(1e8)) break;
}
++p;
}
if (edg >= (int)(1e8)) break;
}
if (edg >= (int)(1e8)) break;
}
cnt = id = 0;
for (int u = 1; u <= n; ++u) {
if (!num[u]) scc(u);
}
for (int u = 1; u <= n; ++u) {
for (int v: g[u]) {
if (pat[u] != pat[v]) {
++in[pat[v]];
}
}
}
ll ans = 0;
for (int u = 1; u <= cnt; ++u) {
ans += (in[u] == 0);
}
cout << ans << "\n";
}
}
int main() {
freopen("WIFI.INP", "r", stdin);
freopen("WIFI.OUT", "w", stdout);
ios_base::sync_with_stdio(false);
cin.tie(nullptr);
cin >> n;
for (int i = 1; i <= n; ++i) {
cin >> a[i].x >> a[i].y >> a[i].r;
}
if (sub1::check()) return sub1::solve(), 0;
if (sub2::check()) return sub2::solve(), 0;
subfull::solve();
return 0;
}
#include <bits/stdc++.h>

#define ff first
#define ss second
#define pb push_back
#define mp make_pair
#define ins insert
#define sz(a) ((int)(a.size()))
#define mask(n) (1LL << (n))
#define bit(n, i) ((n) >> (i) & 1)

using namespace std;
using ll = long long;
using ld = long double;

const int N = 200005;

struct pt {
    ll x, y, r;
    pt(): x(), y(), r() {}
    pt(ll _x, ll _y, ll _r): x(_x), y(_y), r(_r) {}
    int operator<(const pt& o) {
        if (x == o.x) {
            return y < o.y;
        }
        return x < o.x;
    }
};

ll n;
pt a[N];

namespace sub1 {
    int check() {
        return n <= 20;
    }

    int vis[N];
    vector<int> g[N];

    ll disqr(pt a, pt b) {
        return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
    }

    void dfs(int u) {
        vis[u] = 1;
        for (int v: g[u]) {
            if (!vis[v]) dfs(v);
        }
    }

    void solve() {
        for (int i = 1; i <= n; ++i) {
            for (int j = 1; j <= n; ++j) {
                if (i == j) continue;
                if (disqr(a[i], a[j]) <= a[i].r * a[i].r) {
                    g[i].pb(j);
                }
            }
        }
        ll ans = (ll)(1e18);
        for (int msk = 0; msk < mask(n); ++msk) {
            for (int i = 1; i <= n; ++i) {
                vis[i] = 0;
            }
            ll sum = 0;
            for (int i = 0; i < n; ++i) {
                if (bit(msk, i) && !vis[i+1]) {
                    dfs(i+1);
                    ++sum;
                }
            }
            int ok = 1;
            for (int i = 1; i <= n; ++i) {
                if (!vis[i]) {
                    ok = 0;
                    break;
                }
            }
            if (!ok) continue;
            ans = min(ans, sum);
        }
        cout << ans << "\n";
    }
}

namespace sub2 {
    int check() {
        return n <= 1000;
    }

    ll id, cnt, on[N], in[N], low[N], num[N], pat[N];
    stack<int> st;
    vector<int> g[N];

    ll disqr(pt a, pt b) {
        return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
    }

    void scc(int u) {
        on[u] = 1;
        st.push(u);
        low[u] = num[u] = ++id;
        for (int v: g[u]) {
            if (!num[v]) {
                scc(v);
                low[u] = min(low[u], low[v]);
            }
            if (on[v]) {
                low[u] = min(low[u], num[v]);
            }
        }
        if (low[u] == num[u]) {
            ++cnt;
            int v = -1;
            while (v != u) {
                v = st.top();
                st.pop();
                on[v] = 0;
                pat[v] = cnt;
            }
        }
    }

    void solve() {
        for (int i = 1; i <= n; ++i) {
            for (int j = 1; j <= n; ++j) {
                if (i == j) continue;
                if (disqr(a[i], a[j]) <= a[i].r * a[i].r) {
                    g[i].pb(j);
                }
            }
        }
        cnt = id = 0;
        for (int u = 1; u <= n; ++u) {
            if (!num[u]) scc(u);
        }
        for (int u = 1; u <= n; ++u) {
            for (int v: g[u]) {
                if (pat[u] != pat[v]) {
                    ++in[pat[v]];
                }
            }
        }
        ll ans = 0;
        for (int u = 1; u <= cnt; ++u) {
            ans += (in[u] == 0);
        }
        cout << ans << "\n";
    }
}

namespace subfull {
    int check() {
        return 1;
    }

    const int M = 1e6 + 5;

    ll id, cnt, on[N], in[N], low[N], num[N], pat[N];
    stack<int> st;
    pt b[N];
    vector<pair<pt, int>> vts[3*M];
    map<pair<ll, ll>, ll> rep;
    vector<int> g[N];

    ll disqr(pt a, pt b) {
        return (b.x - a.x) * (b.x - a.x) + (b.y - a.y) * (b.y - a.y);
    }

    ll bin(ll x, ll v) {
        ll l = 0, r = sz(vts[x+M]) - 1, res = r+1;
        while (l <= r) {
            ll m = l + (r - l) / 2;
            if (vts[x+M][m].ff.y >= v) {
                res = m;
                r = m - 1;
            }
            else {
                l = m + 1;
            }
        }
        return res;
    }

    void scc(int u) {
        on[u] = 1;
        st.push(u);
        low[u] = num[u] = ++id;
        for (int v: g[u]) {
            if (!num[v]) {
                scc(v);
                low[u] = min(low[u], low[v]);
            }
            if (on[v]) {
                low[u] = min(low[u], num[v]);
            }
        }
        if (low[u] == num[u]) {
            ++cnt;
            int v = -1;
            while (v != u) {
                v = st.top();
                st.pop();
                on[v] = 0;
                pat[v] = cnt;
            }
        }
    }

    void solve() {
        int m = 0;
        for (int i = 1; i <= n; ++i) {
            if (!rep.count(mp(a[i].x, a[i].y))) {
                rep[mp(a[i].x, a[i].y)] = a[i].r;
                b[++m] = a[i];
            }
            else {
                rep[mp(a[i].x, a[i].y)] = max(rep[mp(a[i].x, a[i].y)], a[i].r);
            }
        }
        n = m;
        for (int i = 1; i <= n; ++i) {
            a[i] = pt(b[i].x, b[i].y, rep[mp(b[i].x, b[i].y)]);
        }
        sort(a+1, a+n+1);
        for (int i = 1; i <= n; ++i) {
            vts[a[i].x+M].pb(mp(a[i], i));
        }
        int edg = 0;
        for (int i = 1; i <= n; ++i) {
            for (ll x = a[i].x-a[i].r; x <= a[i].x+a[i].r; ++x) {
                ll p = bin(x, a[i].y-a[i].r);
                while (p < sz(vts[x+M]) && vts[x+M][p].ff.y <= a[i].y + a[i].r) {
                    if (disqr(vts[x+M][p].ff, a[i]) <= a[i].r * a[i].r) {
                        g[i].pb(vts[x+M][p].ss);
                        if (++edg >= (int)(1e8)) break;
                    }
                    ++p;
                }
                if (edg >= (int)(1e8)) break;
            }
            if (edg >= (int)(1e8)) break;
        }
        cnt = id = 0;
        for (int u = 1; u <= n; ++u) {
            if (!num[u]) scc(u);
        }
        for (int u = 1; u <= n; ++u) {
            for (int v: g[u]) {
                if (pat[u] != pat[v]) {
                    ++in[pat[v]];
                }
            }
        }
        ll ans = 0;
        for (int u = 1; u <= cnt; ++u) {
            ans += (in[u] == 0);
        }
        cout << ans << "\n";
    }
}

int main() {
    freopen("WIFI.INP", "r", stdin);
    freopen("WIFI.OUT", "w", stdout);

    ios_base::sync_with_stdio(false);
    cin.tie(nullptr);

    cin >> n;
    for (int i = 1; i <= n; ++i) {
        cin >> a[i].x >> a[i].y >> a[i].r;
    }

    if (sub1::check()) return sub1::solve(), 0;
    if (sub2::check()) return sub2::solve(), 0;
    subfull::solve();

    return 0;
}