注意等边三角形的上顶点是卡不到边界上的。

于是整个凸包分成三部分:左边的连续的三角形、中间的、右边的连续的三角形。

套个计算几何板子求个三角形顶点到圆的切线、三角形顶点到正方形左上角距离啥的就行了,分类比较多。

#include<cstdio>
#include<cmath>
using namespace std;
const double PI=acos(-1.0);
int n;
char a[25];
struct Point{
double x,y;
double length(){
return sqrt(x*x+y*y);
}
};
typedef Point Vector;
Vector operator - (const Point &a,const Point &b){
return (Vector){a.x-b.x,a.y-b.y};
}
Vector operator + (const Vector &a,const Vector &b){
return (Vector){a.x+b.x,a.y+b.y};
}
Vector operator * (const double &K,const Vector &v){
return (Vector){K*v.x,K*v.y};
}
Vector Rotate(Vector A,double rad)
{
return (Vector){A.x*cos(rad)-A.y*sin(rad),A.x*sin(rad)+A.y*cos(rad)};
}
Vector unit(Vector A){
double l=A.length();
return (Vector){A.x/l,A.y/l};
}
double sqr(double x){
return x*x;
}
int main(){
scanf("%d%s",&n,a+1);
bool alltr=1;
for(int i=1;i<=n;++i){
if(a[i]!='T'){
alltr=0;
break;
}
}
if(alltr){
printf("%d\n",2*n+1);
}
else{
double ans=0;
int I,J;
for(int i=1;i<=n;++i){
if(a[i]!='T'){
if(i!=1){
if(a[i]=='S'){
ans+=((Point){0.5,sqrt(3.0)/2.0}-(Point){(double)(i-1),1.0}).length();
}
else if(a[i]=='C'){
Vector v=(Point){(double)i-0.5,0.5}-(Point){0.5,sqrt(3.0)/2.0};
ans+=sqrt(sqr(v.length())-0.5*0.5);
v=Rotate(v,asin(0.5/v.length()));
Point p=(Point){0.5,sqrt(3.0)/2.0}+ans*unit(v);
double xian=((Vector){(double)i-0.5,1.0}-p).length();
double jiao=acos((sqr(xian)-0.5*0.5*2.0)/(-2.0*0.5*0.5));
ans+=jiao*0.5;
}
}
else{
if(a[i]=='S'){
ans+=2.0;
}
else if(a[i]=='C'){
ans+=PI*0.5;
}
}
I=i;
break;
}
}
for(int j=n,i=1;j>=1;++i,--j){
if(a[j]!='T'){
if(j!=n){
if(a[j]=='S'){
ans+=((Point){0.5,sqrt(3.0)/2.0}-(Point){(double)(i-1),1.0}).length();
}
else if(a[j]=='C'){
Vector v=(Point){(double)i-0.5,0.5}-(Point){0.5,sqrt(3.0)/2.0};
double d=sqrt(sqr(v.length())-0.5*0.5);
ans+=d;
v=Rotate(v,asin(0.5/v.length()));
Point p=(Point){0.5,sqrt(3.0)/2.0}+d*unit(v);
double xian=((Vector){(double)i-0.5,1.0}-p).length();
double jiao=acos((sqr(xian)-0.5*0.5*2.0)/(-2.0*0.5*0.5));
ans+=jiao*0.5;
}
}
else{
if(a[j]=='S'){
ans+=2.0;
}
else if(a[j]=='C'){
ans+=PI*0.5;
}
}
J=j;
break;
}
}
if(I!=1 && J!=n){
ans+=((double)(I-1)+1.0);
ans+=((double)(n-J)+1.0);
ans+=(double)(J-I+1);
if(a[I]=='S' && a[J]=='S'){
ans+=(double)(J-I+1);
}
else if(a[I]=='C' && a[J]=='C'){
ans+=(double)(J-I);
}
else{
ans+=((double)(J-I)+0.5);
}
}
else if(I==1 && J==n){
ans+=(double)(n-1)*2.0;
}
else if(I==1 && J!=n){
ans+=((double)(n-J)+1.0);
if(a[J]=='S'){
ans+=((double)J-0.5)*2.0;
}
else{
ans+=(((double)J-0.5)*2.0-0.5);
} }
else{
ans+=((double)(I-1)+1.0);
if(a[I]=='S'){
ans+=((double)(n-I+1)-0.5)*2.0;
}
else{
ans+=(((double)(n-I+1)-0.5)*2.0-0.5);
}
}
printf("%.11f\n",ans);
}
return 0;
}

