// g++ -O3 -std=c++0x iterative_o3.cc
// ./a.out < testcase.txt
// cat result.txt
#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <algorithm>
#include <cmath>
#include <random>
#include <assert.h>
#include <sys/time.h>
#include <stdint.h>
using namespace std;
const int N = 500 ;
typedef long long LL;
class Timer{
protected :
double startTime;
public :
Timer( ) { startTime = now( ) ; }
double elapse( ) { return now( ) - startTime; }
protected :
double now( ) {
timeval tv;
gettimeofday( & tv, 0 ) ;
return tv.tv_sec + tv.tv_usec * 1e-6 ;
}
} ;
template < class T>
struct Matrix{
T data[ N* N] ;
Matrix( ) { }
Matrix( const T& init) {
for ( int y = 0 ; y < N; y++ ) {
for ( int x = 0 ; x < N; x++ ) {
data[ y* N+ x] = init;
}
}
}
const T* operator[ ] ( int y) const {
return & ( data[ y* N] ) ;
}
T* operator[ ] ( int y) {
return & ( data[ y* N] ) ;
}
} ;
Timer timer;
Matrix< int > preset;
Matrix< int > board;
mt19937_64 mt19937_obj;
LL myrand( ) {
return mt19937_obj( ) >> 1 ;
}
template < class T>
void shuffle( T b, T e) {
while ( b ! = e) {
T t = b + myrand( ) % ( e- b) ;
swap( * t, * b) ;
b++ ;
}
}
void initialize( ) {
for ( int y = 0 ; y < N; y++ ) {
for ( int x = 0 ; x < N; x++ ) {
if ( preset[ y] [ x] ) continue ;
board[ y] [ x] = 1 << ( myrand( ) % 9 + 1 ) ;
}
}
}
int calcScore( const Matrix< int > & board) ;
void writeBoard( const Matrix< int > & board, ostream& st) {
for ( int y = 0 ; y < N; y++ ) {
for ( int x = 0 ; x < N; x++ ) {
int v = 1 ;
for ( int i = 1 ; i <= 9 ; i++ ) {
if ( board[ y] [ x] == ( 1 << i) ) v = i;
}
st << v;
}
st << endl;
}
st << endl;
st << "score : " << calcScore( board) << endl;
st << "elapse : " << timer.elapse ( ) << endl;
}
void writeBoard( const Matrix< int > & board, const char * filename) {
ofstream st( filename) ;
writeBoard( board, st) ;
}
void readBoard( Matrix< int > & board, istream& st) {
for ( int y = 0 ; y < N; y++ ) {
string line;
st >> line;
for ( int x = 0 ; x < N; x++ ) {
int n = line[ x] - '0' ;
if ( n == 0 ) {
board[ y] [ x] = 0 ;
} else {
board[ y] [ x] = 1 << n;
}
if ( preset[ y] [ x] ) assert ( preset[ y] [ x] == board[ y] [ x] ) ;
}
}
}
void readBoard( Matrix< int > & board, const char * filename) {
ifstream st( filename) ;
readBoard( board, st) ;
}
int lineScoreH( const Matrix< int > & board, int x, int y, int ex) {
int state = 0 ;
int score = 0 ;
for ( int i = 0 ; i < 8 ; i++ , x++ ) {
state + = board[ y] [ x] ;
}
for ( ; x < ex; x++ ) {
state + = board[ y] [ x] ;
if ( state == 0x3fe ) score++ ;
state - = board[ y] [ x- 8 ] ;
}
return score;
}
int lineScoreV( const Matrix< int > & board, int x, int y, int ey) {
int state = 0 ;
int score = 0 ;
for ( int i = 0 ; i < 8 ; i++ , y++ ) {
state + = board[ y] [ x] ;
}
for ( ; y < ey; y++ ) {
state + = board[ y] [ x] ;
if ( state == 0x3fe ) score++ ;
state - = board[ y- 8 ] [ x] ;
}
return score;
}
int blockScore( const Matrix< int > & board, int y, int sx, int ex) {
if ( y+ 2 >= N) return 0 ;
int state = 0 ;
int score = 0 ;
int x = sx;
for ( ; x < sx+ 2 ; x++ ) {
state + = board[ y+ 0 ] [ x] ;
state + = board[ y+ 1 ] [ x] ;
state + = board[ y+ 2 ] [ x] ;
}
for ( ; x < ex; x++ ) {
state + = board[ y+ 0 ] [ x] ;
state + = board[ y+ 1 ] [ x] ;
state + = board[ y+ 2 ] [ x] ;
if ( state == 0x3fe ) score++ ;
state - = board[ y+ 0 ] [ x- 2 ] ;
state - = board[ y+ 1 ] [ x- 2 ] ;
state - = board[ y+ 2 ] [ x- 2 ] ;
}
return score;
}
int calcScore( const Matrix< int > & board) {
int score = 0 ;
for ( int y = 0 ; y < N; y++ ) {
score + = lineScoreH( board, 0 , y, N) ;
score + = lineScoreV( board, y, 0 , N) ;
}
for ( int y = 0 ; y+ 2 < N; y++ ) {
score + = blockScore( board, y, 0 , N) ;
}
return score;
}
int partialScore( Matrix< int > & board, int x, int y) {
int score = 0 ;
score + = lineScoreH( board, max( 0 ,x- 8 ) , y, min( N,x+ 9 ) ) ;
score + = lineScoreV( board, x, max( 0 ,y- 8 ) , min( N,y+ 9 ) ) ;
for ( int y2 = max( 0 ,y- 2 ) ; y2 <= y; y2++ ) {
score + = blockScore( board, y2, max( 0 ,x- 2 ) , min( N,x+ 3 ) ) ;
}
return score;
}
const int one_left_tbl[ ] = { 1 << 5 ,1 << 6 ,1 << 3 ,1 << 7 ,1 << 9 ,1 << 4 ,1 << 2 ,1 << 8 ,1 << 1 ,0 ,0 } ;
const int one_left_tbl_width = 11 ;
struct OneLeft{
int16_t x;
int16_t y;
int to_be;
OneLeft( ) { }
OneLeft( int x_, int y_, int to_be_) : x( x_) ,y( y_) ,to_be( to_be_) { }
} ;
void getOneLeftLineH( const Matrix< int > & board, int x, int y, vector< OneLeft> & res) {
int cnt = 0 ;
int cnt2 = 0 ;
for ( int i = 0 ; i < 9 ; i++ ) {
cnt | = board[ y] [ x+ i] ;
cnt2 ^ = board[ y] [ x+ i] ;
}
int to_be = one_left_tbl[ cnt % one_left_tbl_width] ;
if ( ( cnt | to_be) ! = 0x3fe ) return ;
int v = one_left_tbl[ ( cnt2| to_be) % one_left_tbl_width] ;
for ( int i = 0 ; i < 9 ; i++ ) {
if ( board[ y] [ x+ i] == v && ! preset[ y] [ x+ i] ) {
res.push_back ( OneLeft( x+ i, y, to_be) ) ;
}
}
}
void getOneLeftLineV( const Matrix< int > & board, int x, int y, vector< OneLeft> & res) {
int cnt = 0 ;
int cnt2 = 0 ;
for ( int y2 = y; y2 < y+ 9 ; y2++ ) {
cnt | = board[ y2] [ x] ;
cnt2 ^ = board[ y2] [ x] ;
}
int to_be = one_left_tbl[ cnt % one_left_tbl_width] ;
if ( ( cnt | to_be) ! = 0x3fe ) return ;
int v = one_left_tbl[ ( cnt2| to_be) % one_left_tbl_width] ;
for ( int y2 = y; y2 < y+ 9 ; y2++ ) {
if ( board[ y2] [ x] == v && ! preset[ y2] [ x] ) {
res.push_back ( OneLeft( x, y2, to_be) ) ;
}
}
}
void getOneLeftBlock( const Matrix< int > & board, int x, int y, vector< OneLeft> & res) {
int cnt = 0 ;
int cnt2 = 0 ;
for ( int y2 = y; y2 < y+ 3 ; y2++ ) {
cnt | = board[ y2] [ x+ 0 ] ;
cnt2 ^ = board[ y2] [ x+ 0 ] ;
cnt | = board[ y2] [ x+ 1 ] ;
cnt2 ^ = board[ y2] [ x+ 1 ] ;
cnt | = board[ y2] [ x+ 2 ] ;
cnt2 ^ = board[ y2] [ x+ 2 ] ;
}
int to_be = one_left_tbl[ cnt % one_left_tbl_width] ;
if ( ( cnt | to_be) ! = 0x3fe ) return ;
int v = one_left_tbl[ ( cnt2| to_be) % one_left_tbl_width] ;
for ( int y2 = y; y2 < y+ 3 ; y2++ ) {
for ( int dx = 0 ; dx < 3 ; dx++ ) {
if ( board[ y2] [ x+ dx] == v && ! preset[ y2] [ x+ dx] ) {
res.push_back ( OneLeft( x+ dx, y2, to_be) ) ;
}
}
}
}
void getOneLeft( const Matrix< int > & board, int x, int y, vector< OneLeft> & res) {
for ( int x2 = max( 0 ,x- 8 ) ; x2 <= min( x, N- 9 ) ; x2++ ) {
getOneLeftLineH( board, x2, y, res) ;
}
for ( int y2 = max( 0 ,y- 8 ) ; y2 <= min( y, N- 9 ) ; y2++ ) {
getOneLeftLineV( board, x, y2, res) ;
}
for ( int x2 = max( 0 ,x- 2 ) ; x2 <= min( x, N- 3 ) ; x2++ ) {
for ( int y2 = max( 0 ,y- 2 ) ; y2 <= min( y, N- 3 ) ; y2++ ) {
getOneLeftBlock( board, x2, y2, res) ;
}
}
}
int rollbackHistory( Matrix< int > & board, vector< pair< int ,int > > & history, int history_idx, int score) {
if ( history.empty ( ) ) return score;
for ( int i = history.size ( ) - 1 ; i >= history_idx; i-- ) {
int x2 = history[ i] .first % N;
int y2 = history[ i] .first / N;
board[ y2] [ x2] = history[ i] .second >> 20 ;
}
score = history[ history_idx] .second & ( ( 1 << 20 ) - 1 ) ;
history.resize ( history_idx) ;
return score;
}
static int HASH_W = 29 ;
uint64_t updateHash( uint64_t hash, int x, int y, int n) {
return hash ^ ( 0x510990bf819745cLL / ( ( ( x * 500 + y) << 10 ) + n+ 1 ) ) ;
}
int search_best = 0 ;
int search( Matrix< int > & board, int x, int y, int depth, int score, int best, vector< pair< int ,int > > & history, uint64_t hash, vector< int > & memo, const int hash_v) {
//if(depth > 15) return score;
if ( depth > 10 ) return score;
if ( score < search_best - 25 ) return score;
int search_w = 4 ;
vector< OneLeft> one_left;
one_left.reserve ( 250 ) ;
getOneLeft( board, x, y, one_left) ;
shuffle( one_left.begin ( ) , one_left.end ( ) ) ;
int history_idx = history.size ( ) ;
for ( int i = 0 ; i < search_w && i < one_left.size ( ) ; i++ ) {
int x2 = one_left[ i] .x ;
int y2 = one_left[ i] .y ;
int n = one_left[ i] .to_be ;
if ( preset[ y2] [ x2] || board[ y2] [ x2] == n) continue ;
if ( x2== x && y2== y && depth > 0 ) {
search_w++ ;
continue ;
}
if ( depth == 0 && ( x2! = x || y2! = y) ) {
search_w++ ;
continue ;
}
uint64_t new_hash = updateHash( hash, x2, y2, n) ;
if ( memo[ new_hash & ( ( 1ULL<< HASH_W) - 1 ) ] == hash_v) {
continue ;
}
int history_idx2 = history.size ( ) ;
history.push_back ( pair< int ,int > ( y2* N+ x2, score+ ( board[ y2] [ x2] << 20 ) ) ) ;
int org_score = score;
int prev_score = partialScore( board, x2, y2) ;
board[ y2] [ x2] = n;
int new_score = partialScore( board, x2, y2) ;
score + = new_score - prev_score;
score = search( board, x2, y2, depth+ 1 , score, best, history, new_hash, memo, hash_v) ;
if ( score < org_score) {
score = rollbackHistory( board, history, history_idx2, score) ;
// memory no changes
memo[ new_hash & ( ( 1ULL<< HASH_W) - 1 ) ] = hash_v;
} else {
hash = new_hash;
}
}
return score;
}
int search( Matrix< int > & board, int x, int y, int best, vector< int > & memo, const int hash_v) {
search_best = best;
vector< pair< int ,int > > history;
history.reserve ( 4090 ) ;
int score = search( board, x, y, 0 , best, best, history, 0 , memo, hash_v) ;
if ( score < best) {
score = rollbackHistory( board, history, 0 , score) ;
}
return score;
}
void optimize( ) {
initialize( ) ;
int best = calcScore( board) ;
int e2 = 0 ;
while ( true ) {
vector< int > memo( 1ULL<< HASH_W) ;
for ( int y = 0 ; y < N; y++ ) {
for ( int x = 0 ; x < N; x++ ) {
best = search( board, x, y, best, memo, x+ y* N) ;
}
cerr << e2 << " : " << y << " , " << best << " , " << timer.elapse ( ) << endl;
}
writeBoard( board, "result.txt" ) ;
assert ( best == calcScore( board) ) ;
e2++ ;
}
}
int main( ) {
readBoard( board, cin ) ;
preset = board;
optimize( ) ;
return 0 ;
}
// g++ -O3 -std=c++0x iterative_o3.cc
// ./a.out < testcase.txt
// cat result.txt
#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <algorithm>
#include <cmath>
#include <random>
#include <assert.h>
#include <sys/time.h>
#include <stdint.h>

