BZOJ-3338: Zju1505 Solitaire(双向BFS)

题目:http://www.lydsy.com/JudgeOnline/problem.php?id=3338

被Source剧透了真不爽。。。话说这么一道水BFS居然能让我调上一天。。。我实在还是太弱了。。。

判重:把棋盘8*8=64压成一个64无符号位整型,然后就直接set好了。。。话说打Hash不是更快?懒呗~

代码:

8ad4b31c8701a18b756007339c2f07082938fec7.jpg.png
#include 
#include 
#include 
#include 
#include 
 
using namespace std ;
 
#define ULL unsigned long long 
#define check( x , y ) ( x > 0 && x < 9 && y > 0 && y < 9 )
 
struct node {
    int p[ 4 ][ 2 ] , dist ;
    bool flag ;
};
 
queue < node > Q ;
 
set < ULL > s0 ,s1 ;
 
ULL Exp[ 64 ] ;
 
ULL Hash( node x ) {
    ULL rec = 0 ;
    for ( int i = 0 ; i < 4 ; i ++ ) rec += Exp[ 8 * ( x.p[ i ][ 0 ] - 1 ) + x.p[ i ][ 1 ] - 1 ] ;
    return rec ;
}
 
node S , T ;
 
const int dir[ 4 ][ 2 ] = { { - 1 , 0 } , { 1 , 0 } , { 0 , - 1 } , { 0 , 1 } } ;
 
bool Check( int x , int y , node u ) {
    for ( int i = 0 ; i < 4 ; i ++ ) if ( u.p[ i ][ 0 ] == x && u.p[ i ][ 1 ] == y ) return false ;
    return true ;
}
 
bool bfs(  ) {
    while ( ! Q.empty(  ) ) Q.pop(  ) ;
    s0.insert( Hash( S ) ) , s1.insert( Hash( T ) ) ; Q.push( S ) , Q.push( T ) ;
    while ( ! Q.empty(  ) ) {
        node v = Q.front(  ) ; Q.pop(  ) ;
        for ( int i = 0 ; i < 4 ; i ++ ) for ( int j = 0 ; j < 4 ; j ++ ) {
            int x = v.p[ i ][ 0 ] + dir[ j ][ 0 ] , y = v.p[ i ][ 1 ] + dir[ j ][ 1 ] ;
            if ( ! check( x , y ) ) continue ;
            if ( ! Check( x , y , v ) ) x += dir[ j ][ 0 ] , y += dir[ j ][ 1 ] ;
            if ( ( ! check( x , y ) ) || ( ! Check( x , y , v ) ) ) continue ;
            node u = v ;
            u.p[ i ][ 0 ] = x , u.p[ i ][ 1 ] = y , u.dist ++ ;
            ULL w = Hash( u ) ;
            if ( s0.find( w ) != s0.end(  ) ) {
                if ( u.flag ) return true ;
                continue ;
            }
            if ( s1.find( w ) != s1.end(  ) ) {
                if ( ! u.flag ) return true ; 
                continue ;
            }
            if ( u.flag ) s1.insert( w ) ; else s0.insert( w ) ;
            if ( u.dist >= 4 ) continue ;
            Q.push( u ) ;
        }
    }
    return false ;
}
 
int main(  ) {
    Exp[ 0 ] = 1 ; for ( int i = 1 ; i < 64 ; i ++ ) Exp[ i ] = Exp[ i - 1 ] * 2 ;
    while ( scanf( "%d%d" , &S.p[ 0 ][ 0 ] , &S.p[ 0 ][ 1 ] ) != EOF ) {
        s0.clear(  ) , s1.clear(  ) ;
        for ( int i = 1 ; i < 4 ; i ++ ) scanf( "%d%d" , &S.p[ i ][ 0 ] , &S.p[ i ][ 1 ] ) ;
        for ( int i = 0 ; i < 4 ; i ++ ) scanf( "%d%d" , &T.p[ i ][ 0 ] , &T.p[ i ][ 1 ] ) ;
        S.dist = T.dist = 0 ; S.flag = 0 , T.flag = 1 ;
        printf( bfs(  ) ? "YES\n" : "NO\n" ) ;
    }
    return 0 ;
}

你可能感兴趣的:(BZOJ-3338: Zju1505 Solitaire(双向BFS))