P1883 函数

Posted liqgnonqfu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了P1883 函数相关的知识,希望对你有一定的参考价值。

题目大意:

   给你n个二次或一次的函数f1(x),f2(x)...fn(x),构造F(x)=min(f1(x),...,fn(x)),现在求F(x)在[0,1000]的最小值。(T < 10, n ≤ 10000)保留4位小数。

思路:
   感觉是二分。一般情况下,F应该是下凸的,所以二分时保证两端点的导数一正一负就好。

   关键是特殊情况。。。。。。。。。。F有可能单调:

            1.只有一个一次函数bx+c;b小于0时,单减;b大于0时,,单增;

            2.只有一个二次函数,对称轴在[0,1000]外;

            3.多个函数但最终也单调,只需在二分前判断一下r和l的符号是否一致。

更重要的:注意精度!注意精度!注意精度!r-l<=1e-10,-9都过不了。

技术分享图片
  1 #include<cstdio>
  2 #include<cstring>
  3 #include<algorithm>
  4 #include<cstdlib>
  5 
  6 using namespace std;
  7 
  8 int func[10004][3],T,n;
  9 double ans=0;
 10 
 11 
 12 
 13 double qiuzhi(int xuhao,double yuan)
 14 {
 15     double ret,a=func[xuhao][0],b=func[xuhao][1],c=func[xuhao][2];
 16     ret=yuan*yuan*a+b*yuan+c;
 17     return ret;
 18 }
 19 
 20 double qiudao(int xuhao,double yuan)
 21 {
 22     double ret,a=func[xuhao][0],b=func[xuhao][1];
 23     ret=2*a*yuan+b;
 24     return ret;
 25 }
 26 
 27 int work(double x)
 28 {
 29     int ret=1;
 30     double zui=qiuzhi(1,x);
 31     for(int i=2;i<=n;i++)
 32     {
 33         if(zui<qiuzhi(i,x))
 34         {
 35             zui=qiuzhi(i,x);
 36             ret=i;
 37         }
 38     }
 39     return ret;
 40 }
 41 
 42 int main()
 43 {
 44     scanf("%d",&T);
 45     for(int I=1;I<=T;I++)
 46     {
 47         scanf("%d",&n);
 48         ans=0;
 49         for(int i=1;i<=n;i++)
 50         {
 51             scanf("%d%d%d",&func[i][0],&func[i][1],&func[i][2]);
 52         }
 53         if(n==1)
 54         {
 55             if(func[1][0]==0)
 56             {
 57                 if(func[1][1]>=0)printf("%d.0000
",func[1][2]);
 58                 else
 59                 {
 60                     double tt=qiuzhi(1,1000);
 61                     printf("%.4lf
",tt);
 62                 }
 63             }
 64             else 
 65             {
 66                 ans=(-func[1][1]);
 67                 ans=ans/(2*func[1][0]);
 68                 if(ans>=0&&ans<=100)
 69                 {
 70                     ans=(4*func[1][0]*func[1][2]-func[1][1]*func[1][1]);
 71                     ans=ans/(4*func[1][0]);
 72                 }
 73                 else
 74                 {
 75                     ans=min(qiuzhi(1,0),qiuzhi(1,1000));
 76                 }
 77                 printf("%.4lf
",ans);
 78             }
 79             continue;
 80         }
 81         double l=0;
 82         double r=1000;
 83         int x1=work(l);
 84         int x2=work(r);
 85         if(qiudao(x1,0)>0&&qiudao(x2,1000)>0)
 86         {
 87             ans=qiuzhi(x1,0);
 88             printf("%.4lf
",ans);
 89             continue;
 90         }
 91         if(qiudao(x1,0)<0&&qiudao(x2,1000)<0)
 92         {
 93             ans=qiuzhi(x2,1000);
 94             printf("%.4lf
",ans);
 95             continue;
 96         }
 97         while(r-l>1e-10)
 98         {
 99             double mid=(l+r)/2;
100             int x;
101             x=work(mid);
102             double dao=qiudao(x,mid);
103             if(dao>0)
104             {
105                 r=mid;
106             }
107             else
108             {
109                 l=mid;
110             }
111         }
112         ans=l;
113         int x=work(ans);
114         ans=qiuzhi(x,ans);
115         printf("%.4lf
",ans);
116     }
117     return 0;
118 }
View Code

 

 

  


以上是关于P1883 函数的主要内容,如果未能解决你的问题,请参考以下文章

P1883 函数

VSCode自定义代码片段——声明函数

VSCode自定义代码片段8——声明函数

使用从循环内的代码片段中提取的函数避免代码冗余/计算开销

在 Visual Studio 中创建构造函数的代码片段或快捷方式

调用模板化成员函数:帮助我理解另一个 *** 帖子中的代码片段