首页 > 其他分享 >THUSC 2022

THUSC 2022

时间:2023-04-25 22:25:53浏览次数:50  
标签:__ int mid pos 2022 front now THUSC

简单写了 T1, T2 的代码, T3, T4 因为没有评测数据就先咕了。

T1 归程 (return)

签到题做了一个多小时,我真废物。

考虑进行 dp ,题目描述中说明了:每个决策根据当前时间,当前雨是否变大和小 S 所处位置决定。因此这三个参数就是我们 dp 的状态。具体的,设 \(f_{i,j,0/1}\) 表示当前位于点 \(i\) ,当前时刻为 \(j\) ,雨是否变大时到达 \(y\) 的最小的期望淋雨量。转移时考虑经过当前边时雨是否变大,设 \(P_i\) 表示前 \(i-1\) 个时刻雨没有变大时,第 \(i\) 个时刻雨变大的概率,显然我们可以得到转移:

\[\begin{aligned} f_{i,j,0}&=\min_{v}(\prod_{t=j+1}^{j+w}(1-P_t)f_{v,j+w,0}+\sum_{k=1}^{w}\prod_{t=j+1}^{j+k-1}(1-P_t)P_k(kA+(w-k)B+f_{v,j+w,1}))\\ f_{i,j,1}&=\min_{v}(wB+f_{v,j+w,1}) \end{aligned} \]

直接 dp 即可。

code

#include <cstdio>
#include <algorithm>
#include <vector>
using namespace std;
const int max1 = 1000, max2 = 2e4, front = 25;
const double inf = 0x3f3f3f3f3f3f3f3f;
int n, m, k, x, y, P[max2 + front + 5], sum[max2 + front + 5];
struct Edge
{
    int v, w, A, B;
    Edge () {}
    Edge ( int __v, int __w, int __A, int __B )
        { v = __v, w = __w, A = __A, B = __B; }
};
vector <Edge> edge[max1 + 5];
double f[max1 + 5][front + 5][2], val[front + 5], g[front + 5];
pair <double, double> h[front + 5];
int main ()
{
    scanf("%d%d%d%d%d", &n, &m, &k, &x, &y);
    for ( int i = 1, u, v, w, A, B; i <= m; i ++ )
    {
        scanf("%d%d%d%d%d", &u, &v, &w, &A, &B);
        edge[u].push_back(Edge(v, w, A, B));
        edge[v].push_back(Edge(u, w, A, B));
    }
    for ( int i = 1, T, w; i <= k; i ++ )
        { scanf("%d%d", &T, &w); P[T] = sum[T] = w; }
    for ( int i = max2 + front; i >= 0; i -- )
        sum[i] += sum[i + 1];
    int now = 0;
    for ( int i = 1; i <= n; i ++ )
        for ( int T = 0; T < front; T ++ )
            f[i][T][0] = f[i][T][1] = inf;
    for ( int T = 0; T < front; T ++ )
        f[y][T][0] = f[y][T][1] = 0;
    for ( int T = max2; T >= 0; T -- )
    {
        now = ( now + front - 1 ) % front;
        for ( int i = 1; i <= n; i ++ )
            f[i][now][0] = f[i][now][1] = inf;
        f[y][now][0] = f[y][now][1] = 0;
        for ( int i = 1; i <= front; i ++ )
        {
            val[i] = 1;
            g[i] = 0;
            h[i] = make_pair(0, 0);
            for ( int j = 1; j <= i; j ++ )
            {
                if ( P[T + j] )
                {
                    g[i] += val[i] * P[T + j] / sum[T + j];
                    h[i].first += val[i] * P[T + j] / sum[T + j] * j;
                    h[i].second += val[i] * P[T + j] / sum[T + j] * ( i - j );
                    val[i] = val[i] * ( 1.0 - 1.0 * P[T + j] / sum[T + j] );
                }
            }
        }
        for ( int i = 1; i <= n; i ++ )
        {
            if ( i == y )
                continue;
            for ( auto Tmp : edge[i] )
            {
                int v = Tmp.v, w = Tmp.w, A = Tmp.A, B = Tmp.B;
                f[i][now][0] = min(f[i][now][0], val[w] * ( f[v][( now + w ) % front][0] + w * A ) + g[w] * f[v][( now + w ) % front][1] + h[w].first * A + h[w].second * B);
                f[i][now][1] = min(f[i][now][1], w * B + f[v][( now + w ) % front][1]);
            }
        }
    }
    printf("%.10lf\n", f[x][now][0]);
    return 0;
}

