美文网首页信息学竞赛题解(IO题解)
BZOJ-3338: Zju1505 Solitaire(双向B

BZOJ-3338: Zju1505 Solitaire(双向B

作者: AmadeusChan | 来源:发表于2018-10-16 20:37 被阅读0次

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

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

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

    代码:

    8ad4b31c8701a18b756007339c2f07082938fec7.jpg.png
    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    #include <queue>
    #include <set>
     
    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(双向B

        本文链接:https://www.haomeiwen.com/subject/jslkzftx.html