樹狀陣列 離散化 簡單計算器

2021-10-09 05:58:50 字數 1938 閱讀 4462

sample output

3
離散化+樹狀陣列

這裡引用至一篇別人的部落格

當資料的範圍較小時,比如maxn=100000,那麼我們可以開乙個陣列c[maxn],來記錄前面資料的出現情況,初始化為0;當資料a出現時,就令c[a]=1。這樣的話,欲求某個數a的逆序數,只需要算出在當前狀態下c[a+1,maxn]中有多少個1,因為這些位置的數在a之前出現且比a大。但是若每新增乙個資料a時,就得從a+1到 maxn搜一遍,複雜度太高了。樹狀陣列卻能很好的解決這個問題,可以把數乙個個插入到樹狀陣列中, 每插入乙個數, 統計比他小的數的個數,對應的逆序為 i - getsum( c[i] ),其中 i 為當前已經插入的數的個數, getsum( c[i] )為比 c[i] 小的數的個數,i- getsum( c[i] ) 即比c[i] 大的個數, 即逆序的個數。最後需要把所有逆序數求和,就是在插入的過程中邊插入邊求和.

這題先計算a和b,c和d的對數,相乘,再減去a=c,a=d,b=c,b=d的情況。

#include

#include

#include

#include

#include

#define ll long long

using

namespace std;

struct asdf aa[

100001];

ll n,t;

ll c[

100001

],a[

100001

],ls[

100001

],lb[

100001

],rs[

100001

],rb[

100001

],suml,sumr,ans;

bool

cmp(asdf aaa, asdf bbb)

ll lowbit

(ll z)

void

build

(ll z)

ll sum

(ll z)

intmain()

sort

(aa +

1, aa +

1+ n, cmp)

;//離散化

aa[0]

.zz =-1

;for

(ll i =

1; i <= n;

++i)

if(aa[i]

.zz!=aa[i -1]

.zz) a[aa[i]

.bh]

=++t;

else a[aa[i]

.bh]

= t;

for(ll i =

1; i <= n;

++i)

memset

(c,0

,sizeof

(c))

;for

(ll i = n; i;

--i)

ans = suml * sumr;

for(ll i =

1; i <= n;

++i)

ans -

= ls[i]

*rs[i]

+ lb[i]

*rb[i]

+ rb[i]

*rs[i]

+ lb[i]

*ls[i]

;//減去abcd重複的情況

printf

("%lld"

,ans)

;}

簡單計算題 離散化 樹狀陣列

3 解題思路 首先我們要知道乙個樹狀陣列與逆序對的東西 所以為了防爆,得用離散化把s序列離散一下在加進樹狀陣列 然後我們把符合 a,b 的方案數p,符合 c,d 的方案數q求出來 樹狀陣列求逆序對,具體得理解樹狀陣列太多了不想打了 ans pq,接著處理a b c d的情況,因為a已經小於b,c已經...

簡單計算器

unit unit1 inte ce uses windows,messages,sysutils,variants,classes,graphics,controls,forms,dialogs,stdctrls,buttons,math math是數 算單元 type tform1 class ...

簡單計算器

a 簡單計算器 crawling in process.crawling failed time limit 1000msmemory limit 32768kb64bit io format i64d i64u submit status description 讀入乙個只包含 的非負整數計算表示...