BZOJ-2877: [Noi2012]魔幻棋盘(线段树)

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

嗯,这次好好写一次题解吧,这题傻叉的写了半天QAQ :

首先,由欧几里德算法,我们知道gcd(a,b)=gcd(a,abs(a-b)),所以:

令G(ai)=gcd(a1,a2,...ai...,an)则,G(ai)=gcd(a1,a2-a1,...,an-a(n-1)),所以对于矩阵[l,r,x,y](表示矩阵四个顶点中,最小的行数为l,最大为r,最小的列数为x,最大的为y),设题目中的定点为(px,py),则答案为:

G(a(i,j))(l<=i<=r,x<=j<=y)

=G( G(a(i,j)(x<=j<=y) ) (l<=i<=r)

=G( gcd( G( a( i , j ) - a( i , j - 1 ) ) , a( i , py ) ) ( x < j <= y ) ) ( l <= i <= r )

=gcd( G( G( a( i , j ) - a( i , j - 1 ) ) ( l <= i <= r ) ) ( x < j <= y ) , G( a( i , py ) ) ( l <= i <= r ) )

那么拆一下:

G( a( i , py ) ) ( l <= i <= r ) = gcd( a( px , py ) , G( a( i , py ) - a( i - 1 , py ) ) ( l < i <= r ) )

G( G( a( i , j ) - a( i , j - 1 ) ) ) ( l <= i <= r ) ( x < j <= y )

= G( gcd( a( px , j ) - a( px , j - 1 ) , G( [ a( i , j ) - a( i , j - 1 ) ] - [ a( i - 1 , j ) - a( i - 1 , j - 1 ) ] ) ( l < i <= r ) ) )( x < j <= y )

= gcd( G( a( px , j ) - a( px , j - 1 ) ) ( x < j <= y ) , G( a( i , j ) - a( i , j - 1 ) - a( i - 1 , j ) + a( i - 1 , j - 1 ) )( l < i <= r && x < j <= y ) )

然后,用两棵一维线段树维护G( a( i , py ) - a( i - 1 , py ) ) ( l < i <= r ) , G( a( px , j ) - a( px , j - 1 ) ) ( x < j <= y ) ,用一棵二维线段树维护G( a( i , j ) - a( i , j - 1 ) - a( i - 1 , j ) + a( i - 1 , j - 1 ) )( l < i <= r && x < j <= y ) ,我们神奇的发现每次修改操作在这三棵线段树上都是单点修改,然后就可以AC了。

时间复杂度:O(n log^3 n)

代码:

#include <cstdio>

#include <algorithm>

#include <cstring>

 

using namespace std ;

 

#define REP( i , l , r ) for ( int i = l ; i <= r ; ++ i )

#define rep( i , x ) for ( int i = 0 ; i ++ < x ; )

#define Rep( i , x ) for ( int i = 0 ; i < x ; ++ i )

 

const int maxn = 501000 ;

 

typedef long long ll ;

 

ll gcd( ll x , ll y ) {

    if ( x < y ) swap( x , y ) ;

    for ( ll t ; y ; t = y , y = x % y , x = t ) ;

    return x ;

}

 

struct node {

     

    int l , r , mid ;

    ll g ;

    node *lc , *rc ;

     

    inline void update(  ) {

        g = gcd( abs( lc -> g ) , abs( rc -> g ) ) ;

    }

     

} sgt[ maxn * 25 ] ;

 

node *pt = sgt ;

 

void build( int l , int r , ll a[] , node* &t ) {

    t = pt ++ ;

    t -> l = l , t -> r = r ;

    if ( l == r ) {

        t -> g = a[ l ] ; return ;

    }

    t -> mid = ( l + r ) >> 1 ;

    build( l , t -> mid , a , t -> lc ) , build( t -> mid + 1 , r , a , t -> rc ) ;

    t -> update(  ) ;

}

 

void change( int p , ll d , node *t ) {

    if ( t -> l == t -> r ) {

        t -> g += d ; return ;

    }

    change( p , d , p <= t -> mid ? t -> lc : t -> rc ) ;

    t -> update(  ) ;

}

 

ll query( int l , int r , node *t ) {

    if ( l > r ) return 0 ;

    if ( l == t -> l && r == t -> r ) return abs( t -> g ) ;

    if ( r <= t -> mid ) return query( l , r , t -> lc ) ;

    if ( l > t -> mid ) return query( l , r , t -> rc ) ;

    return gcd( query( l , t -> mid , t -> lc ) , query( t -> mid + 1 , r , t -> rc ) ) ;

}

 

struct Node {

     

    int l , r , mid ;

    node *t ;

    Node *lc , *rc ;

     

    inline void Update( int p ) {

        node *l = lc -> t , *r = rc -> t , *m = t ;

        for ( ; ; ) {

            m -> g = gcd( abs( l -> g ) , abs( r -> g ) ) ;

            if ( m -> l == m -> r ) break ;

            if ( p <= m -> mid ) {

                l = l -> lc , r = r -> lc , m = m -> lc ;

            } else {

                l = l -> rc , r = r -> rc , m = m -> rc ;

            }

        }

    }

     

} Sgt[ maxn << 2 ] ;

 

Node *Root , *Pt = Sgt ;

 

ll a[ maxn ] , del[ maxn ] , b[ maxn ] ;

int n , m , q , px , py ;

 

#define chose( x , y ) ( ( ( ( x - 1 ) * m + y ) > 0 && ( ( x - 1 ) * m + y ) <= ( n * m ) ) ? ( ( x - 1 ) * m + y ) : 0 )

 

void Merge( node *l , node *r , node* &t ) {

    t = pt ++ ;

    t -> l = l -> l , t -> r = l -> r , t -> mid = l -> mid , t -> g = gcd( l -> g , r -> g ) ;

    if ( t -> l == t -> r ) return ;

    Merge( l -> lc , r -> lc , t -> lc ) , Merge( l -> rc , r -> rc , t -> rc ) ;

}

 

void Build( int l , int r , Node* &t ) {

    t = Pt ++ ;

    t -> l = l , t -> r = r ;

    if ( l == r ) {

        rep( i , m ) b[ i ] = del[ chose( l , i ) ] ;

        build( 1 , m , b , t -> t ) ;

        return ;

    }

    t -> mid = ( l + r ) >> 1 ;

    Build( l , t -> mid , t -> lc ) , Build( t -> mid + 1 , r , t -> rc ) ;

    Merge( t -> lc -> t , t -> rc -> t , t -> t ) ;

}

 

void Change( int x , int y , ll d , Node *t ) {

    if ( t -> l == t -> r ) {

        change( y , d , t -> t ) ; return ;

    }

    Change( x , y , d , x <= t -> mid ? t -> lc : t -> rc ) ;

    t -> Update( y ) ;

}

 

ll Query( int l , int r , int x , int y , Node *t ) {

    if ( l > r ) return 0 ;

    if ( t -> l == l && t -> r == r ) return query( x , y , t -> t ) ;

    if ( r <= t -> mid ) return Query( l , r , x , y , t -> lc ) ;

    if ( l > t -> mid ) return Query( l , r , x , y , t -> rc ) ;

    return gcd( Query( l , t -> mid , x , y , t -> lc ) , Query( t -> mid + 1 , r , x , y , t -> rc ) ) ;

}

 

node *root[ 2 ] ;

ll num ;

 

int cnt = 0 ;

 

inline ll solve( int l , int r , int x , int y ) {

    ll ans = abs( num ) ;

    ans = gcd( ans , Query( l + 1 , r , x + 1 , y , Root ) ) ;

    ans = gcd( ans , query( x + 1 , y , root[ 0 ] ) ) ;

    ans = gcd( ans , query( l + 1 , r , root[ 1 ] ) ) ;

    return ans ;

}

 

inline void modify( int l , int r , int x , int y , ll d ) {

    if ( r + 1 <= n ) {

        if ( y + 1 <= m ) Change( r + 1 , y + 1 , d , Root ) ;

        Change( r + 1 , x , -d , Root ) ;

    }

    if ( y + 1 <= m ) Change( l , y + 1 , -d , Root ) ;

    Change( l , x , d , Root ) ;

    if ( l <= px && r >= px ) {

        change( x , d , root[ 0 ] ) ;

        if ( y + 1 <= m ) change( y + 1 , -d , root[ 0 ] ) ;

        if ( x <= py && y >= py ) num += d ;

    }

    if ( x <= py && y >= py ) {

        change( l , d , root[ 1 ] ) ;

        if ( r + 1 <= n ) change( r + 1 , -d , root[ 1 ] ) ;

    }

}

 

int main(  ) {

    scanf( "%d%d%d%d%d" , &n , &m , &px , &py , &q ) ;

    memset( del , 0 , sizeof( del ) ) , memset( a , 0 , sizeof( a ) ) ;

    rep( i , n ) rep( j , m ) scanf( "%lld" , &a[ chose( i , j ) ] ) ;

    rep( i , n ) rep( j , m ) {

        del[ chose( i , j ) ] = a[ chose( i , j ) ] - a[ chose( i - 1 , j ) ] - a[ chose( i , j - 1 ) ] + a[ chose( i - 1 , j - 1 ) ] ;

    }

    Build( 1 , n , Root ) ;

    rep( i , m ) b[ i ] = a[ chose( px , i ) ] - a[ chose( px , i - 1 ) ] ;

    build( 1 , m , b , root[ 0 ] ) ;

    rep( i , n ) b[ i ] = a[ chose( i , py ) ] - a[ chose( i - 1 , py ) ] ;

    build( 1 , n , b , root[ 1 ] ) ;

    int x , y , z , l , r ; ll d ;

    num = a[ chose( px , py ) ] ;

    while ( q -- ) {

        scanf( "%d" , &z ) ;

        if ( ! z ) {

            scanf( "%d%d%d%d" , &l , &x , &r , &y ) ;

            printf( "%lld\n" , solve( px - l , px + r , py - x , py + y ) ) ;

        } else {

            scanf( "%d%d%d%d%lld" , &l , &x , &r , &y , &d ) ;

            modify( l , r , x , y , d ) ;

        }

    }

    return 0 ;

}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容

  • 拿到这一年的六千块钱的奖励性绩效真的有种绝望的感觉。三十五岁已经是人生最尴尬的时候。别人说你老了,可你觉得自己不过...
    幸福一生_61a7阅读 207评论 0 0
  • 月落了轻纱 掩盖了谁的哀伤 梦留住了画 遮住谁的木兰香 树梢勾勒的黑影 沉淀了谁的思绪 泛黄是那油蜡灯 滚烫而不知...
    玉米粉阅读 210评论 0 4
  • 张艳 焦点网络初级7期 坚持分享第35天 今天改了一下学生的周末作业,除了几个认真写的,都是随便编的答案,应付,气...
    柚橙妈咪阅读 222评论 0 0