T2 最大连续和 (sequence)

发现题意很容易转化为取出 \(a\) 的一段连续区间,取出 \(b\) 序列中一些数来替换区间中的一些数,求解所有操作中区间和的最大值。

首先有一个简单的结论:

设 \(f(l,r)\) 为 \(a\) 中取出的区间为 \([l,r]\) 时,所有替换操作中区间和的最大值,考虑对于每个 \(l\) 找到取值最大的 \(r\) 记作 \(pos_l\) ,那么 \(pos_i\) 随 \(i\) 单调不降。

很容易发现当区间扩张时,一个元素只能从被替换变为不被替换。

于是简单进行证明,假设当前考虑的位置为 \(i\) , 考虑左端点移动到 \(i+1\) 的过程,根据定义我们有 \(f(i,pos_i)\ge f_{i,j},j\in[i,pos_i]\) ,显然存在 \(f_{i,pos_i}-a_i\ge f_{i,j}-a_i,j\in[i,pos_i]\) ,也就是对于所有没有替换掉 \(i\) 的右端点 \(j\) ,此时的 \(f_{i+1,j}\) 一定小于 \(f_{i+1,pos_i}\) ,考虑替换掉 \(i\) 的右端点 \(j\) ,假设替换 \(a_i\) 的元素为 \(x\) ,显然有 \(x\ge a_i\) ,显然此时存在两种情况: \(x\) 用于替换其他元素或 \(x\) 不在被用于替换,后面的情况很容易考虑,很显然 \(f_{i,j}-x\le f_{i,pos_i}-a_i\) ,因此成立,考虑 \(x\) 用于替换其他元素的情况,容易发现区间 \([i+1,pos_i]\) 的替换方案中一定也可以去替换这个元素,仍然可以发现 \(f_{i+1,pos_i}\ge f_{i+1,j}\) 。

因此考虑分治法优化决策单调性,考虑求解 \(f(l,r)\) ,假设当前我们将 \(a\) 取出区间 \([l,r]\) 并从小到大排序,将 \(b\) 从大到小排序,现在我们需要找到一个最大的 \(k\) ,满足 \(a_k<b_k\) ,很容易用权值线段树维护,由于指针的移动次数为 \(O(n\log n)\) ,因此总复杂度为 \(O(n\log^2n)\) 。

code

