分数规划学习笔记

1. 用途

分数规划常用于求一个分式的极值,就是给出两个序列\(a_i\)\(b_i\),使得\(\dfrac{\sum a_i\times w_i}{\sum b_i\times w_i}\)的值最大或最小,其中\(w \in \{0,1\}\)

通常,题目中还会有类似于\(\sum b_i>w\)的限制

2. 解法

通常使用二分,记当前二分的值为mid

\[\begin{aligned}&\because \dfrac{\sum a_i\times w_i}{\sum b_i\times w_i}\ge mid\\&\therefore \sum a_i\times w_i\ge mid\times \sum b_i\times w_i\\&\therefore\sum a_i\times w_i-mid\times \sum b_i\times w_i\ge 0\\&\therefore \sum w_i \times (a_i-mid\times b_i)\end{aligned} \]

所以,在每一次check的时候,求出不等式右边的最大值判断即可,而这一部分显然可以用0/1背包完成

3. 例题

P4377 [USACO18OPEN] Talent Show G

按照上述方法二分即可,对于重量的限制可以以重量为下标,求0/1背包判断即可

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
int n,w,a[255],b[255];
ll f[10005];
bool check(int x)
{
	memset(f,128,sizeof(f));
	f[0]=0;
	ll tmp=f[w];
	for(int i=1;i<=n;i++)
	{
		for(int j=w;j>=0;j--)
		{
			if(tmp==f[j]) continue;
			int k=j+a[i];
			k=min(k,w);
			f[k]=max(f[k],f[j]+b[i]-1ll*a[i]*x);
		}
	}
	if(f[w]>=0) return 1;
	return 0;
}
int main()
{
	scanf("%d%d",&n,&w);
	for(int i=1;i<=n;i++)
	{
		scanf("%d%d",&a[i],&b[i]);
		b[i]*=1000;
	}
	int l=0,r=100000;
	while(l<r)
	{
	//	printf("%d %d\n",l,r);
		int mid=(l+r+1)>>1;
		if(check(mid))
		{
			l=mid;
		}
		else r=mid-1;
	}
	printf("%d",l);
	return 0;
}
posted @ 2024-06-07 18:07  wangsiqi2010916  阅读(12)  评论(0编辑  收藏  举报