using namespace std;
const int N = 500;
typedef long long LL;

class Timer{
protected:
    double startTime;
public:
    Timer(){ startTime = now(); }
    double elapse(){ return now() - startTime; }
protected:
    double now(){
        timeval tv;
        gettimeofday(&tv, 0);
        return tv.tv_sec + tv.tv_usec * 1e-6;
    }
};

template <class T>
struct Matrix{
    T data[N*N];
    Matrix(){}
    Matrix(const T& init){
        for(int y = 0;y < N; y++){
            for(int x = 0;x < N; x++){
                data[y*N+x] = init;
            }
        }
    }
    const T* operator[](int y) const{
        return &(data[y*N]);
    }
    T* operator[](int y){
        return &(data[y*N]);
    }
};

Timer timer;
Matrix<int> preset;
Matrix<int> board;

mt19937_64 mt19937_obj;
LL myrand(){
    return mt19937_obj() >> 1;
}

template <class T>
void shuffle(T b, T e){
    while(b != e){
        T t = b + myrand() % (e-b);
        swap(*t, *b);
        b++;
    }
}

void initialize(){
    for(int y = 0; y < N; y++){
        for(int x = 0; x < N; x++){
            if(preset[y][x]) continue;
            board[y][x] = 1<<(myrand()%9+1);
        }
    }
}

