• 【LOJ】#2443. 「NOI2011」智能车比赛


    题解

    显然是个(n^2)的dp

    我们要找每个点不穿过非赛道区域能到达哪些区域的交点

    可以通过控制两条向量负责最靠下的上边界,和最靠上的下边界,检查当前点在不在这两条向量之间即可,对于每个点可以(O(n))求出来哪些点是可以到达的

    之后dp即可

    注意判断S点所在区域的时候需要找靠后的那个区域……

    代码

    #include <iostream>
    #include <algorithm>
    #include <cstdio>
    #include <cstring>
    #include <queue>
    #include <cmath>
    #define enter putchar('
    ')
    #define space putchar(' ')
    #define mp make_pair
    #define pb push_back
    #define fi first
    #define se second
    #define eps 1e-7
    //#define ivorysi
    using namespace std;
    typedef long long int64;
    typedef double db;
    template<class T>
    void read(T &res) {
    	res = 0;char c = getchar();T f = 1;
    	while(c < '0' || c > '9') {
    		if(c == '-') f = -1;
    		c = getchar();
    	}
    	while(c >= '0' && c <= '9') {
    		res = res * 10 + c - '0';
    		c = getchar();
    	}
    	res *= f;
    }
    template<class T>
    void out(T x) {
    	if(x < 0) {putchar('-');x = -x;}
    	if(x >= 10) {
    		out(x / 10);
    	}
    	putchar('0' + x % 10);
    }
    int N;
    struct Point {
    	db x,y;
    	Point() {}
    	Point(db _x,db _y) {
    		x = _x;y = _y;
    	}
    	friend Point operator + (const Point &a,const Point &b) {
    		return Point(a.x + b.x,a.y + b.y);
    	}
    	friend Point operator - (const Point &a,const Point &b) {
    		return Point(a.x - b.x,a.y - b.y);
    	}
    	friend Point operator * (const Point &a,const db d) {
    		return Point(a.x * d,a.y * d);
    	}
    	friend Point operator / (const Point &a,const db d) {
    		return Point(a.x / d,a.y / d);
    	}
    	friend db operator * (const Point &a,const Point &b) {
    		return a.x * b.y - a.y * b.x;
    	}
    }S,T,U[2005],D[2005],pos[2005][2];
    db dis(Point a,Point b) {
    	return sqrt((a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y));
    }
    db v,ans,dp[2005][2];
    bool vis[2005][2];
    void Init() {
    	read(N);
    	for(int i = 1 ; i <= N ; ++i) {
    		scanf("%lf%lf%lf%lf",&D[i].x,&D[i].y,&U[i].x,&U[i].y);
    	}
    	scanf("%lf%lf",&S.x,&S.y);
    	scanf("%lf%lf",&T.x,&T.y);
    	scanf("%lf",&v);
    	if(S.x > T.x) swap(S,T);
    	for(int i = 2 ; i <= N ; ++i) {
    		pos[i][0] = Point(D[i].x,max(D[i].y,D[i - 1].y));
    		pos[i][1] = Point(D[i].x,min(U[i].y,U[i - 1].y));
    	}
    }
    db dfs(int p,int c) {
    	if(vis[p][c]) return dp[p][c];
    	vis[p][c] = 1;
    	Point M = pos[p][c];
    	if((T.x >= D[p].x && T.x < U[p].x) || (p == N)) return dp[p][c] = dis(M,T);
    	Point Up = pos[p + 1][1] - M,Dw = pos[p + 1][0] - M;
    	dp[p][c] = dfs(p + 1,1) + dis(M,pos[p + 1][1]);
    	dp[p][c] = min(dp[p][c],dfs(p + 1,0) + dis(M,pos[p + 1][0]));
    	for(int i = p + 2; i <= N ; ++i) {
    		if(D[i].x > T.x) break;
    		if(Dw * (pos[i][0] - M) >= 0 && (pos[i][0] - M) * Up >= 0) {
    			dp[p][c] = min(dp[p][c],dfs(i,0) + dis(M,pos[i][0]));
    		}
    		if(Dw * (pos[i][1] - M) >= 0 && (pos[i][1] - M) * Up >= 0) {
    			dp[p][c] = min(dp[p][c],dfs(i,1) + dis(M,pos[i][1]));
    		} 
    		if(Dw * (pos[i][0] - M) >= 0) Dw = pos[i][0] - M;
    		if((pos[i][1] - M) * Up >= 0) Up = pos[i][1] - M;
    	}
    	if(Dw * (T - M) >= 0 && (T - M) * Up >= 0) {
    		return dp[p][c] = dis(M,T);
    	}
    	return dp[p][c];
    }
    void Solve() {
    	for(int i = 1 ; i <= N ; ++i) {
    		if(U[i].x <= S.x) continue;
    		pos[i][0] = S;
    		ans = dfs(i,0);	
    		break;
    	}
    	if(ans == 0) ans = dis(S,T);
    	printf("%.6lf
    ",ans / v);
    }
    int main() {
    #ifdef ivorysi
    	freopen("f1.in","r",stdin);
    #endif
    	Init();
    	Solve();
    }
    

    放假了真是颓啊QwQ

  • 相关阅读:
    委托与事件的关系
    分布式存储ceph——(1)部署ceph
    neutron二
    openstack第五章:cinder
    openstack第六章:dashboard
    openstack第一章:keystone
    openstack第二章:glance
    openstack第三章:nova
    openstack第四章:neutron— 网络服务
    openstack安装
  • 原文地址:https://www.cnblogs.com/ivorysi/p/9195925.html
Copyright © 2020-2023  润新知