zoukankan      html  css  js  c++  java
  • DP水题大礼包

    DP水题大礼包

    一天8题,6道水题

    birthday

    做法一:考虑在完全背包上加点东西
    做法二:在完全背包第一次转移时加值

    chess

    就是个状压DP模板

    #include<bits/stdc++.h>
    using namespace std ;
    inline int read(){
    	int s=0 ; char g=getchar() ; while(g>'9'||g<'0')g=getchar() ;
    	while(g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s ; 
    }
    int N , M , state[ 4096 ] , field[ 13 ] , mod = 100000000 , dp[ 13 ][ 4096 ] ;
    int main(){
    	freopen("chess.in","r",stdin);
    	freopen("chess.out","w",stdout);
    	N = read() , M = read() ; int m1 ; 
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 1 ; j <= M ; ++j )
    	    	m1 = read() , field[ i ] = ( field[ i ]<<1 ) + (m1^1) ;
    	for( int i = 0 ; i < ( 1<<M ) ; ++i )state[ i ] = ( ( !( (i<<1)&i) )&&( !(i&(i>>1) )) ) ;
    	dp[ 0 ][ 0 ] = 1 ; 
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 0 ; j < (1<<M) ; ++j ){//枚举状态 
    	        if( !state[ j ] )continue ;
    	        if( j&field[ i ] )continue ;
    	        for( int k = 0 ; k < (1<<M) ; ++k ){
    	        	if( k&j )continue ;
    	        	dp[ i ][ j ] = ( dp[ i-1 ][ k ] + dp[ i ][ j ] )%mod;
    	        }
    	    }
    	int ans = 0 ; 
    	for( int i = 0 ; i < ( 1<<M )  ; ++i )ans = (dp[ N ][ i ] + ans)%mod;
    	cout<<ans ;
    }
    
    

    question

    悬线法入门题,不懂得请看ssw02的 悬线法总结

    #include<bits/stdc++.h>
    using namespace std;
    const int MAXN = 1005 ;
    inline int read(){
    	int s=0 ; char  g=getchar() ; while( g>'9'||g<'0' )g=getchar() ;
    	while( g>='0'&&g<='9')s=s*10+g-'0',g=getchar(); return s ; 
    }
    int N , M , a[ MAXN ][ MAXN ] , flag = 0 ;
    int l[ MAXN ][ MAXN ] , r[ MAXN ][ MAXN ] , up[ MAXN ][ MAXN ] , ans = 0 ;
    int main(){
    	freopen("question.in","r",stdin);
    	freopen("question.out","w",stdout);
    	N = read() , M = read() ; 
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 1 ; j <= M ; ++j )
    		    a[ i ][ j ] = read() , l[ i ][ j ] = r[ i ][ j ] = j , up[ i ][ j ] = 1 , flag = max(flag,a[i][j]) ; 
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 2 ; j <= M ; ++j )
    	    	if( a[i][j] && a[i][j-1] )l[i][j] = l[i][j-1] ;
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = M-1 ; j >= 1 ; --j )
    		    if( a[i][j] && a[i][j+1] )r[i][j] = r[i][j+1] ;
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 1 ; j <= M ; ++j ){
    		    if( i > 1 )
    		    	if( a[i][j] && a[i-1][j] ){
    		    		l[i][j] = max( l[i][j] , l[i-1][j] ) ;
    		    		r[i][j] = min( r[i][j] , r[i-1][j] ) ;
    		    		up[i][j] = up[i-1][j] + 1 ; 
    		    	}
    		    int lon = r[i][j] - l[i][j] + 1 ;
    		    ans = max( ans , lon*up[i][j] ) ;
    	    }
    	if( !flag )cout<<0;
    	else cout<<ans; 
    }
    
    

    road

    DAG上DP , 按拓扑排序保证无后效性即可

    #include<bits/stdc++.h>
    using namespace std ;
    const int MAXN = 100005 ; 
    #define ll long long
    inline int read(){
    	int s=0,w=1 ; char g=getchar() ; while(g>'9'||g<'0'){if(g=='-')w=-1;g=getchar();} ;
    	while(g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s*w ; 
    }
    int N , M , tot = 1 , head[ MAXN ] , to[ MAXN*20 ] , nex[ MAXN*20 ] , a[ MAXN ] , du[ MAXN ] , ru[ MAXN ] ;
    ll dis[ MAXN ] ;
    bool vis[ MAXN ] ;
    queue<int>q ;
    void add( int x , int y ){
    	to[ ++tot ] = y , nex[ tot ] = head[ x ] , head[ x ] = tot ;
    }
    void  toposort(){
    	while( !q.empty() ){
    		int u = q.front() ; q.pop() ; vis[ u ] = false ;
    		for( int i = head[ u ] ; i ; i = nex[ i ] ){
    			dis[ to[i] ] = max( dis[ u ]+(ll)a[ to[i] ] , dis[ to[i] ] ) ;
    			du[ to[i] ]-- ; 
    			if( du[ to[i] ]==0 && vis[ to[i] ]==0 ){
    				q.push( to[i] ) ; vis[ to[i] ] = true ; 
    			}
    		}
    	}
    }
    int main(){
    	freopen("road.in","r",stdin);
    	freopen("road.out","w",stdout);
    	N = read() , M = read() ; int m1 , m2 ;
    	for( int i = 1 ; i <= N ; ++i )a[ i ] = read() , dis[ i ] = -2000000005  ; 
    	for( int i = 1 ; i <= M ; ++i ){
    		m1 = read() , m2 = read() , du[ m1 ]++ , ru[ m2 ]++ ;
    		add( m2 , m1 ) ;
    	}
    	for( int i = 1 ; i <= N ; ++i )
    	   if( !du[ i ] ){
    	   	  vis[ i ] = true , dis[ i ] = (ll)a[ i ] ; q.push( i ) ;
    	   }
    	toposort() ;
    	ll ans = -2000000005 ;
    	for( int i = 1 ; i <= N ; ++i )
    	   if( !ru[ i ] )
    	       ans = max( ans , dis[ i ] ) ;
    	cout<<ans ; 
    }
    
    

    Gray

    我们考虑到,每一位的影响只和当前位置的字符和先前一位的字符有关,然后就是分情况的线性DP

    #include<bits/stdc++.h>
    using namespace std ;
    #define ll long long
    inline ll read(){
    	ll s=0,w=1 ; char g=getchar() ; while(g>'9'||g<'0'){if(g=='-')w=-1;g=getchar();}
    	while(g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s*w ;
    }
    char s[ 200005 ] ;
    ll  N ,  val[ 200005 ] , dp[ 200005 ][ 2 ] ;
    inline ll max( ll x , ll y ){
    	if( x > y )return x ;
    	else return y ; 
    }
    void  updata( int i , int x ){
    	if( x ){
    		if( dp[ i-1 ][ 0 ] != -1 )dp[ i ][ 1 ] = dp[ i-1 ][ 0 ] + val[ i ] ;
    		if( dp[ i-1 ][ 1 ] != -1 )dp[ i ][ 1 ] = max( dp[ i ][ 1 ] , dp[ i-1 ][ 1 ] ) ;
    		return ;
    	}
    	if( dp[ i-1 ][ 1 ] != -1 )dp[ i ][ 0 ] = dp[ i-1 ][ 1 ] + val[ i ] ;
    	if( dp[ i-1 ][ 0 ] != -1 )dp[ i ][ 0 ] = max( dp[ i ][ 0 ] , dp[ i-1 ][ 0 ] ) ;
    } 
    int main(){
    	freopen("gray.in","r",stdin);
    	freopen("gray.out","w",stdout);
    	scanf("%s",s) ;
    	N = strlen(s) ; dp[ 0 ][ 1 ] = -1 ;
    	for( int i = 1 ; i <= N ; ++i )val[ i ] = read() , dp[ i ][ 1 ] = dp[ i ][ 0 ] = -1e10 ;//坑人 
    	for( int i = 1 ; i <= N ; ++i ){
    		if( s[i-1] == '0' ){
    			dp[ i ][ 1 ] = -1 ;
    			updata( i , 0 ) ;
    		}
    		else if( s[i-1] == '1' ){
    			dp[ i ][ 0 ] = -1 ;
    			updata( i , 1 ) ;
    		}
    		else{
    			updata( i , 0 ) ;
    			updata( i , 1 ) ; 
    		}
    	}
    	cout<<max( dp[ N ][ 1 ] , dp[ N ][ 0 ] ) ;
    	/*for( int i = 1 ; i <= N ; ++i )cout<<val[ i ]<<" ";cout<<endl ;
    	for( int i = 1 ; i <= N ; ++i )cout<<dp[ i ][ 0 ]<<" " ;cout<<endl ;
    	for( int i = 1 ; i <= N ; ++i )cout<<dp[ i ][ 1 ]<<' ' ;*/
    	return 0 ;
    }
    

    number

    RMQ问题,线段树或者树状数组或者ST表维护(留个ST表板子)

    #include<bits/stdc++.h>
    using namespace std ;
    const int MAXN = 100005 ;
    int N , M , a[ MAXN ] , st[ MAXN ][ 20 ] , log_a[ MAXN ] ;
    inline int read(){
    	int s=0 ,w=1; char g=getchar() ; while(g>'9'||g<'0'){if(g=='-')w=-1;g=getchar();}
    	while(g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s*w ;
    }
    void  prepare(){
    	for( int i = 1 ; i <= 17 ; ++i )
    	    for( int u = 1 ; u + (1<<i) - 1 <= N ; ++u )
    	        st[ u ][ i ] = min( st[ u ][ i-1 ] , st[ u + ( 1<<i-1 ) ][ i-1 ]  ) ; 
    }
    int  ask( int l , int r ){
        int k = log_a[ r-l+1 ] ; 
        return min( st[ l ][ k ] , st[ r-(1<<k)+1 ][ k ] ) ;
    }
    int main(){
    	freopen("number.in","r",stdin);
    	freopen("number.out","w",stdout);
        N = read() ;  int m1 , m2 ;  log_a[0]=-1;
        for( int i = 1 ; i <= N ; ++i )st[ i ][ 0 ] = read() , log_a[i]=log_a[ i>>1 ] +1 ;
    	M = read() ;prepare() ;
        for( int i = 1 ; i <= M ; ++i ){
        	m1 = read() , m2 = read() ; 
        	printf("%d
    ",ask(m1,m2) ) ;
        }
        return 0 ; 
    }
    
    

    hotel

    好题,我们考虑到从每个房间数的人并不好维护,然后考虑到经过的人只能是从左右两个位置进入和出去

    具体写代码里了

    #include<bits/stdc++.h>
    using namespace std ;
    #define ll long long
    const int MAXN = 1e5+5 ;
    inline ll read(){
    	ll s=0 ; char g=getchar() ; while(g>'9'||g<'0')g=getchar() ;
    	while( g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s ;
    }
    ll N , dp[ MAXN ][ 15 ] , a[ MAXN ] ;
    inline ll min( ll x , ll y ){
    	return ( x<y )?x:y ;
    }  
    inline bool check( int x ){
    	if( x==0||x==4||x==7 )return true ;
    	else return false ; 
    }
    int main(){// dp[i][0 ~ 14]表示从1~i再向外走出去-7(负数就是进来)~7个人就能使1~i房间合法的最小费用.
    	freopen("hotel.in","r",stdin);
    	freopen("hotel.out","w",stdout);
    	N = read() ;
    	for( int i = 1 ; i <= N ; ++i )a[ i ] = read() ;
    	for( int i = 0 ; i <= N ; ++i )
    	    for( int j = 0 ; j <= 14 ; ++j )dp[ i ][ j ] = 1e10 ; 
    	dp[ 0 ][ 7 ] = 0 ;
    	for( int i = 1 ; i <= N ; ++i )
    	    for( int j = 0 ; j <= 14 ; ++j )
    	        for( int k = 0 ; k <= 14 ; ++k ){
    	        	int in = j - 7 , out = k - 7 ;//存在负值 7为基数位 
    				if( check( a[ i ] + in - out ) ){
    					dp[ i ][ k ] = min( dp[ i ][ k ] , dp[ i-1 ][ j ] + abs(in) ) ;
    				}
    	        }
    	cout<<( (dp[ N ][ 7 ]==1e10 )?-1:dp[ N ][ 7 ] ) ;
    }
    

    gift

    区间DP , 移项后将 dp[ i ][ k ] - k * val[ i ]丢到单调队列中维护

    #include<bits/stdc++.h>
    using namespace std ;
    const int MAXN = 16005 ;
    #define ll long long
    inline ll read(){
    	ll s=0,w=1 ; char g=getchar() ; while(g>'9'||g<'0'){if(g=='-')w=-1;g=getchar();}
    	while(g>='0'&&g<='9')s=s*10+g-'0',g=getchar() ; return s*w ;
    }
    ll N , M , ans , dp[ 105 ][ MAXN ] , q[ MAXN ] ;
    struct ap{
    	ll l , val , pre ;
    }t[ 105 ] ;
    inline bool cmp( ap x , ap y ){
    	return x.pre < y.pre ;
    }
    inline ll max( ll x , ll y ){
    	return ( x > y )?x:y ;
    }
    int main(){
    	freopen("gift.in","r",stdin) ;
    	freopen("gift.out","w",stdout);
    	N = read() , M = read() ; 
    	for( int i = 1 ; i <= M ; ++i )
    	    t[ i ].l = read() , t[ i ].val = read() , t[ i ].pre = read() ;
    	sort( t+1 , t+M+1 , cmp ) ;
    	for( int i = 1 ; i <= M ; ++i ){
    		int head = 1 , tail = 0 ;
    		q[ ++tail ] = max( t[ i ].pre - t[ i ].l , 0 ) ;
    		for(int j = 1; j <= N; ++j ){
                dp[i][j] = max( dp[i - 1][j], dp[i][j - 1] ) ;//不放 
                if( j >= t[ i ].pre + t[ i ].l )continue ;//放 
                while( head <= tail && q[ head ] + t[ i ].l < j )++head ;
                if( j < t[ i ].pre ){
                    int tmp = dp[ i-1 ][ j ] - j*t[ i ].val ;//移项后  dp[ i-1 ][ k ] - k*val[ i ] 
                    while( head <= tail && tmp > dp[ i-1 ][ q[tail] ] - t[i].val*q[tail] )--tail ;//cz 
                    q[ ++tail ] = j ; continue ;
                }
                else dp[i][j] = max( dp[i][j] , dp[ i-1 ][ q[head] ] + (j-q[head])*t[i].val ) ;
            }
    	}    
    	cout<<dp[M][N] ;
    }
    
  • 相关阅读:
    小程序 筛选
    Travel 项目环境配置
    ajax
    vue 项目编译打包
    自学网
    使用npm打包vue项目
    vue音乐播放器项目 二级路由跳转
    better-scroll (下拉刷新、上拉加载)
    Linux命令
    hibernate存储过程 3
  • 原文地址:https://www.cnblogs.com/ssw02/p/11640159.html
Copyright © 2011-2022 走看看