int calcScore(const Matrix<int>& board);
void writeBoard(const Matrix<int>& board, ostream& st){
    for(int y = 0; y < N; y++){
        for(int x = 0; x < N; x++){
            int v = 1;
            for(int i = 1; i <= 9; i++){
                if(board[y][x] == (1<<i)) v = i;
            }
            st << v;
        }
        st << endl;
    }
    st << endl;
    st << "score : " << calcScore(board) << endl;
    st << "elapse : " << timer.elapse() << endl;
}

void writeBoard(const Matrix<int>& board, const char* filename){
    ofstream st(filename);
    writeBoard(board, st);
}

void readBoard(Matrix<int>& board, istream& st){
    for(int y = 0; y < N; y++){
        string line;
        st >> line;
        for(int x = 0; x < N; x++){
            int n = line[x]-'0';
            if(n == 0){
                board[y][x] = 0;
            }else{
                board[y][x] = 1<<n;
            }
            if(preset[y][x]) assert(preset[y][x] == board[y][x]);
        }
    }
}

void readBoard(Matrix<int>& board, const char* filename){
    ifstream st(filename);
    readBoard(board, st);
}

int lineScoreH(const Matrix<int>& board, int x, int y, int ex){
    int state = 0;
    int score = 0;
    for(int i = 0; i < 8; i++, x++){
        state += board[y][x];
    }
    for(; x < ex; x++){
        state += board[y][x];
        if(state == 0x3fe) score++;
        state -= board[y][x-8];
    }
    return score;
}

