CSP22.3 T4通信系统管理

之前在CCF CSP认证2022年3月完整题解这篇博客记录了自己花了两天时间乱搞出来的方法,但是实际上动态维护区间最值,通过 s e t set set实现会更简洁,用优先队列需要额外开数组记录节点的实时信息,而且堆中节点需要一个有效标记——失效时间,因为节点失效之后不会立刻从堆中 erase,而是作为堆顶弹出。因此需要这样一个标记判断节点是否有效,是否代表了实时流量,是否应弹出。

然而这样设计导致在流量变化时需要更新该标记,而两个节点之间可能存在多股流量,应使用最近失效时间更新标记,因此我通过结构 m a p < p a i r < i n t , i n t > , p r i o r i t y _ q u e u e < i n t , v e c t o r < i n t > > , g r e a t e r < i n t > > map<pair<int,int>,priority\_queue<int,vector<int>>,greater<int>> map<pair<int,int>,priority_queue<int,vector<int>>,greater<int>> 维护所有失效时间点,并快速查找最近者。

还存在空间问题,节点数量为 1 e 5 1e5 1e5,而 ( 1 e 5 ) 2 = 1 e 10 (1e5)^2=1e10 (1e5)2=1e10,因此维护实时信息的数组只能动态使用空间(用 v e c t o r vector vector p u s h _ b a c k push\_back push_back e r a s e erase erase 方法)。

更多地,我的代码封装几乎没有,直接 copy 改一改不用动脑省事呵呵,导致最后行数 250+。。。

综上,通过 s e t set set 动态维护区间最值为最佳。

摘自之前的 blog:
在这里插入图片描述

在处理额度失效上,我也使用了最小堆,类似一个“沙漏”,其实没必要,甚至增加了复杂度。用 m a p < l l , t u p l e > map<ll,tuple> map<ll,tuple> 即可,tuple 存储端点,和失效额度值。实际上,时间是以天为单位连续的,范围为1e5,用数组存储也可以,用 m a p map map 是动态开辟空间,牺牲时间换空间。

之前我将问题复杂化主要是不了解 s e t set set m a p map map 这两个结构的底层实现是平衡树,节点之间存在大小关系,通过迭代器访问获得的是有序的节点序列,这也要求我们定义结构体储存与这两个结构时重载小于号,定义这种大小关系。

以下是看了 yhf 学长的 live coding ,自己写了一遍并整理思路。

思路

主要通信对象
维护每个节点的主要通信对象,而节点和其他诸多节点时间有多个流量(额度),要维护其中最大者,流量相同则编号最小者,定义流量如下:

struct node {
    ll v; int to;

    node(ll v, int to) : v(v), to(to) {}
    bool operator < (const node &d) const {
        return v == d.v ? to < d.to : v > d.v;
    }
};

存储于 s e t < n o d e > d [ m a x n ] set<node> d[maxn] set<node>d[maxn] 结构,如何更新?将原流量 erase ,重新 insert 新流量。最大者为 d [ i ] . b e g i n ( ) − > t o d[i].begin()->to d[i].begin()>to.

孤岛、通信对
通过 i s l o n e l y islonely islonely 函数判断更新前后节点是否为孤岛,判据为没有流量或者流量为0,;
通过 i s p a i r ispair ispair 判断节点是否包含“通信对”关系,首先节点不为孤岛,然后,找到它的主要通信对象 y y y ,看 y y y 的对象是否是自己,这里注意,由于 w o r k work work 对节点对的对称操作是先后进行,不同步(实际肯定是同时发生), y y y 可能先操作并且变为“孤岛”了,因此要保证 y y y 不是“孤岛”:

return (!islonely(y) && d[y].begin()->to == x);

代码

using ll = long long;
const int maxn = 1e5 + 10;

// 通信主要通信对象
// 通信孤岛、通信对
// n, m:1e5

struct node {
    ll v; int to;

    node(ll v, int to) : v(v), to(to) {}
    bool operator < (const node &d) const {
        return v == d.v ? to < d.to : v > d.v;
    }
};

struct info {
    int u, v, x;
    info(int u, int v, int x) : u(u), v(v), x(x) {}
};

set<node> d[maxn];
map<pair<int, int> , ll> save;  // 维护点对实时额度
vector<info> decr[maxn];
int pv, qv; // 通信孤岛、通信对数

int islonely(int x) {
    return (d[x].begin() == d[x].end() || d[x].begin()->v == 0);
}   // 检查是否孤岛

int ispair(int x) {
    if (islonely(x)) return 0;
    int y = d[x].begin()->to;
    return (!islonely(y) && d[y].begin()->to == x);
}   // 检查是否包含通讯对

void work(int u, int v, int x) {
    ll origVal = save[{u, v}];
    save[{u, v}] += x;

    pv -= islonely(u);
    qv -= ispair(u);

    node orig(origVal, v);
    d[u].erase(orig);
    d[u].emplace(save[{u, v}], v);

    pv += islonely(u);
    qv += ispair(u);
}   // 节点u的额度申请、过期

