#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#include <climits>
#include <cstring>
#include <string>
#include <set>
#include <map>
#include <queue>
#include <stack>
#include <vector>
#include <list>
#define mod 1000000007
#define inf 0x3f3f3f3f
#define pi acos(-1.0)
using namespace std;
typedef long long ll;
const int N=2859;
int n,m,k,cnt=-1;
int d[4][2]= {0,1,1,0,0,-1,-1,0};
int parent[N+60];
bool vis[N][N];
char mp[55][55];
struct man {
int u,v,w;
} edg[N];//结构体存节点与节点之间的权值
bool cmp(man g,man h) {
return g.w>h.w;
}//将权值即得分从大到小排序
int change(char ch) {
if(ch>=‘0‘&&ch<=‘9‘)return (ch-‘0‘);
else if(ch>=‘a‘&&ch<=‘z‘)return (ch-87);
else return (ch-29);
}//字符转为数字
void init() {
for(int i=0; i<=2855; i++)parent[i]=i;
}//初始化
int Find(int x) {
if(parent[x] != x) parent[x] = Find(parent[x]);
return parent[x];
}//查找并返回节点x所属集合的根节点
void Union(int x,int y) {
x = Find(x);
y = Find(y);
if(x == y) return;
parent[y] = x;
}//将两个不同集合的元素进行合并
void Build()//将矩形图转化为树
{
for(int i=0; i<n; i++) {
for(int j=0; j<m; j++) {
for(int k=0; k<4; k++) {
int xx=i+d[k][0],yy=j+d[k][1];
if(xx>=0&&xx<n&&yy>=0&&yy<m) {
int I=i*55+j,J=xx*55+yy;
if(!vis[I][J]) {
edg[++cnt].u=I;
edg[cnt].v=J;
edg[cnt].w=abs(change(mp[i][j])-change(mp[xx][yy]));
vis[I][J]=true;
}
}
}
}
}
}
void Kruskal() {
int sumweight=0;//生成树的总权值,即所得分数
int num=0;//已经选用边的数目
int u,v;//顶点
init();
for(int i=0; i<=cnt; i++) {
u=edg[i].u;
v=edg[i].v;
if(Find(u)!=Find(v)) {
sumweight+=edg[i].w;
num++;
Union(u,v);
}
if(num>=n*m-1) break;//边已经全部建好
}
printf("%d\n",sumweight);
}
int main() {
memset(vis,false,sizeof(vis));
int u,v,w;
cin>>n>>m;
for(int i=0; i<n; i++)cin>>mp[i];
Build();
sort(edg,edg+cnt+1,cmp);
Kruskal();
return 0;
}