int lineScoreV(const Matrix<int>& board, int x, int y, int ey){
    int state = 0;
    int score = 0;
    for(int i = 0; i < 8; i++, y++){
        state += board[y][x];
    }
    for(; y < ey; y++){
        state += board[y][x];
        if(state == 0x3fe) score++;
        state -= board[y-8][x];
    }
    return score;
}

int blockScore(const Matrix<int>& board, int y, int sx, int ex){
    if(y+2 >= N) return 0;
    int state = 0;
    int score = 0;
    int x = sx;
    for(; x < sx+2; x++){
        state += board[y+0][x];
        state += board[y+1][x];
        state += board[y+2][x];
    }
    for(; x < ex; x++){
        state += board[y+0][x];
        state += board[y+1][x];
        state += board[y+2][x];
        if(state == 0x3fe) score++;
        state -= board[y+0][x-2];
        state -= board[y+1][x-2];
        state -= board[y+2][x-2];
    }
    return score;
}

int calcScore(const Matrix<int>& board){
    int score = 0;
    for(int y = 0; y < N; y++){
        score += lineScoreH(board, 0, y, N);
        score += lineScoreV(board, y, 0, N);
    }
    for(int y = 0; y+2 < N; y++){
        score += blockScore(board, y, 0, N);
    }
    return score;
}

