bzoj1027【JSOI2007】合金

题目描述

  某公司加工一种由铁、铝、锡组成的合金。他们的工作很简单。首先进口一些铁铝锡合金原材料,不同种类的
原材料中铁铝锡的比重不同。然后,将每种原材料取出一定量,经过融解、混合,得到新的合金。新的合金的铁铝
锡比重为用户所需要的比重。 现在,用户给出了n种他们需要的合金,以及每种合金中铁铝锡的比重。公司希望能
够订购最少种类的原材料,并且使用这些原材料可以加工出用户需要的所有种类的合金。


输入格式

  第一行两个整数m和n(m, n ≤ 500),分别表示原材料种数和用户需要的合金种数。第2到m + 1行,每行三
个实数a, b, c(a, b, c ≥ 0 且 a + b + c = 1),分别表示铁铝锡在一种原材料中所占的比重。第m + 2到m +
 n + 1行,每行三个实数a, b, c(a, b, c ≥ 0 且 a + b + c = 1),分别表示铁铝锡在一种用户需要的合金中
所占的比重。


输出格式

  一个整数,表示最少需要的原材料种数。若无解,则输出–1。


  • 题解

    • $a+b+c=1$所以忽略$c$,根据定比分点原则两个点可以表示的点组成它们连线段,推广一下;
    • 所以对$n$个点求出凸包,问题即用$m$个点去圈住凸包求最小点数;
    • 对两个材料$i,j$,如果凸包上的点都在连线$\vec{ij}$的左边就$i$向$j$连边,$floyd$求所有$dis[i][i]$即可;
    • 特判:
    • 当凸包是一个点时特判;
    • 当凸包是一条直线时,如果叉积为$0$还需要盖住所有点才可以加边,用点积特判;
    •  1 #include<bits/stdc++.h>
       2 #define db double
       3 #define inf 0x3f3f3f3f
       4 #define il inline
       5 #define eps 1e-7
       6 using namespace std;
       7 const int N=510;
       8 int n,m,top,dis[N][N];
       9 il int dcmp(db x){return fabs(x)<eps?0:x<0?-1:1;}
      10 struct poi{
      11     db x,y;
      12     poi(db _x=0,db _y=0):x(_x),y(_y){};
      13     il poi operator -(const poi&A)const{return poi(x-A.x,y-A.y);}
      14     il bool operator <(const poi&A)const{return x==A.x?y<A.y:x<A.x;}
      15 }p1[N],p2[N],q[N];
      16 il db crs(poi A,poi B){return A.x*B.y-A.y*B.x;}
      17 il db dot(poi A,poi B){return A.x*B.x+A.y*B.y;}
      18 il bool spj(){
      19     int fg=0;
      20     for(int i=2;i<=n;++i)if(dcmp(p1[1].x-p1[i].x)||dcmp(p1[1].y-p1[i].y)){fg=1;break;}
      21     for(int i=1;i<=m;++i)if(dcmp(p1[1].x-p2[i].x)||dcmp(p1[1].y-p2[i].y)){fg=1;break;}
      22     if(!fg){puts("1");return true;}
      23     else return false;
      24 }
      25 il void convex(){
      26     sort(p2+1,p2+m+1);
      27     q[top=1]=p2[1];
      28     if(m==1)return;
      29     for(int i=2;i<=m;++i){
      30         while(top>1&&dcmp(crs(q[top]-q[top-1],p2[i]-q[top]))<=0)top--;
      31         q[++top]=p2[i];
      32     }
      33     int now=top;
      34     for(int i=m-1;i;--i){
      35         while(top>now&&dcmp(crs(q[top]-q[top-1],p2[i]-q[top]))<=0)top--;
      36         q[++top]=p2[i];
      37     }
      38     top--;
      39 }
      40 il bool judge(int a,int b){
      41     int i; poi p = p1[b]-p1[a];
      42     for(i=1;i<=top;++i){
      43         int c = dcmp(crs(p, q[i]-p1[a]));
      44         if(c>0)break;
      45         if(!c&&dcmp(dot(q[i]-p1[b], q[i]-p1[a]))>0)break;
      46     }
      47     return i==top+1?true:false;
      48 }
      49 int main(){
      50     #ifndef ONLINE_JUDGE
      51     freopen("bzoj1027.in","r",stdin);
      52     freopen("bzoj1027.out","w",stdout);
      53     #endif
      54     db tmp;
      55     scanf("%d%d",&n,&m);
      56     for(int i=1;i<=n;++i)scanf("%lf%lf%lf",&p1[i].x,&p1[i].y,&tmp);
      57     for(int i=1;i<=m;++i)scanf("%lf%lf%lf",&p2[i].x,&p2[i].y,&tmp);
      58     if(spj())return 0;
      59     convex();
      60     memset(dis,0x3f,sizeof(dis));
      61     for(int i=1;i<=n;++i)
      62     for(int j=1;j<=n;++j)if(i!=j){
      63         if(judge(i,j))dis[i][j]=1;
      64     }
      65     for(int k=1;k<=n;++k)
      66     for(int i=1;i<=n;++i)
      67     for(int j=1;j<=n;++j){
      68         if(dis[i][j]>dis[i][k]+dis[k][j])
      69             dis[i][j] =  dis[i][k]+dis[k][j];
      70     }
      71     int ans=inf;
      72     for(int i=1;i<=n;++i)ans = min(ans, dis[i][i]);
      73     if(ans==inf)puts("-1"); else printf("%d\n",ans);
      74     return 0;
      75 }

      bzoj1027