#include <cstdio>
#include <algorithm>
using namespace std;
const int max1 = 1e5;
const int inf = 0x3f3f3f3f;
int n, m, a[max1 + 5], b[max1 + 5];
long long sum[max1 + 5];
int save[max1 + 5], len;
struct Segment_Tree
{
    #define lson(now) ( now << 1 )
    #define rson(now) ( now << 1 | 1 )
    struct Data
        { int maxval, count; long long sum; } tree[max1 * 4 + 5];
    void Push_Up ( int now )
    {
        tree[now].maxval = max(tree[lson(now)].maxval, tree[rson(now)].maxval);
        tree[now].count = tree[lson(now)].count + tree[rson(now)].count;
        tree[now].sum = tree[lson(now)].sum + tree[rson(now)].sum;
        return;
    }
    void Build ( int now, int L, int R )
    {
        tree[now].maxval = -inf;
        tree[now].count = 0;
        tree[now].sum = 0LL;
        if ( L == R )
            return;
        int mid = L + R >> 1;
        Build(lson(now), L, mid);
        Build(rson(now), mid + 1, R);
        return;
    }
    void Insert ( int now, int L, int R, int pos, int x )
    {
        if ( L == R )
        {
            tree[now].maxval = -inf;
            tree[now].count += x;
            tree[now].sum += 1LL * x * save[L];
            if ( tree[now].count )
                tree[now].maxval = save[L];
            return;
        }
        int mid = L + R >> 1;
        if ( pos <= mid )
            Insert(lson(now), L, mid, pos, x);
        else
            Insert(rson(now), mid + 1, R, pos, x);
        Push_Up(now);
        return;
    }
    void Insert ( int pos, int x )
        { return Insert(1, 1, len, pos, x); }
    long long Query ( int now, int L, int R, int start )
    {
        if ( L == R )
        {
            int l = start + 1, r = min(m, start + tree[now].count), pos = start;
            while ( l <= r )
            {
                int mid = l + r >> 1;
                if ( b[mid] > save[L] )
                    pos = mid, l = mid + 1;
                else
                    r = mid - 1;
            }
            return sum[pos] + 1LL * ( start + tree[now].count - pos ) * save[L];
        }
        int mid = L + R >> 1;
        if ( start + tree[lson(now)].count <= m && tree[lson(now)].maxval < b[start + tree[lson(now)].count] )
            return Query(rson(now), mid + 1, R, start + tree[lson(now)].count);
        return Query(lson(now), L, mid, start) + tree[rson(now)].sum;
    }
    long long Query ()
        { return Query(1, 1, len, 0); }
}Tree;
int nowl, nowr;
long long Solve ( int L, int R, int ql, int qr )
{
    if ( L > R )
        return 1LL * -inf * inf;
    int mid = L + R >> 1;
    while ( nowr < max(mid, ql) - 1 )
        Tree.Insert(a[++nowr], 1);
    while ( nowl > mid )
        Tree.Insert(a[--nowl], 1);
    while ( nowr > max(mid, ql) - 1 )
        Tree.Insert(a[nowr--], -1);
    while ( nowl < mid )
        Tree.Insert(a[nowl++], -1);
    long long ans = 1LL * -inf * inf;
    int pos = 0;
    while ( nowr < qr )
    {
        Tree.Insert(a[++nowr], 1);
        long long up = Tree.Query();
        if ( up > ans )
            ans = up, pos = nowr;
    }
    return max(ans, max(Solve(L, mid - 1, ql, pos), Solve(mid + 1, R, pos, qr)));
}
int main ()
{
    scanf("%d%d", &n, &m);
    for ( int i = 1; i <= n; i ++ )
        scanf("%d", &a[i]), save[i] = a[i];
    for ( int i = 1; i <= m; i ++ )
        scanf("%d", &b[i]);
    sort(b + 1, b + 1 + m);
    reverse(b + 1, b + 1 + m);
    sum[0] = 0;
    for ( int i = 1; i <= m; i ++ )
        sum[i] = sum[i - 1] + b[i];
    sort(save + 1, save + 1 + n);
    len = unique(save + 1, save + 1 + n) - ( save + 1 );
    for ( int i = 1; i <= n; i ++ )
        a[i] = lower_bound(save + 1, save + 1 + len, a[i]) - save;
    nowl = 1, nowr = 0;
    Tree.Build(1, 1, len);
    printf("%lld\n", Solve(1, n, 1, n));
    return 0;
}

T3 寻宝

由于给定的图是一张二分图,因此考虑对其进行二分图染色,一个比较暴力的想法是让每层之间的边指向藏宝点,于是可以得到一个 \(O(n)\) 的暴力做法,正解考虑随机化,设置阈值 \(B\) ,如果我们找到了一个深度位于 \(B\) 以内的点,复杂度可以做到 \(O(B)\) ,同时期望在 \(\tfrac{n}{B}\) 次随机得到这个点,因此考虑判断当前点深度是否小于 \(B\) ,可以得到一个想法是(我也不知道怎么想到的), \(B\) 层以内的点指向藏宝点, \(B\) 层以外的点交替指向藏宝箱和与藏宝点所在方向相反,容易发现 \(B\) 层以外的点满足出度或入度为 \(0\) ,可以直接判断。

T4 字符树

这显然是一道暴力题。

首先对整棵树树剖,维护线段树,树上每个节点维护 unordered_map 记录这个区间内所有长度为 \(X\) 的子串的哈希值,合并区间时可以取出两个区间的长度为 \(X\) 的前后缀暴力合并,复杂度 \(O(n\log^2n)\) 。(真正考试我大概一定调不出来吧)

标签:__,int,mid,pos,2022,front,now,THUSC
From: https://www.cnblogs.com/KafuuChinocpp/p/17354116.html

