noi2019模拟测试赛(四十七)

T1与运算(and)

题意:

​ 给你一个序列\(a_i\),定义\(f_i=a_1\&a_2\&\cdots\&a_i\),求这个序列的所有排列的\(\Sigma_i f_i\)的最大值。

题解:

​ dp,记\(dp_i\)表示前面的数与和为\(i\)的最大值,转移要一个超集的东西,fwt搞一搞就行了。

#include<bits/stdc++.h>
#define fo(i,l,r) for(int i=l;i<=r;i++)
#define of(i,l,r) for(int i=l;i>=r;i--)
#define fe(i,u) for(int i=head[u];i;i=e[i].next)
#define el putchar('\n')
#define ta putchar('    ')
using namespace std;
typedef long long ll;
inline void open(const char *s)
{
    #ifndef ONLINE_JUDGE
    char str[20];
    sprintf(str,"%s.in",s);
    freopen(str,"r",stdin);
//  sprintf(str,"%s.out",s);
//  freopen(str,"w",stdout);
    #endif
}
inline int rd()
{
    static int x,f;
    x=0;f=1;
    char ch=getchar();
    for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
    for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
    return f>0?x:-x;
}
const int N=1<<20;
ll n,cnt[N],f[N];

inline void fwt(ll *a,int len)
{
    for(int i=2;i<=len;i<<=1)
        for(int j=0;j<len;j+=i)
            for(int k=0;k<(i>>1);++k)a[j+k]+=a[j+k+(i>>1)];
}

int main()
{
    n=rd();fo(i,1,n)++cnt[rd()];
    fwt(cnt,N);
    fo(i,1,N-1)for(int j=1;j<N;j<<=1)if(i&j)f[i]=max(f[i],f[i^j]+cnt[i]*j);
    cout<<f[N-1]<<endl;
    return 0;
}

T2 序列(sequence)

题意:

​ 随机生成一个数列:

  • \(a_0=0\)

  • \(\forall i>0,a_i\)有\(p_i\)的概率等于\(a_{i-1}+1\),有\(1-p_i\)的概率等于\(0\)。

    求\(\Sigma_{i=1}^na^2\)的期望。

题解:

​ 签到推式子题。dp转移就是枚举最后一段连续+1的长度。式子大力差分后可以线性做。代码就不放了。

T3 平均值(average)

题意:

​ 给你一个序列\(a_i\),求:
\[
\Sigma_{l=1}^n\Sigma_{r=l}^n\frac{ mex(a_l,a_l+1,\cdots,a_r)}{r-l+1}
\]

题解:

​ 式子转成:
\[
\Sigma_{i=1}^{max}\Sigma_{l=1}^n\Sigma_{r=l}^n\frac{ [mex(a_l,a_l+1,\cdots,a_r)\ge i]}{r-l+1}
\]
​ 然后就是对于每一个\(l\),求\(r\),贡献是倒数的前缀和。线段树2分+区间覆盖。然后这个区间覆盖要加一个函数,用技巧可以实现。

#include<bits/stdc++.h>
#define fo(i,l,r) for(int i=l;i<=r;i++)
#define of(i,l,r) for(int i=l;i>=r;i--)
#define fe(i,u) for(int i=head[u];i;i=e[i].next)
#define el putchar('\n')
#define ta putchar('    ')
using namespace std;
typedef long long ll;
typedef vector<int> vi;
inline void open(const char *s)
{
    #ifndef ONLINE_JUDGE
    char str[20];
    sprintf(str,"%s.in",s);
    freopen(str,"r",stdin);
//  sprintf(str,"%s.out",s);
//  freopen(str,"w",stdout);
    #endif
}
namespace IO{
const int N=(1<<26)+10;
char buf[N],*H,*T;
inline char Get()
{
    if(H==T)T=(H=buf)+fread(buf,1,N,stdin);
    return H==T?-1:*H++;
}
inline int rd()
{
    static int x,f;
    x=0;f=1;
    char ch=Get();
    for(;ch<'0'||ch>'9';ch=Get())if(ch=='-')f=-1;
    for(;ch>='0'&&ch<='9';ch=Get())x=x*10+ch-'0';
    return f>0?x:-x;
}
}using IO::rd;
const int N=500010,Inf=1000000000,mod=998244353;
int n,m,inv[N],s[N],ss[N],rt;
vi vec[N];

