链接
题意:
给出你一个长度为 n n n的序列,然后让你从中找出子串要求第一个数比最后一个小,要求找出的子串求方案数?
分析:
首先我们肯定是要找两个数符合条件的两个数。确定这两个数之后,其中的方案数就是(
2
j
−
i
2^{j-i}
2j−i)方案, 就是从中间选0,1,2,3,…都选,一共这么多方案数。当我们枚举一个数,然后比他小的数有多少个时,我们肯定想到用树状数组来维护。然后剩下的问题,就是如何维护这个方案数
2
j
−
i
2^{j-i}
2j−i
我们看 答案应该是
∑
2
j
−
i
=
∑
2
j
2
i
)
\sum2^{j-i}=\sum\frac{2^j}{2^i})
∑2j−i=∑2i2j)提取
2
j
2^j
2j得
2
j
∑
1
2
i
2^j\sum\frac{1}{2^i}
2j∑2i1我们将
1
2
\frac{1}{2}
21转化成逆元将其每次相加。就完成了者一系列的操作。
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
typedef unsigned long long ull;
#define x first
#define y second
#define sf scanf
#define pf printf
#define PI acos(-1)
#define inf 0x3f3f3f3f
#define lowbit(x) ((-x)&x)
#define mem(a,x) memset(a,x,sizeof(a))
#define rep(i,n) for(int i=0;i<(n);++i)
#define repi(i,a,b) for(int i=int(a);i<=(b);++i)
#define repr(i,b,a) for(int i=int(b);i>=(a);--i)
#define debug(x) cout << #x << ": " << x << endl;
const int MOD = 998244353;
const int mod = 1e9 + 7;
const int maxn = 3e5 + 10;
const int dx[] = {0, 1, -1, 0, 0};
const int dy[] = {0, 0, 0, 1, -1};
const int dz[] = {1, -1, 0, 0, 0, 0 };
int day[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
void init()
{
}
ll n, k,cnt;
ll a[maxn], b[maxn];
map<ll, ll > mp;
struct BIT {///树状数组
ll c[maxn];
void add(int x,ll d){
while(x<=n+1){
c[x]+=d;
c[x]%=MOD;
x+=lowbit(x);
}
}
ll query(int x){
ll ans=0;
while(x){
ans+=c[x];
ans%=MOD;
x-=lowbit(x);
}
return ans;
}
}bit;
ll qpow(ll a, ll b,ll p)
{
ll ans = 1;
while(b)
{
if(b & 1) ans = ans * a % p;
a = a * a % p;
b >>= 1;
}
return ans;
}
void solve()
{
cin>>n;
for(int i=1;i<=n;i++){
scanf("%lld",&a[i]);
if(mp[a[i]]==0){
mp[a[i]]=1;
b[++cnt]=a[i];
}
}
sort(b+1,b+1+cnt);
for(int i=1;i<=cnt;i++) mp[b[i]]= i;
for(int i=1;i<=n;i++){
a[i]=mp[a[i]];
}
ll ans=0;
ll divv= qpow(2,MOD-2,MOD);
for(int i=1;i<=n;i++){
ans=(ans+qpow(2,i-1,MOD)*bit.query(a[i])%MOD)%MOD;
bit.add(a[i],qpow(divv,i,MOD));
}
cout<<ans<<endl;
}
int main()
{
init();
ll t = 1;
//scanf("%lld",&t);
while(t--)
{
solve();
}
return 0;
}