int partialScore(Matrix<int>& board, int x, int y){
    int score = 0;
    score += lineScoreH(board, max(0,x-8), y, min(N,x+9));
    score += lineScoreV(board, x, max(0,y-8), min(N,y+9));
    for(int y2 = max(0,y-2); y2 <= y; y2++){
        score += blockScore(board, y2, max(0,x-2), min(N,x+3));
    }
    return score;
}

const int one_left_tbl[] = {1<<5,1<<6,1<<3,1<<7,1<<9,1<<4,1<<2,1<<8,1<<1,0,0};
const int one_left_tbl_width = 11;

struct OneLeft{
    int16_t x;
    int16_t y;
    int to_be;
    OneLeft(){}
    OneLeft(int x_, int y_, int to_be_):x(x_),y(y_),to_be(to_be_){}
};

void getOneLeftLineH(const Matrix<int>& board, int x, int y, vector<OneLeft>& res){
    int cnt = 0;
    int cnt2 = 0;
    for(int i = 0; i < 9; i++){
        cnt |= board[y][x+i];
        cnt2 ^= board[y][x+i];
    }
    int to_be = one_left_tbl[cnt % one_left_tbl_width];
    if((cnt | to_be) != 0x3fe) return;
    int v = one_left_tbl[(cnt2|to_be) % one_left_tbl_width];
    for(int i = 0; i < 9; i++){
        if(board[y][x+i] == v && !preset[y][x+i]){
            res.push_back(OneLeft(x+i, y, to_be));
        }
    }
}

void getOneLeftLineV(const Matrix<int>& board, int x, int y, vector<OneLeft>& res){
    int cnt = 0;
    int cnt2 = 0;
    for(int y2 = y; y2 < y+9; y2++){
        cnt |= board[y2][x];
        cnt2 ^= board[y2][x];
    }
    int to_be = one_left_tbl[cnt % one_left_tbl_width];
    if((cnt | to_be) != 0x3fe) return;
    int v = one_left_tbl[(cnt2|to_be) % one_left_tbl_width];
    for(int y2 = y; y2 < y+9; y2++){
        if(board[y2][x] == v && !preset[y2][x]){
            res.push_back(OneLeft(x, y2, to_be));
        }
    }
}

void getOneLeftBlock(const Matrix<int>& board, int x, int y, vector<OneLeft>& res){
    int cnt = 0;
    int cnt2 = 0;
    for(int y2 = y; y2 < y+3; y2++){
        cnt |= board[y2][x+0];
        cnt2 ^= board[y2][x+0];
        cnt |= board[y2][x+1];
        cnt2 ^= board[y2][x+1];
        cnt |= board[y2][x+2];
        cnt2 ^= board[y2][x+2];
    }
    int to_be = one_left_tbl[cnt % one_left_tbl_width];
    if((cnt | to_be) != 0x3fe) return;
    int v = one_left_tbl[(cnt2|to_be) % one_left_tbl_width];
    for(int y2 = y; y2 < y+3; y2++){
        for(int dx = 0; dx < 3; dx++){
            if(board[y2][x+dx] == v && !preset[y2][x+dx]){
                res.push_back(OneLeft(x+dx, y2, to_be));
            }
        }
    }
}