void solve() {
    int n, m;
    cin >> n >> m;
    pv = n; qv = 0;

    for (int i = 1; i <= m; i++) {
        // 处理过期额度
        for (const auto &t : decr[i]) {
            work(t.u, t.v, -t.x);
            work(t.v, t.u, -t.x);
        }
        int k, l, num, p, q;
        // 当天额度申请
        cin >> k;
        int u, v, x, y;
        for (int j = 1; j <= k; j++) {
            cin >> u >> v >> x >> y;
            if (i + y <= m) decr[i + y].emplace_back(u, v, x);
            work(u, v, x);
            work(v, u, x);
        }
        // 查询通信主要通信对象
        cin >> l;
        for (int j = 1; j <= l; j++) {
            cin >> num;
            if (islonely(num)) {
                cout << "0\n";
            }
            else {
                cout << d[num].begin()->to << '\n';
            }
        }
        // 查询通信孤岛、通信对
        cin >> p >> q;
        if (p) cout << pv << '\n';
        if (q) cout << qv << '\n';

//        cout << '\n';
    }
}
  • 5
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 15
    评论
### 回答1: CCF-CSP(中国计算机学会认证系统程序员)认证考试是由中国计算机学会主办的,旨在选拔高水平的系统程序员和开发人员。该认证考试包括两个级别:初级和高级。初级考试主要测试候选人的基础知识和编程能力,高级考试则主要测试候选人的系统设计和开发能力。CCF-CSP认证考试的评分标准严格,要求考生表现出扎实的计算机理论基础和熟练的编程技巧。 为了顺利通过CCF-CSP认证考试,考生需要掌握扎实的计算机基础知识和熟练的编程技巧,并且需要在考试前进行充分的准备。以下是一些复习技巧指导: 1. 确认考试大纲 考生需要仔细阅读考试大纲,了解考试范围和题型。要特别注意考试大纲中的重点内容,重点复习。 2. 提前规划复习进度 准备CCF-CSP认证考试需要时间,而时间是很宝贵的,考生需要提前规划好复习进度和复习内容,尽可能利用零散时间进行复习。 3. 多做题 要想掌握CCF-CSP认证考试所需的知识和技能,考生需要多做题,并针对性地练习一些经典的考题。在练习中,考生需要注重练习思路和解题方法,该方法可以加深对知识点的理解。 4. 关注考试动态 考试动态指的是考试历年来的趋势和变化,考生需要不断关注考试动态,了解考试趋势,及时更新复习内容和方法,以便更好地应对考试。 总之,要想通过CCF-CSP认证考试,考生需要理论基础和开发经验充足,并且需要在复习时切实遵循学科规范,灵活应用知识点,较好地掌握考试技巧,这是顺利通过CCF-CSP认证考试的关键。 ### 回答2: ccf-csp认证考试是中国计算机学会(CCF)认证的一个计算机软件专业技能等级考试,也是中国IT行业里非常重要的证书之一。该考试分为初级、中级和高级三个等级,其中初级考试包含了基础的程序设计和算法等内容,中级考试中主要考察了软件工程、数据结构算法设计和网络安全等领域的知识,而高级考试则更注重的是对软件工程和软件项目管理的理解和应用,以及对复杂系统的设计和实现方法的掌握。为了能够成功通过ccf-csp认证考试,首先需要在考前充分准备,并且需要有一些好的复习技巧,这里向大家推荐以下几点: 1.学好基础知识。初级、中级和高级csp认证考试所涉及到的内容都是建立在一定的基础知识上的。由于考试的难度逐级递增,所以建议在准备考试之前先花时间学好基础知识。 2.多做练习题。做练习题能够帮助我们加深对知识的理解,熟悉考试形式和规则,同时也能提高我们的答题速度和准确度。 3.注意时间管理。不同级别的csp认证考试都有时限,在考试过程中需要注意时间的分配。建议在每次做练习题的时候都设置一个时间限制,并逐步缩短时间限制,以提高答题效率。 4.了解考试规则。在考前一定要熟悉考试规则和题型,知道考试时可以携带哪些资料和工具,了解考试的答题和评分规则等。 总之,ccf-csp认证考试是一项非常重要的认证考试,通过这个认证不仅可以增加个人的职业竞争力,还能证明个人在软件开发和IT行业方面的专业能力。因此,我们需要认真地准备和复习,字斟句酌地做好每一个细节,以确保顺利通过考试。 ### 回答3: ccf-csp认证考试是由中国计算机学会主办的全国性计算机考试,考试内容涵盖了计算机应用、技术、算法等多个方面,分为初级、中级和高级三个等级。参加此考试不仅有助于检测个人的计算机专业水平,更能提升自身的竞争力和职业发展前景。 为了通过ccf-csp认证考试,考生需要重点复习考试要求中的各个内容模块,如数据类型、运算符、控制结构、数组、函数、文件操作等程序设计相关方面;还包括计算机网络、操作系统、数据库等计算机技术类知识;另外,还需要掌握常见算法数据结构,如排序、查找、树等。此外,考生还需要通过大量的编程练习和模拟题练习,熟悉考试题型和答题技巧,提升自己的完成效率和准确性。 在复习过程中,需要注重细节和实践。例如,对于程序设计相关方面,需要理解每个知识点的原理和应用场景,在编写代码时要注重细节避免出现错误。对于算法数据结构方面,需要熟悉常见算法的实现方式和优缺点,掌握好算法的时间复杂度和空间复杂度,做到熟练掌握。 总之,通过充分准备和钻研,考生有望成功通过ccf-csp认证考试,展现自己的计算机专业能力和职业素养。
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

u小鬼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值