[POJ3723]Conscription[并查集][kruskal]

题目链接:[POJ3723]Conscription[并查集][kruskal]

第一次做并查集的题目,其实之前我连什么是并查集豆不知道TAT

今儿一天就混这两个概念了,mark下。代码中有些注释,诸君请便。

关于并查集的理解,推荐一篇博文:快点我~~

这题kruskal找到的是所有人都纳入进来,能得到的最大回扣值。

具体代码如下:

#include <iostream>
#include <cstring>
#include <algorithm>
#include <cstdio>
using namespace std;
const int MAXN = 20200;
int n, m, r;
struct Edge{
    int u, v, cost;
    bool operator < (const Edge& rhs) const{
        return cost < rhs.cost;
    }
}G[50005];

int par[MAXN], ran[MAXN];

void init(int n){
    for (int i = 0; i < n; ++i)
    {
        par[i] = i;  //parant数组。记录点的父亲节点
        ran[i] = 0;  //ran数组。记录节点高度
    }
    return;
}

int find(int x){ //查找父亲节点,如果父亲节点不是自己,接着往上查找
    if (par[x] == x) return x;
    else return par[x] = find(par[x]); //压缩树,使每个节点都直接指向父节点,节约查找时间
}

void unite(int x, int y){ //低位节点向高位节点合并
    x = find(x); y = find(y);
    if (ran[x] < ran[y]) par[x] = y;
    else {
        par[y] = x;
        if (ran[x] == ran[y]) ++ran[x];
    }
    return;
}

bool same(int x, int y){  //是否拥有同一个父节点
    return find(x) == find(y);
}

int kruskal(){
    int ret = 0;
    sort(G + 1, G + 1 + r);
    init(n + m);
    for (int i = 1; i <= r; ++i){
        Edge &e = G[i];
        if (!same(e.u, e.v)){
            ret += e.cost;
            unite(e.u, e.v);
        }
    }      //小mark:当两个子节点被联结后,如果又有一个同样的关系,这样联结的结果就是不记入钱数,满足发生两人多次关系取其最大减少费用这一情况。(毕竟排过序嘛)
    return ret;
}

int main() {
    ios_base::sync_with_stdio(0);
    int t; scanf("%d", &t);
    while(t--){
        scanf("%d%d%d",&n, &m, &r);
        for (int i = 1; i <= r; ++i){
            int x, y, v; scanf("%d%d%d", &x, &y, &v);
            G[i] = (Edge){x, n + y, -v};
        }
        int ans = 10000 * (n + m) + kruskal();
        printf("%d\n",ans);
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,这个问题属于技术问题。以下是一个简单的Python模拟登录POJ提交代码并抓取评测结果的代码示例: ```python import requests # 登录POJ,获取cookie def login(username, password): s = requests.Session() login_url = "http://poj.org/login" login_data = { "user_id1": username, "password1": password, "B1": "login", "url": "/" } s.post(login_url, data=login_data) return s # 提交代码 def submit_code(s, problem_id, language, source_code): submit_url = "http://poj.org/submit" submit_data = { "problem_id": problem_id, "language": language, "source": source_code } s.post(submit_url, data=submit_data) # 获取评测结果 def get_result(s, run_id): status_url = "http://poj.org/status" params = { "user_id": "", "result": "", "language": "", "top": run_id } r = s.get(status_url, params=params) table_start = r.text.find("<table cellpadding=0 cellspacing=0 border=0 width=100%>") table_end = r.text.find("</table>", table_start) table_html = r.text[table_start:table_end + 8] return table_html # 使用示例 username = "your_username" password = "your_password" problem_id = "1000" language = "G++" source_code = """ #include <iostream> using namespace std; int main() { int a, b; cin >> a >> b; cout << a + b << endl; return 0; } """ s = login(username, password) submit_code(s, problem_id, language, source_code) table_html = get_result(s, "12345678") # 替换成实际提交的run id print(table_html) ``` 其中,`login`函数模拟登录POJ并返回一个`Session`对象,`submit_code`函数提交代码,`get_result`函数获取评测结果。你可以根据实际需要修改代码中的`username`、`password`、`problem_id`、`language`和`source_code`等参数,并替换`get_result`函数中的`run_id`。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值