相关文章

  • 2022-04-25:给定两个长度为N的数组,a[]和b[] 也就是对于每个位置i来说,有a[i]和b[i]两个
    2022-04-25:给定两个长度为N的数组,a[]和b[]也就是对于每个位置i来说,有a[i]和b[i]两个属性ia[i]b[i]ja[j]b[j]现在想为了i,选一个最好的j位置,搭配能得到最小的如下值:(a[i]+a[j])^2+b[i]+b[j]我们把这个最小的值,定义为i的最in值比如:a={2,3,6,5,1......
  • ApacheCN 翻译/校对活动进度公告 2022.5.4
    命令行上的数据科学第二版【校对】仓库:https://github.com/apachecn/ds-cmd-line-2e-zh整体进度:https://github.com/apachecn/ds-cmd-line-2e-zh/issues/1贡献指南:https://github.com/apachecn/ds-cmd-line-2e-zh/blob/master/CONTRIBUTING.md进度:0/14从零开始的计算机图形学【校对......
  • 2022年终总结
    今年主要成就是:(1)身体调整到最佳状态(2)工作有些变动但最终换成了自己满意的工作(我的优先级:研究院>券商>web3>传统工业>web2)(3)入门玄学并认识了很多朋友(4)工作站各项配件的配置拉满(双路7t83,sx1000lpt,ssgx2,p4800xx2,利民b12)(5)确立布客社区的四大方向,并且开设讨论群,逐渐走向正轨。明年规划:(1)玄......
  • 2022CSP游记
    目录CSP-J20227:458:158:278:389:129:2310:3411:57中午CSP-S20222:274:156:12估分普及提高自查出分废物鸭子菜菜菜CSP-J2022废了7:45跟随校车到了考场,纪中考点不给矿泉水可还行老朋友都见到了LJHDZRLAFZWTWTCZHWYWJ....WTC已经是ISIJ的金牌了,当年我还跟他是一个......
  • GB/T28181-2022相对2016版“基于TCP协议的视音频媒体传输要求“规范解读和技术实现
    规范解读GB/T28181-2022和GB/T28181-2016规范,有这么一条“更改了附录D基于TCP协议的视音频媒体传输要求(见附录D,2016年版的附录L)。”。本文主要是针对GB/T28181-2022里面提到的“基于TCP协议的视音频媒体传输要求”做相应的接口适配,在此之前,我们先回顾下规范里面针对这部分......
  • 题解:【CTS2022】 独立集问题
    题目链接来自2023SDPT-Round1-Day4课上Qingyu的讲解。考虑对于一个点多次操作会发生什么?第一次操作会将周围的点的权值吸过来,自己对答案的贡献乘\(-1\),周围的点的贡献乘\(+1\),得到新的权值\(a_x'=\pma_x\mp\sum_{y\inson_x}a_y\);再一次操作,我们可以将这个新的贡......
  • [2022编思1062]找出最少动作数
    [2022编思1062]找出最少动作数题面有一个栈,这个栈有\(m\)个状态,每个状态记为\(S_i\)每个状态里面有\(n\)种数字,数字\(i\)有\(a_i\)个。考虑从全空,依次经历\(S_1...S_m\),让操作数最小化。sov是一个神奇的区间DP。考虑对于某个区间\(S_i...S_j\),从开始塞进去不用动的数字有\(......
  • 关于在visual Studio 2022中无法找到 ASP.NET Core Web Application 或 ASP.NET Core
    在学习ASP.NETCoreWebApplication时发现无论如何都无法找到这个模板,在翻遍论坛后都没有看到解决的方法,在我下载 visualStudio2017中终于找到了但是,你会发现他只能选择.netcore2.0这肯定是不符合我们写代码的,因为他太老了,但在2022中确实找不到    这......
  • 2022-第十三届蓝桥杯大赛个人赛省赛(软件类)真题C大学C组
    返回目录题目一览:A.排列字母B.特殊时间C.纸张尺寸D.求和E.数位排序F.选数异或G.消除游戏H.重新排序I.技能升级J.重复的数   A.排列字母   B.特殊时间   C.纸张尺寸   D.求和   E.数位排序   F.选数异或   G.消除游戏......
  • 2022-第十三届蓝桥杯大赛个人赛省赛(软件类)真题C大学A组
    返回目录题目一览:A.裁纸刀B.灭鼠先锋C.求和D.选数异或E.爬树的甲壳虫F.青蛙过河G.最长不下降子序列H.扫描游戏I.数的拆分J.推导部分和   A.裁纸刀   B.灭鼠先锋   C.求和   D.选数异或   E.爬树的甲壳虫   F.青蛙过河  ......