【计算几何】【分类讨论】Gym - 101173C - Convex Contour的更多相关文章

  1. dp+分类讨论 Gym 101128E

    题目链接:http://codeforces.com/gym/101128 感觉这个人写的不错的(我只看了题目大意):http://blog.csdn.net/v5zsq/article/detail ...

  2. Bzoj4558:分类讨论 计算几何 组合数学

    国际惯例的题面: 这题让我爆肝啦......这种计数显然容斥,正好不含任何坏点的我们不会算,但是我们能算至少含零个坏点的,至少含一个坏点的,至少含两个坏点的......所以最终的答案就是(至少含零个坏 ...

  3. Codeforces 460D Little Victor and Set --分类讨论+构造

    题意:从区间[L,R]中选取不多于k个数,使这些数异或和尽量小,输出最小异或和以及选取的那些数. 解法:分类讨论. 设选取k个数. 1. k=4的时候如果区间长度>=4且L是偶数,那么可以构造四 ...

  4. BZOJ-1067 降雨量 线段树+分类讨论

    这道B题,刚的不行,各种碎点及其容易忽略,受不鸟了直接 1067: [SCOI2007]降雨量 Time Limit: 1 Sec Memory Limit: 162 MB Submit: 2859 ...

  5. UVaLive 6862 Triples (数学+分类讨论)

    题意:给定一个n和m,问你x^j + y^j = z^j 的数量有多少个,其中0 <= x <= y <= z <= m, j = 2, 3, 4, ... n. 析:是一个数 ...

  6. 枚举(分类讨论):BZOJ 1177: [Apio2009]Oil

    1177: [Apio2009]Oil Time Limit: 15 Sec  Memory Limit: 162 MBSubmit: 1477  Solved: 589[Submit] Descri ...

  7. Educational Codeforces Round 63 (Rated for Div. 2) D. Beautiful Array 分类讨论连续递推dp

    题意:给出一个 数列 和一个x 可以对数列一个连续的部分 每个数乘以x  问该序列可以达到的最大连续序列和是多少 思路: 不是所有区间题目都是线段树!!!!!! 这题其实是一个很简单的dp 使用的是分 ...

  8. 【cf789B】Masha and geometric depression(分类讨论/暴力)

    B. Masha and geometric depression 题意 在黑板上写数列,首项是b,公比是q,超过l时就停止不写.给定m个数,遇到后跳过不写.问一共写多少个数,如果无穷个输出inf. ...

  9. P2331 [SCOI2005]最大子矩阵 (动规:分类讨论状态)

    题目链接:传送门 题目: 题目描述 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 输入输出格式 输入格式: 第一行为n,m,k( ...

随机推荐

  1. cnn 卷积神经网络 人脸识别

    卷积网络博大精深,不同的网络模型,跑出来的结果是不一样,在不知道使用什么网络的情况下跑自己的数据集时,我建议最好去参考基于cnn的手写数字识别网络构建,在其基础上进行改进,对于一般测试数据集有很大的帮 ...

  2. bzoj 3524 可持久化线段树

    我们可以先离散化,然后建立权值的可持久化线段树,记录每个数出现的次数,对于区间询问直接判断左右儿子的cnt是不是大于(r-k+1)/2,然后递归到最后一层要是还是大于就有,否则不存在. 反思:挺简单一 ...

  3. ...args剩余参数用法

      剩余参数语法允许我们将一个不定数量的参数表示为一个数组. function sum(...theArgs) { return theArgs.reduce((previous, current) ...

  4. 命令行创建KVM虚拟机

    qemu命令创建虚拟机: qemu-img create -f qcow2 /home/ubuntu.img 20G   qemu-system-x86_64 -m 2048 -enable-kvm ...

  5. 【转】CVE-2010-4258 漏洞分析

    一. 漏洞简介 CVE-2010-4258这个漏洞很有意思,主要思路是如果通过clone函数去创建进程,并且带有CLONE_CHILD_CLEARTID标志,那么进程在退出的时候,可以造成内核任意地址 ...

  6. CiteSeer统计的计算机领域的期刊和会议的影响因子(2005)

    产生自CiterSeer 2005数据库,实际的影响因子可能更高.仅供参考使用.真实的IF还需去官网查看 . OSDI: 3.31 (top 0.08%) . USENIX Symposium on ...

  7. devm_xxx机制

    前言 devm是内核提供的基础机制,用于方便驱动开发者所分配资源的自动回收.参考内核文档devres.txt.总的来说,就是驱动开发者只需要调用这类接口分配期望的资源,不用关心释放问题.这些资源的释放 ...

  8. 使用cURL POST上传文件

    使用cURL POST上传文件 http://blog.csdn.net/v6543210/article/details/20152575

  9. 【LabVIEW技巧】代码块快速放置

    前言 之前的文章中介绍了如何使用QuickDrop来实现快速代码放置,今天我们来详细的聊一下如何进行代码块的快速放置. 正文 LabVIWE程序设计中,我们在架构层级总是进行重复性的编写.举一个例子: ...

  10. FineReport——自定义控件实现填报提交事件和校验

    在报表内部或者在引用报表的HTML页面,定义一个按钮标签,通过FR提供的方法实现提交功能. <button onclick="_g('${sessionID}').writeRepor ...