原文地址:https://www.cnblogs.com/Paul-Guderian/p/10269974.html

时间: 2024-10-14 00:22:23

bzoj1027【JSOI2007】合金的相关文章

bzoj千题计划123:bzoj1027: [JSOI2007]合金

http://www.lydsy.com/JudgeOnline/problem.php?id=1027 因为x+y+z=1,所以z=1-x-y 第三维可以忽略 将x,y 看做 平面上的点 简化问题: 若只有两种 材料,那么可以合成 两点线段 上的所有的点 推广到多种材料: 若 用户点 在 材料点 构成的 凸包内,则可以合成 所以题目转化为 求材料点 组成的 能包含所有用户点 的 最小环 用Floyd 枚举 每两对材料点,若所有用户点在 在线段的左侧或在线段上 则 f[i][j]=1 表示i到j

[bzoj1027][JSOI2007]合金

来自FallDream的博客,未经允许,请勿转载,谢谢. 某公司加工一种由铁.铝.锡组成的合金.他们的工作很简单.首先进口一些铁铝锡合金原材料,不同种类的原材料中铁铝锡的比重不同.然后,将每种原材料取出一定量,经过融解.混合,得到新的合金.新的合金的铁铝锡比重为用户所需要的比重. 现在,用户给出了n种他们需要的合金,以及每种合金中铁铝锡的比重.公司希望能够订购最少种类的原材料,并且使用这些原材料可以加工出用户需要的所有种类的合金. n,m<=500 看到这道神题根本没头绪,只好看看题解. 因为三

bzoj 1027[JSOI2007]合金 - floyd + 凸包

1027: [JSOI2007]合金 Time Limit: 4 Sec  Memory Limit: 162 MB Description 某公司加工一种由铁.铝.锡组成的合金.他们的工作很简单.首先进口一些铁铝锡合金原材料,不同种类的原材料中铁铝锡的比重不同.然后,将每种原材料取出一定量,经过融解.混合,得到新的合金.新的合金的铁铝锡比重为用户所需要的比重. 现在,用户给出了n种他们需要的合金,以及每种合金中铁铝锡的比重.公司希望能够订购最少种类的原材料,并且使用这些原材料可以加工出用户需要

[BZOJ 1027][JSOI2007]合金(计算几何+Floyd最小环)

