首页 > 其他分享 >题解|2024暑期牛客多校01

题解|2024暑期牛客多校01

时间:2024-07-16 22:57:46浏览次数:18  
标签:01 const int 题解 ll 多校 cin tms dir

【原文链接】
太菜了就先挂4题,其他的写出来了再回来补QwQ

A.A Bit Common

组合数学

题目大意

题目概括:
给定两个整数 n n n 和 m m m,我们需要计算所有包含 n n n 个非负整数且每个整数都小于 2 m 2^m 2m 的序列中,满足存在一个非空子序列使得这些整数的按位与(bitwise AND)为 1 的序列的数量。

注意:一个序列的非空子序列是指从序列中删除零个或多个元素后得到的子序列,并保持剩余元素的原始顺序。

由于结果可能非常大,请输出结果对一个正整数 q q q 取模后的值。

解题思路

对于序列 A A A 中的每个整数,我们可以将其看作一个 m m m 位的二进制数。

符合条件的序列中,所有末位为 1 的元素的&和一定为 1 ,因为其他元素的末位为 0 ,再做&运算会消灭末位 1 。

枚举序列包含的末位为 1 的元素个数 i i i 。

对于这 i i i 个元素,要满足除末位外每一个二进制位不全为 1 。这 i i i 个数每个二进制位上的方案数为 2 i − 1 2^{i}-1 2i−1 ,除末位外有 m − 1 m-1 m−1 个二进制位,(有序)方案数为 a = ( 2 i − 1 ) m − 1 a=(2^{i}-1)^{m-1} a=(2i−1)m−1 。

对于剩余的 n − i n-i n−i 个元素,只需要保证末位为 0 。可能的数字有 2 m − 1 2^{m-1} 2m−1 种, n − i n-i n−i 个数的(有序)方案数为 b = ( 2 m − 1 ) n − i b=(2^{m-1})^{n-i} b=(2m−1)n−i 。

综合两部分,方案数为 a ∗ b ∗ ( C ( n , i ) ) a*b*(C(n,i)) a∗b∗(C(n,i)) 。 C ( n , i ) C(n,i) C(n,i) 表示序列的 n n n 个位置中选取 i i i 个用于顺序放置末位为 1 的 i i i 个数。

参考程序

#include<bits/stdc++.h>
using namespace std;

typedef long long ll;

const ll N=5005;
ll C[N][N]={{0}};
ll MOD=1e9+7;
ll n,m;

#define Get_Mod(x) (((x)+MOD)%MOD)
#define endl '\n'

void preC(){
    for(ll i=0;i<N;i++){
        C[i][0]=1;
        for(ll j=1;j<=i;j++){
            C[i][j]=(C[i-1][j]+C[i-1][j-1])%MOD;
        }
    }
}

ll qcpow(ll a,ll b,ll p){
    ll ret = 1;
    a%=p;
    while(b){
        if(b&1) ret = ret*a%p;
        a = a*a%p;
        b>>=1;
    }
    return ret;
}

signed main(){
    ios::sync_with_stdio(false);
    cin.tie(nullptr); cout.tie(nullptr);
    cin >> n >> m >> MOD;
    preC();
    ll t,a,b,ans=0;
    for(ll i=0;i<=n;i++){
        t = qcpow(2,i,MOD)-1;
        a = qcpow(t,m-1,MOD);
        t = qcpow(2,m-1,MOD);
        b = qcpow(t,n-i,MOD);
        t = Get_Mod(a*b);
        t = Get_Mod(C[n][i]*t);
        ans = Get_Mod(ans+t);
    }
    if(n==1) cout << Get_Mod(1) << endl;
    else if(m==1) cout << Get_Mod(ans-1) << endl;
    else cout << ans << endl;

    return 0;
}

C.Sum of Suffix Sums

题目大意

给定一个初始为空的数组,你需要进行 q 次操作:

  • 对于每次操作,给定两个非负整数 tv,先从数组末尾取出 t 个元素,然后将 v 添加到数组末尾。保证 t 不会超过操作前数组的长度。

每次操作后,假设当前数组为 a1, a2, ..., an,计算 s1, s2, ..., sn 的总和,其中 si = ai + ai+1 + ... + an 是从位置 i 开始的后缀和。

由于结果可能非常大,输出时需对 1000000007 取模。

解题思路

