日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

hdu 4923 Room and Moor (单调栈+思维)

發布時間:2025/4/5 编程问答 21 豆豆
生活随笔 收集整理的這篇文章主要介紹了 hdu 4923 Room and Moor (单调栈+思维) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

題意:

給一個0和1組成的序列a,要構造一個相同長度的序列b。b要滿足非嚴格單調,且

值為0到1的實數。最后使得 ?sum((ai-bi)^2)最小。


算法:

首先a序列開始的連續0和末尾的連續1是能夠不考慮的。

由于僅僅要b序列相應開頭為0、

末尾為1,既不影響單調性又能使相應的(ai-bi)^2=0。

然后,

先找111100、11100、10這樣以1開始以0結束的序列塊。每一塊相應的b值相等且均為

這一塊的平均值,即1的個數/0和1的總個數。

可是要滿足b的單調性,則我們用棧來維護,假設后面一塊的均值<前面一塊的均值,則

合并這兩塊。也就是僅僅要棧頂的塊的均值小于要壓入棧的塊的均值,就一直合并,直到

滿足單調性。再把當前的值壓入棧。

最后僅僅要一塊塊統計相應的sum((ai-bi)^2)就是答案了。


逗逼記憶---->比賽的時候我沒有想到塊,僅僅想到除去前面的0和后面的1。中間的值都是一樣。

用二分或者三分解決,僅僅要控制精度=。

=


P.S: ? 這題必須注意細節,否則極易丟失精度。主要是我代碼凝視的兩個我開始沒有控制好的地方。

o(╯□╰)o


#include<cstdio> #include<iostream> #include<cstring> #define maxn 100010 using namespace std;struct node {double num,v; //v表示1的個數,num表示0和1的總個數 }; node stk[maxn]; double sum[maxn],a[maxn];int main() {int T,n;scanf("%d",&T);while(T--){scanf("%d",&n);memset(sum,0,sizeof(sum));for(int i=1;i<=n;i++){scanf("%lf",&a[i]);sum[i] = sum[i-1]+a[i];}int i = 1,k = n;while(a[i]==0) i++;while(a[k]==1) k--;int top = 0,le = i;node x,y;for(;i<=k;i++){if(a[i]==0){while(a[i]==0 && i<=k) //這里假設不加控制i<=k,i可能超出ki++;double a = sum[i-1]-sum[le-1];double b = (double)i-le;while(top>0 && a/b<stk[top].v/stk[top].num) //這里也不能忘了top>0{y = stk[top--];a += y.v;b += y.num;}x.v = a;x.num = b;stk[++top] = x;le = i;}}double ans = 0;while(top){y = stk[top--];double val = y.v/y.num;ans += (1-val)*(1-val)*y.v + val*val*(y.num-y.v);}printf("%.6lf\n",ans);}return 0; }





總結

以上是生活随笔為你收集整理的hdu 4923 Room and Moor (单调栈+思维)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。