#define _CRT_SECURE_NO_WARNINGS

#include <stdio.h>
#include <algorithm>
#include <string.h>

using namespace std;

#define FOR(i,a,b) for(int i=a;i<b;i++)
#define REP(i,b) FOR(i,0,b)

const int Nmax = 1000001;
int bucket[Nmax];

template <class T>
void CreateBeginBucket(T* data, int size, int maxVal){
	REP(i, maxVal + 1) bucket[i] = 0;
	REP(i, size) bucket[data[i]]++;
	int sum = 0;
	REP(i, maxVal + 1){ bucket[i] += sum; swap(bucket[i], sum); }
}

template <class T>
void CreateEndBucket(T* data, int size, int maxVal){
	REP(i, maxVal + 1) bucket[i] = 0;
	REP(i, size) bucket[data[i]]++;
	int sum = 0;
	REP(i, maxVal + 1){ sum += bucket[i]; bucket[i] = sum; }
}

template<class T>
void InducedSort(T* data, int size, int* SA, int maxVal, bool* isL){
	CreateBeginBucket(data, size, maxVal);
	REP(i, size) if (SA[i] > 0 && isL[SA[i] - 1]) SA[bucket[data[SA[i] - 1]]++] = SA[i] - 1;
}

template<class T>
void InvertInducedSort(T* data, int size, int* SA, int maxVal, bool* isL){
	CreateEndBucket(data, size, maxVal);
	for (int i = size - 1; i >= 0; --i) if (SA[i] > 0 && !isL[SA[i] - 1]) SA[--bucket[data[SA[i] - 1]]] = SA[i] - 1;
}

template <class T>
void DBGOUT(T* sa, int size){
	REP(i, size) printf("%d ", int(sa[i]));
	printf("\n");
}

template<class T>
void SA_IS(T* data, int size, int* SA, int maxVal, bool* isL){
	REP(i, size) SA[i] = -1;
#define isLMS(x) (x>0 && isL[x-1] && !isL[x])
	isL[size - 1] = false;
	for (int i = size - 2; i >= 0; i--) isL[i] = data[i] > data[i + 1] || (data[i] == data[i + 1] && isL[i + 1]);
	CreateEndBucket(data, size, maxVal);
	FOR(i, 1, size) if (isLMS(i)) SA[--bucket[data[i]]] = i;
	InducedSort(data, size, SA, maxVal, isL);
	InvertInducedSort(data, size, SA, maxVal, isL);

	int c = 0;
	REP(i, size) if (isLMS(SA[i])) SA[c++] = SA[i];
	FOR(i, c, size) SA[i] = -1;

	int idx = -1;
	int prev = -1;
	REP(i, c){
		bool diff = false;
		REP(d, size){
			if (prev == -1 || data[SA[i] + d] != data[prev + d] || isL[SA[i] + d] != isL[prev + d]){
				diff = true;
				break;
			}
			else if (d > 0 && isLMS(SA[i] + d)) break;
		}
		if (diff){ idx++; prev = SA[i]; }
		SA[c + SA[i] / 2] = idx;
	}
	int j = size;
	for (int i = size - 1; i >= c; i--) if (SA[i] >= 0) SA[--j] = SA[i];

	int* nxdata = SA + size - c;
	int* nxsa = SA;
	if (c == idx + 1) REP(i, c) nxsa[nxdata[i]] = i;
	else SA_IS(nxdata, c, nxsa, idx, isL + size);

	j = c;
	for (int i = size - 1; i >= 1; i--) if (isLMS(i)) nxdata[--j] = i;
	REP(i, c) nxsa[i] = nxdata[nxsa[i]];
	FOR(i, c, size) SA[i] = -1;
	CreateEndBucket(data, size, maxVal);
	for (int i = c - 1; i >= 0; i--) swap(nxsa[i], SA[--bucket[data[nxsa[i]]]]);
	InducedSort(data, size, SA, maxVal, isL);
	InvertInducedSort(data, size, SA, maxVal, isL);
}

bool isLPool[Nmax * 2];
void SA_IS(unsigned char* input, int size, int* SA){
	int mv = 0;
	REP(i, size) if (mv < input[i]) mv = input[i];
	SA_IS(input, size, SA, mv, isLPool);
}

int lcp[Nmax];
int invertSA[Nmax];
void CreateLCP(unsigned char* data, int size, int* SA){
	lcp[0] = -1;
	REP(i, size) invertSA[SA[i]] = i;
	int prev = 0;
	REP(i, size){
		if (invertSA[i] > 0){
			while (data[i + prev] == data[SA[invertSA[i] - 1] + prev]){
				++prev;
				if (i + prev >= size || SA[invertSA[i] - 1] + prev >= size)
					break;
			}
			lcp[invertSA[i]] = prev;
		}
		prev = max(prev - 1, 0);
	}
}

int st[21][Nmax];
void InitSparseTable(int n){
	int h = 1;
	while ((1 << h) < n) h++;
	REP(i, n) st[0][i] = lcp[i];
	FOR(j, 1, h + 1){
		REP(i, n - (1<<j) + 1){
			st[j][i] = min(st[j - 1][i], st[j - 1][i + (1 << (j - 1))]);
		}
	}
}

inline int TopBit(int t){
	for (int i = 20; i >= 0; i--){
		if ((1 << i)&t) return i;
	}
	return -1;
}

int GetLCP(int f, int s){
	if (f > s) swap(f, s);
	int diff = TopBit(s-f);
	return min(st[diff][f], st[diff][s - (1 << diff)]);
}

unsigned char str[Nmax];
int indices[Nmax];

int compare(int f, int s, int l){
	int fi = invertSA[f];
	int si = invertSA[s];
	if (GetLCP(fi + 1, si + 1) >= l)
		return 0;
	else
		return 2 * (fi > si) - 1;
}

int dpd[Nmax];
int main(){
	int n;
	scanf("%d", &n);
	scanf("%s", str);
	SA_IS(str, n + 1, indices);
	CreateLCP(str, n + 1, indices);
	InitSparseTable(n + 1);
	fill(dpd, dpd + n + 1, 1000000000);
	int* dp = dpd + 1;
	dp[n - 2] = 1;
	int ans = 1000000000;
	for (int i = n - 2; i >= 0; i--){
		if (str[i + 1] != '0'){
			int nx = i - dp[i];
			if (nx >= -1){
				if (compare(nx + 1, i + 1, dp[i]) <= 0){
					nx--;
				}
				if (nx >= -1){
					dp[nx] = min(dp[nx], i - nx);
				}
			}
		}
		dp[i - 1] = min(dp[i - 1], dp[i] + 1);
	}
	printf("%d\n", dp[-1]);
}
