二分法作为分治中最常见的方法,适用于单调函数,逼近求解某点的值。但是当函数是凸性函数或者凹性函数时,二分法就不再适用,这时我们便用到的三分法。
如上图所示,类似二分的定义,mid = (left + right)/ 2,midmid = (mid + right)/ 2,如果mid靠近极值点,则 right = midmid,否则 left = mid。
三分搜索法的模板如下:
double Calc(Type a)
{
/* 根据题目的意思计算 */
}
void Solve()
{
double left, right;
double mid, midmid;
double mid_value, midmid_value;
left = MIN; right = MAX;
while (left + EPS < right)
{
mid = (left + right) / 2;
midmid = (mid + right) / 2;
mid_value = Calc(mid);
midmid_value = Calc(midmid);
// 假设求解最大极值.
if (mid_value >= midmid_value) right = midmid;
else left = mid;
}
}
2008 Asia Harbin Regional Contest Online
http://acm.hrbeu.edu.cn/index.php?act=problem&id=1280
汽车拐弯问题,给定X, Y, l, w判断是否能够拐弯。首先当X或者Y小于w,那么一定不能。
其次我们发现随着角度θ的增大,最大高度h先增长后减小,即为凸性函数,可以用三分法来求解。
这里的Calc函数需要比较繁琐的推倒公式:
s = l * cos(θ) + w * sin(θ) - x;
h = s * tan(θ) + w * cos(θ);
其中s为汽车最右边的点离拐角的水平距离, h为里拐点