BZOJ-1038: [ZJOI2008]瞭望塔(半平面交)

作者: AmadeusChan | 来源:发表于2018-11-29 11:09 被阅读0次

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

裸裸的半平面交,WA了N次之后实在感叹O2下的精度实在DT,改成long double之后还发现居然还可以输出-0的额。。。唉~

代码:

#include <cstdio> 
#include <algorithm>
#include <cstring>
#include <cmath>
   
using namespace std ;
   
#define rep( i , x ) for ( int i = 0 ; i ++ < x ; )
#define maxn 1010
   
typedef long double ld ;
   
const ld esp = 0.0000001 ;
const ld inf = ld( 0x7fffffff ) * ld( 0x7fffffff ) ;
   
struct point {
       
    ld x , y ;
       
    void oper( ld _x , ld _y ) {
        x = _x , y = _y ;
    }
       
} p[ maxn ] , pos[ maxn ] ;
   
struct line {
       
    ld k , b ;
       
    void oper( point x , point y ) {
        k = ( x.y - y.y ) / ( x.x - y.x ) ;
        b = x.y - k * x.x ;
    }
       
    point cross( line l ) {
        point rec ;
        rec.x = - ( b - l.b ) / ( k - l.k ) ;
        rec.y = cal( rec.x ) ;
        return rec ;
    }
       
    ld cal( ld x ) {
        return k * x + b ;
    }
       
    bool operator < ( const line &a ) const {
        return k < a.k || ( k == a.k && b > a.b ) ;
    }
       
} li[ maxn ] , st[ maxn ] ;
   
int n , m = 0 , Top = 0 , M = 0 ;
    
ld calh( ld x , int left , int right , point *ret ) {
    if ( abs( ret[ left ].x - x ) < esp ) return ret[ left ].y ;
    if ( abs( ret[ right ].x - x ) < esp ) return ret[ right ].y ;
    while ( right - left > 1 ) {
        int mid = ( left + right ) >> 1 ;
        if ( abs( ret[ mid ].x - x ) < esp ) return ret[ mid ].y ;
        if ( x < ret[ mid ].x ) right = mid ; else left = mid ;
    }
    line temp ;
    temp.oper( ret[ left ] , ret[ right ] ) ;
    return temp.cal( x ) ;
}
   
ld calheight( ld x ) {
    ld h1 = calh( x , 1 , M , pos ) , h2 = calh( x , 1 , n , p ) ;
    return h1 - h2 ;
}
   
int main(  ) {
    scanf( "%d" , &n ) ;
    rep( i , n ) {
        double x ; scanf( "%lf" , &x ) ;
        p[ i ].x = ld( x ) ;
    }
    rep( i , n ) {
        double y ; scanf( "%lf" , &y ) ;
        p[ i ].y = ld( y ) ;
    }
    rep( i , n - 1 ) li[ ++ m ].oper( p[ i ] , p[ i + 1 ] ) ;
    sort( li + 1 , li + m + 1 ) ;
    rep( i , m ) {
        if ( abs( li[ i ].k - li[ i - 1 ].k ) <= esp ) continue ;
        while ( Top > 1 ) {
            point temp = st[ Top ].cross( li[ i ] ) ;
            if ( st[ Top - 1 ].cal( temp.x ) >= temp.y ) -- Top ;
            else break ;
        }
        st[ ++ Top ] = li[ i ] ;
    } 
    rep( i , Top ) {
        point temp = st[ i ].cross( st[ i + 1 ] ) ;
    }
    int left = 1 , right = Top ;
    while ( left < right ) {
        point temp = st[ left ].cross( st[ left + 1 ] ) ;
        if ( temp.x <= p[ 1 ].x ) ++ left ; else break ;
    }
    while ( left < right ) {
        point temp = st[ right ].cross( st[ right - 1 ] ) ;
        if ( temp.x >= p[ n ].x ) -- right ; else break ;
    }
    pos[ ++ M ].oper( p[ 1 ].x , st[ left ].cal( p[ 1 ].x ) ) ;
    for ( int i = left ; i < right ; ++ i ) {
        pos[ ++ M ] = st[ i ].cross( st[ i + 1 ] ) ;
    }
    pos[ ++ M ].oper( p[ n ].x , st[ right ].cal( p[ n ].x ) ) ;
    ld ans = inf ;
    rep( i , M ) ans = min( ans , calheight( pos[ i ].x ) ) ;
    rep( i , n ) ans = min( ans , calheight( p[ i ].x ) ) ;
    printf( "%.3f\n" , double( ans + esp ) ) ;
    return 0 ;
}

相关文章

  • BZOJ-1038: [ZJOI2008]瞭望塔(半平面交)

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

  • 瞭夏半烟尘

    听说,你过得很好。 听说,她美得像画。 听说听说,听说我爱你。

  • [ZJOI2008]树的统计Count

    [ZJOI2008]树的统计Count

  • 半平面交

  • 半平面交

    具体步骤看训练指南 ( 一 )求解半平面交 Uyuw's Concert 类似的题目:hdu 1632 Polyg...

  • 昨日的雨, 湿润整个院落。 阻隔我, 困倦纠缠… 今日阳光, 数落茶花花瓣, 桃花细小, 撩我青丝… 隔绝风, 暖...

  • 背着臃肿的背包 从医院的一楼到十六楼 没有能治我病的房间 所有人都说 “你没病” 彩虹底下迷迷糊糊的雾气里 成群猴...

  • 天高,云淡,望断南飞雁。

  • 盼春

    三月冰城雪将尽 长念芳菲岁岁新 拂帘透纱瞭窗外 半掩枯枝半掩春

  • 时时望塔

    望而可触天,看而未入云。 回神欲转探,意欲无忘尽。

网友评论

    本文标题:BZOJ-1038: [ZJOI2008]瞭望塔(半平面交)

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