考虑每个元素的贡献:第 i i i 个元素对答案的贡献为 i ∗ a i i*a_i i∗ai​ 。

在加入元素时,直接将其贡献加入答案。

同时对这个序列维护一个前缀和,以便快速移除元素和减去贡献。

参考程序

const ll N = 200005;
void solve()
{
    ll q,t,rm;cin >> q;
    ll n=1;
    vector<ll> v,s;
    v.emplace_back(0);
    s.emplace_back(0);
    ll tm;
    ll sum=0;
    while(q--){
        cin >> rm >> t;
        sum = Get_Mod(sum-Get_Mod(s[n-1]-s[n-rm-1]));
        s.erase(s.end()-rm,s.end());
        v.erase(v.end()-rm,v.end());
        n-=rm;

        tm = t;
        tm = Get_Mod(tm*n); // 元素贡献
        sum = Get_Mod(sum+tm);
        v.emplace_back(tm);
        s.emplace_back(sum);
        n++;

        cout << sum << endl;
    }
}

H.World Finals

题目大意

两场ICPC比赛,已知(预测的)两场比赛的所有队伍的解题数和罚时。

如果一支队伍同时具有两场比赛的资格,只能参加其中一场。

现在,lzr010506可以决定 同时具有两场比赛的资格 的队伍具体参加哪一场,根据预测数据求lzr010506可以得到的最高名次。

解题思路

开2个map分别记录两场比赛情况。

假设lzr010506参加其中一场,去掉那一场所有的 同时具有两场比赛的资格 的队伍,排序即可得到这一场的最佳名次。

如此求出两场比赛的最佳名次,取最小值即可。

参考程序

const ll N = 200005;
string me="lzr010506";
bool cmp(const pair<string,pll> &a,const pair<string,pll> &b)
{
    pll pa,pb;
    pa=a.second; pb=b.second;
    if(pa.first==pb.first) return pa.second<pb.second;
    return pa.first>pb.first;
}
void solve()
{
    map<string,pll> v1,v2;
    ll n,m;
    cin >> n;
    string ts; ll t1,t2;
    ll ans1,ans2;ans1=ans2=0;
    FORLL(i,1,n)
    {
        cin >> ts >> t1 >> t2;
        v1[ts]=make_pair(t1,t2);
    }
    cin >> m;
    FORLL(i,1,m)
    {
        cin >> ts >> t1 >> t2;
        v2[ts]=make_pair(t1,t2);
    }
    map<string,pll> tmp;
    tmp = v1;
    for(auto &x:v2){
        if(tmp.count(x.first)&&x.first!=me){
            tmp.erase(x.first);
        }
    }
    auto tv = vector<pair<string,pll>>(tmp.begin(),tmp.end());
    sort(ALL(tv),cmp);
    for(auto &x:tv){
        ans1++;
        if(x.first==me) break;
    }
    tmp = v2;
    for(auto &x:v1){
        if(tmp.count(x.first)&&x.first!=me){
            tmp.erase(x.first);
        }
    }
    tv = vector<pair<string,pll>>(tmp.begin(),tmp.end());
    sort(ALL(tv),cmp);
    for(auto &x:tv){
        ans2++;
        if(x.first==me) break;
    }
    cout << min(ans1,ans2) << endl;
}

I.Mirror Maze

DFS

题目大意

在一个 n × m n \times m n×m 的镜子迷宫中,每个格子都有一面镜子。镜子的类型有以下四种:

  1. -:来自上方或下方的光线将被反射回去,来自左方或右方的光线则继续前进而不会被反射。
  2. |:来自左方或右方的光线将被反射回去,来自上方或下方的光线则继续前进而不会被反射。
  3. /:来自左方、右方、上方、下方的光线将分别被反射到上方、下方、左方、右方。
  4. \:来自左方、右方、上方、下方的光线将分别被反射到下方、上方、右方、左方。

现在有 q q q 个光源(给定位置和方向)。光的信徒小G想知道,对于每个光源,在足够长的时间内,发出的光线会被反射经过的不同镜子的数量。

解题思路

对于一个光源,如果它的传播路径没有首尾相连成环,那么这个路径就只可能是一条链。
(证明:光的路径是可逆的,假设一条传播路径在某一点突然形成了环,那回溯时在这个点就可以有两条路径,这显然是不合理的)

对于给定的镜子阵列,我们可以直接处理出每个点向每个方向的答案,对最后的询问打表。

