占个坑,等自己数学好一点以后再来重新把这个题写一遍

附上链接

附上大牛代码:


#include <stdio.h>
#include <algorithm>
#define INF 99999999
#define ll long long
using namespace std; struct line {
int x, y1, y2;//y1 > y2
}l[10005], id[10005]; //斜率存储
struct vec {
int x, y;
vec(int a = 0, int b = 0):x(a), y(b) {};
}; inline int cross(vec a, vec b) {//伪叉积,大于0就返回1,小于返回-1,等于返回0
ll temp = (ll)(a.x)*(ll)(b.y) - (ll)(a.y)*(ll)(b.x);
if (temp > 0) return 1;
else if (temp == 0) return 0;
return -1;
} int n;
vec krmax, krmin, klmax, klmin; inline bool cmp(const line& a, const line& b) {
return a.x < b.x;
}
inline bool cmp2(const line& a, const line& b) {
return (a.y1 - a.y2) < (b.y1 - b.y2);
} int main() {
scanf("%d", &n);
for (int i = 1; i <= n; i++)
scanf("%d%d%d", &l[i].x, &l[i].y1, &l[i].y2); sort(l + 1, l + n + 1, cmp);//先根据x坐标从小到大排个序
for (int i = 1; i <= n; i++) {
id[i].x = i;//这里的x就是记录竖线ID了(即记录是从左到右的第几根竖线)
id[i].y1 = l[i].y1;
id[i].y2 = l[i].y2;
}
sort(id + 1, id + n + 1, cmp2);//再根据y1-y2从小到大排个序,优先尝试用“短”的竖线下方的端点作“卡子”
for (int idx = 1; idx <= n; idx++) {
int i = id[idx].x;//尝试以从左到右的第i根竖线下方的端点作“卡子”
krmax = vec(1, INF);
krmin = vec(1, -INF);
klmax = vec(-1, -INF);
klmin = vec(-1, INF);
for (int j = i + 1; j <= n; j++) {//向右找允许的最大斜率和最小斜率
vec v1(l[j].x - l[i].x, l[j].y1 - l[i].y2);
vec v2(l[j].x - l[i].x, l[j].y2 - l[i].y2);
if (cross(krmax, v1) < 0) krmax = v1;
if (cross(krmin, v2) > 0) krmin = v2;
}
for (int j = i - 1; j >= 1; j--) {//向左找允许的最大斜率和最小斜率
vec v1(l[j].x - l[i].x, l[j].y1 - l[i].y2);
vec v2(l[j].x - l[i].x, l[j].y2 - l[i].y2);
if (cross(klmin, v1) > 0) klmin = v1;
if (cross(klmax, v2) < 0) klmax = v2;
}
//判断以从左到右的第i根竖线下方的端点作“卡子”,是否可行,不可行直接continue
if (cross(krmax, krmin) > 0) continue;
if (cross(klmax, klmin) > 0) continue;
if (cross(krmax, klmin) < 0) continue;
if (cross(klmax, krmin) < 0) continue;
//可行,分情况讨论
if (cross(klmax, krmax) * cross(klmax, krmin) <= 0)
printf("%d %d %d %d", l[i].x + klmax.x, l[i].y2 + klmax.y, l[i].x, l[i].y2);
else if (cross(klmin, krmax) * cross(klmin, krmin) <= 0)
printf("%d %d %d %d", l[i].x + klmin.x, l[i].y2 + klmin.y, l[i].x, l[i].y2);
else if (cross(klmax, krmax) > 0 && cross(klmin, krmin) < 0)
printf("%d %d %d %d", l[i].x + krmax.x, l[i].y2 + krmax.y, l[i].x, l[i].y2);
else if (cross(krmax, klmax) > 0 && cross(krmin, klmin) < 0)
printf("%d %d %d %d", l[i].x + klmin.x, l[i].y2 + klmin.y, l[i].x, l[i].y2);
break;
}
return 0;
}