void getOneLeft(const Matrix<int>& board, int x, int y, vector<OneLeft>& res){
    for(int x2 = max(0,x-8); x2 <= min(x, N-9); x2++){
        getOneLeftLineH(board, x2, y, res);
    }
    for(int y2 = max(0,y-8); y2 <= min(y, N-9); y2++){
        getOneLeftLineV(board, x, y2, res);
    }
    for(int x2 = max(0,x-2); x2 <= min(x, N-3); x2++){
        for(int y2 = max(0,y-2); y2 <= min(y, N-3); y2++){
            getOneLeftBlock(board, x2, y2, res);
        }
    }
}

int rollbackHistory(Matrix<int>& board, vector<pair<int,int> >& history, int history_idx, int score){
    if(history.empty()) return score;
    for(int i = history.size()-1; i >= history_idx; i--){
        int x2 = history[i].first % N;
        int y2 = history[i].first / N;
        board[y2][x2] = history[i].second >> 20;
    }
    score = history[history_idx].second & ((1<<20)-1);
    history.resize(history_idx);
    return score;
}

static int HASH_W = 29;
uint64_t updateHash(uint64_t hash, int x,  int y, int n){
    return hash ^ (0x510990bf819745cLL / (((x * 500 + y) << 10) + n+1));
}

int search_best = 0;
int search(Matrix<int>& board, int x, int y, int depth, int score, int best, vector<pair<int,int> >& history, uint64_t hash, vector<int>& memo, const int hash_v){
    //if(depth > 15) return score;
    if(depth > 10) return score;
    if(score < search_best - 25) return score;

    int search_w = 4;
    vector<OneLeft> one_left;
    one_left.reserve(250);
    getOneLeft(board, x, y, one_left);
    shuffle(one_left.begin(), one_left.end());
    int history_idx = history.size();
    for(int i = 0; i < search_w && i < one_left.size(); i++){
        int x2 = one_left[i].x;
        int y2 = one_left[i].y;
        int n = one_left[i].to_be;
        if(preset[y2][x2] || board[y2][x2]==n) continue;
        if(x2==x && y2==y && depth > 0){
            search_w++;
            continue;
        }
        if(depth == 0 && (x2!=x || y2!=y)){
            search_w++;
            continue;
        }
        uint64_t new_hash = updateHash(hash, x2, y2, n);
        if(memo[new_hash & ((1ULL<<HASH_W)-1)] == hash_v){
            continue;
        }
        int history_idx2 = history.size();
        history.push_back(pair<int,int>(y2*N+x2, score+(board[y2][x2]<<20)));
        int org_score = score;
        int prev_score = partialScore(board, x2, y2);
        board[y2][x2] = n;
        int new_score = partialScore(board, x2, y2);
        score += new_score - prev_score;
        score = search(board, x2, y2, depth+1, score, best, history, new_hash, memo, hash_v);
        if(score < org_score){
            score = rollbackHistory(board, history, history_idx2, score);
            // memory no changes
            memo[new_hash & ((1ULL<<HASH_W)-1)] = hash_v;
        }else{
            hash = new_hash;
        }
    }
    return score;
}

int search(Matrix<int>& board, int x, int y, int best, vector<int>& memo, const int hash_v){
    search_best = best;
    vector<pair<int,int> > history;
    history.reserve(4090);
    int score = search(board, x, y, 0, best, best, history, 0, memo, hash_v);
    if(score < best){
        score = rollbackHistory(board, history, 0, score);
    }
    return score;
}

void optimize(){
    initialize();
    int best = calcScore(board);
    int e2 = 0;
    while(true){
        vector<int> memo(1ULL<<HASH_W);
        for(int y = 0; y < N; y++){
            for(int x = 0; x < N; x++){
                best = search(board, x, y, best, memo, x+y*N);
            }
            cerr << e2 << " : " << y << " , " << best << " , " << timer.elapse() << endl;
        }
        writeBoard(board, "result.txt");
        assert(best == calcScore(board));
        e2++;
    }
}

int main(){
    readBoard(board, cin);
    preset = board;
    optimize();
    return 0;
}