inline int pls(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int mns(int a,int b){return a<b?a-b+mod:a-b;}
inline int mul(int a,int b){return (ll)a*b%mod;}

namespace Seg{
#define lson tr[o].ls,l,mid
#define rson tr[o].rs,mid+1,r
#define qlson lson,L,min(mid,R)
#define qrson rson,max(mid+1,L),R
struct tree{
    int ls,rs,mn,sum,ps,tag;
    tree(){ls=rs=mn=sum=tag=0;}
}tr[N<<1];int cnt=0;

inline void calc(int o,int d,int l,int r){tr[o].sum=mns(ss[d-l],ss[d-r-1]);tr[o].mn=tr[o].tag=d;}
inline void pushup(int o)
{
    tr[o].sum=pls(tr[tr[o].ls].sum,tr[tr[o].rs].sum);
    tr[o].mn=min(tr[tr[o].ls].mn,tr[tr[o].rs].mn);
}
inline void pushdown(int o,int l,int mid,int r)
{static int ls,rs;
    if(!tr[o].tag)return;
    ls=tr[o].ls,rs=tr[o].rs;
    calc(ls,tr[o].tag,l,mid);
    calc(rs,tr[o].tag,mid+1,r);
    tr[o].tag=0;
}

void build(int &o,int l,int r)
{
    o=++cnt;if(l==r)return tr[o].ps=s[n-l+1],void();
    int mid=(l+r)>>1;build(lson);build(rson);
    tr[o].ps=pls(tr[tr[o].ls].ps,tr[tr[o].rs].ps);
}
void modify(int o,int l,int r,int L,int R,int d)
{if(L>R)return;
    if(l==L&&r==R)return calc(o,d,l,r);
    int mid=(l+r)>>1;pushdown(o,l,mid,r);
    if(L<=mid)modify(qlson,d);
    if(R>mid)modify(qrson,d);
    pushup(o);
}
int qsum(int o,int l,int r,int L,int R)
{if(L>R)return 0;
    if(l==L&&r==R)return mns(tr[o].ps,tr[o].sum);
    int mid=(l+r)>>1,res=0;pushdown(o,l,mid,r);
    if(L<=mid)res=pls(res,qsum(qlson));
    if(R>mid)res=pls(res,qsum(qrson));
    return res;
}
int query(int o,int l,int r,int L,int R,int d)
{
    if(l==r)return tr[o].mn<d?l:l-1;
    int mid=(l+r)>>1;pushdown(o,l,mid,r);
    if(l==L&&r==R){
        if(tr[tr[o].rs].mn<d)return query(qrson,d);
        return query(qlson,d);
    }
    int res=mid;
    if(R>mid)res=query(qrson,d);
    if(res>mid)return res;
    if(L<=mid)res=query(qlson,d);
    return res;
}

}

int main()
{open("c");
    n=rd();inv[0]=inv[1]=1;
    fo(i,1,n){
        vec[rd()].push_back(i);
        if(i-1)inv[i]=mns(0,mul(mod/i,inv[mod%i]));
    }
    int ans=0;m=n;
    fo(i,1,n)s[i]=pls(s[i-1],inv[i]);
    fo(i,1,n)ss[i]=pls(ss[i-1],s[i]);
    Seg::build(rt,1,n);
    fo(t,0,500000){
        if(!vec[t].size())break;
        int hh=0;
        of(i,vec[t].size()-1,0){
            if(++hh==1)m=min(m,vec[t][i]);
            int x=Seg::query(rt,1,n,1,m,vec[t][i]);
            int ql=i?vec[t][i-1]+1:1;
            Seg::modify(rt,1,n,ql,min(m,x),vec[t][i]);
        }
        if(!hh)break;
        ans=pls(ans,Seg::qsum(rt,1,n,1,m));
    }
    printf("%d\n",ans);
    return 0;
}

noi2019模拟测试赛(四十七)的更多相关文章

  1. 2017.8.1 Noip2018模拟测试赛(十七)

    日期: 八月第一天  总分: 300分  难度: 提高 ~ 省选    得分: 100分(不应该啊!) 题目目录: T1:战争调度 T2:选数 T3:由乃的OJ 赛后心得: MMP,首先第一题花了大概 ...

  2. [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania

    [2018冬令营模拟测试赛(二十一)]Problem A: Decalcomania 试题描述 输入 见"试题描述" 输出 见"试题描述" 输入示例 见&quo ...

  3. 2016北京集训测试赛(十七)Problem C: 数组

    Solution 线段树好题. 我们考虑用last[i]表示\(i\)这个位置的颜色的上一个出现位置. 考虑以一个位置\(R\)为右端点的区间最远能向左延伸到什么位置: \(L = \max_{i \ ...

  4. 2016北京集训测试赛(十七)Problem B: 银河战舰

    Solution 好题, 又是长链剖分2333 考虑怎么统计答案, 我场上的思路是统计以一个点作为结尾的最长上升链, 但这显然是很难处理的. 正解的方法是统计以每个点作为折弯点的最长上升链. 具体的内 ...

  5. 2016北京集训测试赛(十七)Problem A: crash的游戏

    Solution 相当于要你计算这样一个式子: \[ \sum_{x = 0}^m \left( \begin{array}{} m \\ x \end{array} \right) \left( \ ...

  6. 2018.8.8 Noip2018模拟测试赛(二十一)

    日期: 八月七号  总分: 300分  难度: 提高 ~ 省选    得分: 112分(OvO) 题目目录: T1:幸福的道路 T2:Solitaire T3:Flags 赛后心得: 第一题裸树d啊! ...

  7. 2018.8.7 Noip2018模拟测试赛(二十)

    日期: 八月七号  总分: 300分  难度: 提高 ~ 省选    得分: 100分(呵呵一笑) 题目列表: T1:SS T2:Tree Game T3:二元运算 赛后反思: Emmmmmm…… 开 ...

  8. 2018.8.6 Noip2018模拟测试赛(十九)

    日期: 八月六号  总分: 300分  难度: 提高 ~ 省选    得分: 10分(MMP) 题目目录: T1:Tree T2:异或运算 T3:Tree Restoring 赛后反思: Emmmmm ...

  9. 2017.8.2 Noip2018模拟测试赛(十八)

     日期: 八月二日  总分: 300分  难度: 提高 ~ 省选  得分: 40分(又炸蛋了!!) 题目列表: T1:分手是祝愿 T2:残缺的字符串 T3:树点涂色 赛后心得: 哎,T1求期望,放弃. ...

随机推荐

  1. 【模板】后缀排序(SA数组)

    [模板]后缀排序 题目背景 这是一道模板题. 题目描述 读入一个长度为 \(n\) 的由大小写英文字母或数字组成的字符串,请把这个字符串的所有非空后缀按字典序从小到大排序,然后按顺序输出后缀的第一个字 ...

  2. 洛谷 P3199 [HNOI2009]最小圈

    P3199 [HNOI2009]最小圈 题目背景 如果你能提供题面或者题意简述,请直接在讨论区发帖,感谢你的贡献. 题目描述 对于一张有向图,要你求图中最小圈的平均值最小是多少,即若一个圈经过k个节点 ...

  3. poj 2533 Longest Ordered Subsequence 最长递增子序列(LIS)

    两种算法 1.  O(n^2) #include<iostream> #include<cstdio> #include<cstring> using namesp ...

  4. NYOJ_77 开灯问题

    题目地址 分析: 用一个数组来保存每盏灯的操作的次数.推断奇偶就可以推断灯的状态. 最后的输出格式须要注意一下空格的位置,思路就是现输出一个.剩下来的输出在前面加一个空格. 空格用_表示: 1_3_5 ...

  5. vue29-vue2.0组件通信_recv

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  6. Exception: Operation xx of contract xx specifies multiple request body parameters to be serialized without any wrapper elements.

    Operation 'CreateProductCodeStock' of contract 'IChileService' specifies multiple request body param ...

  7. google浏览器修改网页字符编码

    google浏览器修改网页字符编码 直接在google浏览器的应用拓展程序里面搜 Charset,第一个就是 于是就有了

  8. Centos7 网络出错(failed to start LSB: Bring up/down networking )

    这是我更换了VM虚拟机内存,重启后无法连接网络. 然后这是因为NetworkManager.service这个程序造成 解决方法: systemctl disable NetworkManager.s ...

  9. Redis封装之List

    /// <summary> /// Redis list的实现为一个双向链表,即可以支持反向查找和遍历,更方便操作,不过带来了部分额外的内存开销, /// Redis内部的很多实现,包括发 ...

  10. GOLANG 加密,解密,GUID 小方法

    golang的 MD5加密.BASE64解密  guid 的代码: /** * 用于加密,解密,(包含MD5加密和base64加密/解密)以及GUID的生成 * 时间: * zhifieya */ p ...