只要是一条链,它的最初起点和最后终点一定在阵列的边缘位置。我们可以从边缘位置开始向内DFS,直到遇到边缘位置,这样就可以得到一条链的路径,路径上的答案都可以处理得到。

没有遍历到的点,说明它们是环的一部分,我们可以通过时间戳来寻找和处理环上的答案。

参考程序

//                D  U  R  L
const int dx[] = {1, -1, 0, 0};
const int dy[] = {0, 0, 1, -1};
const int c0[] = {1, 0, 2, 3}; // -
const int c1[] = {0, 1, 3, 2}; // |
const int c2[] = {3, 2, 1, 0}; // /
const int c3[] = {2, 3, 0, 1}; // \'

int conv(int d, char c){
    switch(c){
        case '-': return c0[d];
        case '|': return c1[d];
        case '/': return c2[d];
        case '\\': return c3[d];
    }
    return -1;
}

int n,m,q,tms,cnt; //tms:timestamp
vector<string> mp;
vector<vector<int>> vis;
vector<vector<array<int,4>>> ans,visd;
vector<tuple<int,int,int>> buf;

void dfs_line(int x,int y,int dir){
    buf.push_back({x,y,dir});
    if(x<0||x>=n||y<0||y>=m) return;
    int nd=conv(dir,mp[x][y]); //new direction
    dfs_line(x+dx[nd],y+dy[nd],nd);
}
void f_line(int sx,int sy,int sd){
    buf.clear();
    dfs_line(sx,sy,sd);
    reverse(buf.begin(),buf.end());
    int res=0; tms++;
    for(int i=0;i<buf.size()-1;i++){ //回溯
        auto [x,y,d]=buf[i];
        if(i){
            int fl=1;
            if(mp[x][y]=='-'&&(d&2)||mp[x][y]=='|'&&!(d&2)) fl=0; // 通过
            if(fl){
                if(vis[x][y]!=tms) res+=fl; // 反射
                vis[x][y]=tms;
            }
        }
        visd[x+dx[d^1]][y+dy[d^1]][d]=tms;
        ans [x+dx[d^1]][y+dy[d^1]][d]=res;
    }
}

void dfs_loop(int x,int y,int dir){
    visd[x][y][dir]=tms;
    x += dx[dir];
    y += dy[dir];
    int nd = conv(dir,mp[x][y]);
    if(nd!=dir){
        if(vis[x][y]!=tms) cnt++;
        vis[x][y]=tms;
    }
    if(visd[x][y][nd]!=tms) dfs_loop(x,y,nd);
}
void mk_loop(int x,int y,int dir){
    ans[x][y][dir] = cnt;
    visd[x][y][dir] = tms;
    x+=dx[dir];
    y+=dy[dir];
    int nd=conv(dir,mp[x][y]);
    if(visd[x][y][nd]!=tms) mk_loop(x,y,nd);
}
void f_loop(int sx,int sy,int sd){
    cnt=0; tms++;
    dfs_loop(sx,sy,sd);
    tms++;
    mk_loop(sx,sy,sd);
}

void solve()
{
    cin >> n >> m;
    mp.clear();
    mp.resize(n);
    for(auto& s:mp) cin >> s;
    ans.resize(n,vector<array<int,4>>(m,{0,0,0,0}));
    visd.resize(n,vector<array<int,4>>(m,{0,0,0,0}));
    vis.resize(n,vector<int>(m,0));
    for(int i=0;i<n;i++){
        f_line(i,0,2);
        f_line(i,m-1,3);
    }
    for(int i=0;i<m;i++){
        f_line(0,i,0);
        f_line(n-1,i,1);
    }
    for(int i=0;i<n;i++){
        for(int j=0;j<m;j++){
            for(int d=0;d<4;d++){
                if(visd[i][j][d]==0)
                    f_loop(i,j,d);
            }
        }
    }
    cin >> q;
    while(q--){
        int x,y,d;
        string s;
        cin >> x >> y >> s;
        x--,y--;
        if(s[0]=='b') d=0;
        if(s[0]=='a') d=1;
        if(s[0]=='r') d=2;
        if(s[0]=='l') d=3;
        cout << ans[x][y][d] << endl;
    }
}

标签:01,const,int,题解,ll,多校,cin,tms,dir
From: https://blog.csdn.net/m0_73786319/article/details/140479093