Description 某公司加工一种由铁.铝.锡组成的合金.他们的工作很简单.首先进口一些铁铝锡合金原材料,不同种类的原材料中铁铝锡的比重不同.然后,将每种原材料取出一定量,经过融解.混合,得到新的合金.新的合金的铁铝锡比重为用户所需要的比重. 现在,用户给出了n种他们需要的合金,以及每种合金中铁铝锡的比重.公司希望能够订购最少种类的原材料,并且使用这些原材料可以加工出用户需要的所有种类的合金. Solution 今天考试T3的70分算法似乎是这道原题(然而我没做过QwQ) 思路值得学习一下(

【BZOJ】1027: [JSOI2007]合金(凸包+floyd)

http://www.lydsy.com/JudgeOnline/problem.php?id=1027 没special judge wa了一发好蛋疼... 首先这题大概和黑书上的差不多...由于知道任意两种材料就能得到第三种材料的含量,所以可以忽略第三种含量... 首先来看,如果有两种材料,那么能合成的材料一定在这个线段上,证明很简单... 假设材料A和B,要合成材料C,将他们材料的含量放到二维坐标系上(x轴为材料1,y轴为材料2) 假设用了a的A,则用了1-a的B,有 $$\begin{a

BZOJ 1027 JSOI2007 合金 计算几何+Floyd

题目大意:给定一些合金,选择最少的合金,使这些合金能够按比例合成要求的合金 首先这题的想法特别奇异 看这题干怎么会想到计算几何 并且计算几何又怎么会跟Floyd挂边 好强大 首先因为a+b+c=1 所以我们仅仅要得到a和b就可以 c=1-a-b 所以c能够不读入了 然后我们把每种原料抽象成一个点 可知两个点能合成的合金一定在两点连线的线段上 证明:设两个点为(x1,y1)和(x2,y2),新合成的合金为(ax1+bx2,ay2+by2) (a+b=1,a,b>0) 两点连线为(y-y1)/(x-

[bzoj 1027][JSOI2007]合金(解析几何+最小环)

题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1027 分析: 首先因为一个合金的和为1,所以考虑2个材料合金能否合成一个需求合金的时候,只要考虑前两个值就可以了. 我们如果把这两个值放到平面直角坐标系中,设两个材料合金坐标分别为(x,y)和(m,n),那么易得这两个材料合金可以合成的需求合金对应的点在(x,y)(m,n)两点之间的线段上.(高中数学向量的性质可以证) 那么问题就转化为了:平面上有m个材料合金点,n个需求合金点,要求

[JSOI2007]合金

题目描述 某公司加工一种由铁.铝.锡组成的合金.他们的工作很简单.首先进口一些铁铝锡合金原材料,不同种类的原材料中铁铝锡的比重不同.然后,将每种原材料取出一定量,经过融解.混合,得到新的合金.新的合金的铁铝锡比重为用户所需要的比重. 现在,用户给出了n种他们需要的合金,以及每种合金中铁铝锡的比重.公司希望能够订购最少种类的原材料,并且使用这些原材料可以加工出用户需要的所有种类的合金. 输入输出格式 输入格式: 第一行两个整数m和n(m, n ≤ 500),分别表示原材料种数和用户需要的合金种数.

P4049 [JSOI2007]合金

传送门 我数学可能白学了-- 因为三个数加起来等于\(1\),那么只要用前两个数就能表示,那么就能把每一种金属看成一个二维向量.考虑只有两个向量的时候,设这两个向量为\(a,b\),那么一个向量\(c\)能被表示也就是说存在\(ax+by=c\)且\(x+y=1\),根据数学老师说的那么\(c\)在\(a\)和\(b\)的终点连成的直线上,那么这里因为\(x\)和\(y\)非负所以是在这条线段上.推广一下(我也不知道怎么推广),有\(n\)个向量的时候能表示的范围就在这\(n\)个点的凸包里 于

bzoj 1027: [JSOI2007]合金

由于总和一定所以看前两个即可,可以他们抽象成平面上的点,两个点能合出来的点在以两点为端点的线段上,三个点的在以三点为顶点的三角形内,... 所以预先处理那两个点的连线在所有被需要的点的一边,便把距离设为1,跑一遍最小环就行了. 1 #include<cstdio> 2 #include<iostream> 3 #include<cstring> 4 #include<cstdlib> 5 #include<cmath> 6 #include<