PAT L3-012 水果忍者的更多相关文章

  1. 前端优秀作品展示,JavaScript 版水果忍者

    <水果忍者>是一款非常受喜欢的手机游戏,刚看到新闻说<水果忍者>四周年新版要上线了.网页版的切水果游戏由百度 JS 小组开发,采用 vml + svg 绘图,使用了 Rapha ...

  2. 作品展示,JavaScript 版水果忍者

    点这里 <水果忍者>是一款非常受喜欢的手机游戏,刚看到新闻说<水果忍者>四周年新版要上线了.网页版的切水果游戏由百度 JS 小组开发,采用 vml + svg 绘图,使用了 R ...

  3. 基于html5 canvas和js实现的水果忍者网页版

    今天爱编程小编给大家分享一款基于html5 canvas和js实现的水果忍者网页版. <水果忍者>是一款非常受喜欢的手机游戏,刚看到新闻说<水果忍者>四周年新版要上线了.网页版 ...

  4. L3-012 水果忍者 (30 分)

    2010年风靡全球的“水果忍者”游戏,想必大家肯定都玩过吧?(没玩过也没关系啦~)在游戏当中,画面里会随机地弹射出一系列的水果与炸弹,玩家尽可能砍掉所有的水果而避免砍中炸弹,就可以完成游戏规定的任务. ...

  5. Cocos2d-x 水果忍者划痕效果

    网上找的一个关于水果忍者划痕的,效果还算凑合.其原理就是基于OpenGL绘制直线,因为版本号过老,此处笔者改动了一些方法,粘贴后可直接使用 适用于Cocos2d-x 2.2.1 .h文件里须要添�的代 ...

  6. java游戏制作之水果忍者

    水果忍者的原理很简单,主要就是采用随机的方式是画面上面出现水果. package Fruitninja; import java.awt.Dimension; import java.awt.Grap ...

  7. JavaScript实现的水果忍者游戏,支持鼠标操作

    智能手机刚刚普及时,水果忍者这款小游戏可谓风靡一时.几年过去了,现在,让我们用纯JavaScript来实现这个水果忍者游戏,就算是为了锤炼我们的JavaScript开发技能吧. 大家可以通过这个链接在 ...

  8. 团体程序设计天梯赛(CCCC) L3012 水果忍者 上凸或下凹的证明

    团体程序设计天梯赛代码.体现代码技巧,比赛技巧.  https://github.com/congmingyige/cccc_code #include <cstdio> #include ...

  9. 团体程序设计天梯赛 L3-012. 水果忍者

    /*对于一条满足条件的直线,向下移,直到触碰一条线段的下端点,仍然经过其它线段,该直线仍然满足条件 即以一条线段的下(上)端点作为直线上的一点,求为了经过一条线段的最小.最大斜率值(mink,maxk ...

随机推荐

  1. [genome shell]标题栏优化

    参考地址:https://wiki.archlinux.org/index.php/GNOME_(%E7%AE%80%E4%BD%93%E4%B8%AD%E6%96%87)#.E4.BB.8E.E5. ...

  2. [已解决]ubuntu下chrome和firefox输入框内无法快捷键全选

    问题现象: 在chrome或firefox浏览器(其他地方没试)的输入框中使用ctr + a进行全选失效,在google中找到了这个已经解决的http://askubuntu.com/question ...

  3. Flask路由系统与模板系统

    路由系统 @app.route('/user/<username>') @app.route('/post/<int:post_id>') @app.route('/post/ ...

  4. SQLALchemy的其他常用操作

    使用连接池的两种方式 第一种方式: 直接从SessionFactory里获取,此时如果需要开启多个进程,那么创建连接池的代码一定要放在循环里面 不然的话每个进程都是用一个session了 from s ...

  5. WCF服务对于处理客户端连接的一点思考

    对于每个客户端的,服务端是否为每个客户端有专门的“通道”? 目的:想在服务端记录下来客户端的访问记录(进入.各个操作.离开等信息),并将其执行的操作独立记录在各个客户端对应的日志中. 下面是代码: 契 ...

  6. IntelliJ IDEA的几个常用快捷键

    一.将IntelliJ IDEA的快捷键设置为Eclipse环境的快捷键 如果之前长期使用Eclipse作为开发工具的程序员在刚开始接触IDEA的时候肯定会很不习惯,所以如果你没有太多时间去研究的话可 ...

  7. 机器学习第5周--炼数成金-----决策树,组合提升算法,bagging和adaboost,随机森林。

    决策树decision tree 什么是决策树输入:学习集输出:分类觃则(决策树) 决策树算法概述 70年代后期至80年代初期,Quinlan开发了ID3算法(迭代的二分器)Quinlan改迚了ID3 ...

  8. cocoon + carrierwave 多图片上传用法

    gem 'cocoon' gem 'carrierwave' gem 'mini_magick' 1.图片上传carrierwave配置,github 自己手动添加的配置,没用命令生成 在app下新建 ...

  9. 无线路由和无线AP的区别

    一.答疑解惑 1.什么是无线AP? AP:Access Point 接入点 无线AP:无线接入点是一个无线网络的接入点,俗称“热点”.主要有路由交换接入一体设备和纯接入点设备,一体设备执行接入和路由工 ...

  10. java知识框架

    从网上摘录的一张很经典的java学习框架图,和大家分享一下.