相关文章

  • [题解]POJ2074 Line of Sight
    POJ2074LineofSight题意简述多测。给定若干条线段,全部与\(x\)轴平行。其中有\(2\)条线段表示房子和人行道(虽然翻译不是人行道就是了),保证房子在人行道上面。其他线段表示障碍物(不保证在房子和人行道之间)。请找出人行道上最长的连续部分,使得在这中间可以完整地看到房子的全......
  • [[BJWC2012] 冻结]
    [BJWC2012]冻结题目大意在能有\(k\)次机会使得某些道路变为其\(\frac{1}{2}\)长度的情况下,求\(1\)到\(n\)的最短路做法其实就是分层最短路的模板题,有\(k\)次机会减少,那么对于一个点来说就有可能是在这前使用了\(0\)~\(k\)次减少的机会到达的,每个点都有\(k\)个分身,故要建\(k+1......
  • CF825F String Compression题解
    思路容易想到是个动态规划。首先设\(f_i\)表示字符串前\(i\)个字符所组成的字符串的答案。状态定义好了,接下来就是考虑如何转移了。因为由\(f_i\)可以得到所有\(f_j\),其中\(i+j\lelen\),转移方程为\(f_i=f_j+x\),其中\(x\)为字符串\(i+1\)至\(j\)的最优压缩。接......
  • [ABC347E] Set Add Query题解
    思路通过读题发现,每个数变化当且仅当这个数在集合内。所以不妨设它被添加进来的时间点为\(L_i\),它被删除的时间点为\(R_i\),所以它被增加的数量就是这段时间内集合数量之和。所以用一个变量\(cnt\)模拟当前集合内有多少个数,前缀和维护即可。具体实现参见代码。代码#include<......
  • CF1898D Absolute Beauty 题解
    思路容易发现,如果\(a_i>b_i\)则将\(a_i\)和\(b_i\)交换。在数轴上标出要交换的四个数的位置若线段\(a_ib_i\)和线段\(a_jb_j\)互不相交,此时交换比两条线段处于其他位置时更优。具体证明这里就不再赘述,其他题解讲的已经很清楚了。所以只需交换最大的\(a_i\)和最小......
  • [ABC338E] Chords 题解
    思路思路还是很显然的,简单总结一下思路:首先,将圆环从点\(1\)到\(2N\)切开,并将其拉直成一条直线。在切开状态下,原来的弦变成了直线上的曲线。我们需要判断这些曲线之间是否存在交点。在切开状态下,曲线之间的交点等价于满足\(A_i<A_j<B_i<B_j\)的不同曲线\(i\)和......
  • [ABC258Ex] Odd Steps 题解
    思路拿到这道题,第一时间肯定想到是\(dp\)题目。朴素DP用\(dp_i\)表示序列和为\(i\)的序列个数。因为原数组由奇数组成,所以\(dp\)只可能由\(dp_{i-1}\),\(dp_{i-3}\)等等转移过来,若\(i\inA\),\(dp_i=0\)。即:\[dp_i=\begin{cases}0&i\inA\\dp_{i-1}+dp_{i-3}+\c......
  • 2024牛客暑期多校训练营1
    Preface第一场牛客多校,直接被创飞了开局本来舒畅无比,签了A,C,H后马上上机RushI,然后写了一坨东西后过了编译就交然后就过了此时祁神给出了J题的做法,遂让机然后看榜上过的比较多的B,D两题,徐神开出了字符串G但感觉十分难写然后后面我和徐神花样对着B,D两个题想办法却......
  • CF1859A题解
    CF1859A题解思路考虑一种极端情况,\(b\)数组内的数全部比\(a\)大,这样也无法整除,所以这就是这道题的突破口。我们让\(b\)数组内的数全部比\(a\)里的大,最方便的实现方法就是把原数组内的最大的数放进\(b\)数组,剩下的放进\(a\)数组。注意特判无解情况:原数组内的数一样,这......
  • [ABC352D]题解
    题意在长为\(n\)的序列\(a\)中找出\(k\)个数,设它们的下表为$p_1\(,\)p_2$到\(p_k\),满足这\(k\)个数从小到大排列过后是一个公差为\(1\)的等差数列。求满足条件的\(k\)个数的最大的\(p\)减去最小的\(p\)最小。输出这个值。思路把数组\(a\)排一遍序,滑动窗......