【Link】:

【Description】



给你一个圆和圆周上的n(3≤n≤40)个不同点。请选择其中的m(3≤m≤n)个,按照在圆 周上的顺序连成一个m边形,使得它的面积最大。

【Solution】



DP;

设f[i][j][k]表示在第i到第j个点之间一定选择了i和j的条件下选择了k个点组成的多边形的最大面积;

这样表示状态;在增加一个点的时候;

只要算i..j和新加的那个点组成的三角形面积,然后把这个三角形的面积加上去就好了;

逆推的话;

f[i][j][k] = f[i][l][k-1] + area(i,l,j)

l∈[i..j-1];

当k<3的时候,f[i][j][k]=0;

其中area(i,l,j)表示点i,l,j组成的三角形的面积;

面积可以用海伦公式搞;

其中两点之间的弦长为2∗sin(a2)

则用题中给的东西表示就为

2∗sin(|p[i]−p[j]|∗π)

这样就能求出三角形的三边了;



【NumberOf WA】



0



【Reviw】



状态表示得清晰一点,DP就不难写了;



【Code】

#include <bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define ms(x,y) memset(x,y,sizeof x)
#define ri(x) scanf("%d",&x)
#define rl(x) scanf("%lld",&x)
#define rs(x) scanf("%s",x+1)
#define rd(x) scanf("%lf",&x)
#define oi(x) printf("%d",x)
#define ol(x) printf("%lld",x)
#define oc putchar(' ')
#define os(x) printf(x)
#define all(x) x.begin(),x.end()
#define Open() freopen("F:\\rush.txt","r",stdin)
#define Close() ios::sync_with_stdio(0) typedef pair<int,int> pii;
typedef pair<LL,LL> pll; const int dx[9] = {0,1,-1,0,0,-1,-1,1,1};
const int dy[9] = {0,0,0,-1,1,-1,1,-1,1};
const double pi = acos(-1.0);
const int N = 40;
const int INF = 0x3f3f3f3f; int n,m;
double p[N+10],f[N+10][N+10][N+10]; double dis(int i,int j){
double tp = abs(p[j]-p[i]);
if (tp>0.5) tp = 1 - tp;
return 2*sin(tp*pi);
} double area(int i,int j,int k){
double a = dis(i,j),b = dis(i,k),c = dis(j,k);
double temp = (a+b+c)/2.0;
return sqrt(temp*(temp-a)*(temp-b)*(temp-c));
} double dfs(int l,int r,int num){
if (f[l][r][num]>=0) return f[l][r][num];
if (num < 3) return 0;
double temp = 0;
rep1(i,l,r-1)
temp = max(temp,dfs(l,i,num-1) + area(l,i,r));
return f[l][r][num] = temp;
} int main(){
//Open();
//Close();
while (~ri(n)){
ri(m);
if (n == 0 && m == 0) break;
rep1(i,1,n) rd(p[i]);
rep1(i,1,n)
rep1(j,1,n)
rep1(k,1,n)
f[i][j][k] = -1;
double ans = 0;
rep1(i,1,n)
rep1(j,i+1,n)
ans = max(ans,dfs(i,j,m));
printf("%.6f\n",ans);
}
return 0;
}

最新文章

  1. T-SQL:毕业生出门需知系列(九)
  2. 去除inline-block元素间间距
  3. java 配置文件读取
  4. iOS----CocoaPods的安装、使用和,原理+参考流程+常见问题
  5. 小白初学ABP框架,着实累啊
  6. 边工作边刷题:70天一遍leetcode: day 79
  7. Windows下安装Cygwin
  8. shell -Z- d等等代表
  9. django分页工具
  10. 通过jquery获取后台传过来的值进行全选
  11. C#关于使用枚举遇到的问题----Type运算符使用的必要性
  12. Spring MVC 和Struts2对比
  13. JVM基础
  14. (网页)SQLserver中在上线的项目中遇到科学计数法怎么办?
  15. 洛谷UVA12995 Farey Sequence(欧拉函数,线性筛)
  16. python 列表推导
  17. C++ 简单实现MFC ListControl 点击列头排序
  18. [CF587F]Duff is Mad[AC自动机+根号分治+分块]
  19. PostgreSQL远程代码执行漏洞(CVE-2018-1058)学习笔记
  20. C#List的创建例程

热门文章

  1. 【Codeforces Round #420 (Div. 2) B】Okabe and Banana Trees
  2. Java基础学习总结(16)——Java制作证书的工具keytool用法总结
  3. HDU Victor and World (最短路+状态压缩)
  4. oracle 10g/11g RAC 启停归档模式
  5. 前端编程提高之旅(十二)----position置入值应用
  6. pip报错
  7. spring之DelegatingFilterProxy
  8. 洛谷P3807 【模板】卢卡斯定理exgcd
  9. OpenGL编程逐步深入(四)Shaders
  10. Servlet doPost方法同时上传图片和传递参数