暂且把这个问题叫做带权最长上升子序列。
显然,类似于求 L I S LIS LIS,如果我们在 a a a 序列的前 i i i 个数中已经选了一个好的序列 c c c,那么 c c c 的最后一个一定是最小的(因为后面更容易满足条件增加长度)。
于是用二分, l o w i low_i lowi 表示长度为 i i i 的带权最长上升子序列的 a i ⋅ b i a_i\cdot b_i ai⋅bi 的最小值。
每次用 lower_bound \texttt{lower\_bound} lower_bound 在 l o w low low 中查找大于等于 a i a_i ai 的第一个位置,用 a i ⋅ b i a_i\cdot b_i ai⋅bi 更新该位置,同时记录答案。
这样就做完了。
细节详见代码。
#include
using namespace std;
typedef long long ll;
int n,ans=1;
ll a[1000001],b[1000001],low[1000001];
ll read()
{
ll sum=0;
int c=getchar();
while(c<48||c>57) c=getchar();
while(c>=48&&c<=57) sum=sum*10+c-48,c=getchar();
return sum;
}
int main()
{
freopen("C.in","r",stdin);
freopen("C.out","w",stdout);
n=read();
for(int i=1;i<=n;i++) a[i]=read();
for(int i=1;i<=n;i++) b[i]=read();
memset(low,0x3f,sizeof(low));
for(int i=1;i<=n;i++){
int czn=lower_bound(low+1,low+1+ans,a[i])-low;
ans=max(ans,czn);
low[czn]=min(a[i]*b[czn],low[czn]);
}
cout<<ans;
}