题面传送门

好题。

主要思路和另一位巨佬差不多,详细讲一下判断的部分。

解决思路:

首先考虑本题与拓扑排序有和关系。可以想到,某些棍子的先后移动顺序是有限制的。比如:

这里红色的必须比蓝色的先移动,因为它们在 \(x\) 轴的投影有重叠,蓝色在上,会被红色卡住。

所以,棍子两两之间可能存在限制关系,这就符合拓扑排序的条件了。考虑根据每一对限制关系建边。若 \(u\) 必须比 \(v\) 先移动,就从 \(u\) 向 \(v\) 连边,这样就转化为求拓扑序问题了。


其次,也是较麻烦的一部分,就是如何根据两线段的坐标判断其移动先后限制。

为了方便,在读入时判断并交换好,用 \(x1,y1\) 表示左边端点,\(x2,y2\) 表示左边端点。

\(\text {check}\) 函数,分以下几种情况讨论:

  1. 没有限制关系,返回 \(0\)。
  2. \(u\) 比 \(v\) 要先移动,返回 \(-1\)。
  3. \(v\) 比 \(u\) 要先移动,返回 \(1\)。

为了方便,设 \(u\) 为靠左的线段,若不是,在开始判断前将交换一下,并需要把 \(op\)(返回值)取反。

首先看 \(op=0\),即两线段在 \(x\) 轴上投影不重合:

肉眼可见,\(u.x2<v.x1\),注意等号不可以取到(照提交意思来看...)。

然后看一般情况。多画几个图,可以发现,只需要比较 \(u\) 上 \(x=v.x1\) 时 \(u.y'\) 的值与 \(v.y1\) 的大小 即可(或 \(v\) 上 \(x=u.x2\) 时 \(v.y'\) 的值与 \(u.y2\) 的大小)。在下的先移,在上的后移。

至于如何求函数值。。上过初中数学都会。具体可以看程序,变量名都遵从 \(y=kx+b\) 的基本形式了。

然鹅,这样写获得了 \(95\) 分的高分。哪里出问题了?

还有一种比较坑的情况,就是 \(u\) 是竖直的!

这时候函数 \(u\) 的 \(k\) 是无限大的,不是一次函数,无法求出值。所以需要特判,算出 \(x=u.x1\) 时 \(v\) 的函数值再比较。


Code:

#include<bits/stdc++.h>
using namespace std;
int n,m,in[5005];
struct node{
int x1,x2,y1,y2;
}b[5005];
vector<int> a[5005];
queue<int> q;
int check(node u,node v){ //0:无关,-1:先移u,1:先移v
int op=1;
if(u.x1>v.x1) swap(u,v),op=-op;
if(u.x2<v.x1) return 0;
double K,B,tmp;
if(!(u.x2-u.x1)){
K=1.0*(v.y2-v.y1)/(v.x2-v.x1);
B=(double)v.y1-K*v.x1;
tmp=K*u.x1+B;
if(u.y1>tmp) return op;
return -op;
}
K=1.0*(u.y2-u.y1)/(u.x2-u.x1);
B=(double)u.y1-K*u.x1;
tmp=K*v.x1+B; //求函数
if(tmp>v.y1) return op;
return -op;
}
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d%d%d%d",&b[i].x1,&b[i].y1,&b[i].x2,&b[i].y2);
if(b[i].x1>b[i].x2) swap(b[i].x1,b[i].x2),swap(b[i].y1,b[i].y2);
}
for(int i=1;i<n;i++){
for(int j=i+1;j<=n;j++){
int op=check(b[i],b[j]);
if(op==-1) a[i].push_back(j),in[j]++;
if(op==1) a[j].push_back(i),in[i]++; //连边
}
}
for(int i=1;i<=n;i++) if(!in[i]) q.push(i),printf("%d ",i);
while(q.size()){ //拓扑
int k=q.front();
q.pop();
for(int i=0;i<a[k].size();i++){
int tmp=a[k][i];
in[tmp]--;
if(!in[tmp]) q.push(tmp),printf("%d ",tmp);
}
}
return 0;
}

最新文章

  1. ASP.NET 5 和Entity Framework 7公告仓库
  2. 【转载】LoadRunner添加windows多台压力机
  3. backgroundworker的应用
  4. ActiveMQ点对点的消息发送案例
  5. hiho_1067_最近公共祖先2
  6. .NET学习笔记(4) — C#数据类型
  7. 《JavaScript高级程序设计》 阅读计划
  8. OC基础 可变字典与不可变字典的使用
  9. 洛谷 [P4011] 孤岛营救问题
  10. MyBatis(二):Select语句传递参数的集中方案
  11. 「JavaScript面向对象编程指南」对象
  12. p,np,npc,np难问题,确定图灵机与非确定图灵机
  13. 1.struts 防止表单重复提交 2. 拦截器
  14. 应当将指针变量用“==”或“!=”与 NULL 比较
  15. 亲热接触Redis-第二天(Redis Sentinel)
  16. 常用RGB颜色表
  17. Unit05: JavaScript对象概述 、 常用内置对象一 、 常用内置对象二 、 常用内置对象三
  18. 将中国标准时间转成yyyy-MM-dd
  19. scala中同步块
  20. php数组&#183;的方法3-数组和变量之间的转换

热门文章

  1. 【Java】学习路径51-线程组
  2. (最简单详细)IronPython下载、安装及简单使用
  3. Linux之Samba服务器搭建
  4. LibTorch | 使用神经网络求解一维稳态对流扩散方程
  5. 跟羽夏学 Ghidra ——引用
  6. 使用NextCloud搭建私有网络云盘并支持Office文档在线预览编辑以及文件同步
  7. 将 Docker Engine 节点从 dockershim 迁移到 cri-dockerd
  8. 查找Linux下的大目录或文件
  9. 16. 综合使用tail、forward、copy和stdout
  10. 案例分享 生产环境逐步迁移至k8s集群 - pod注册到consul