結果
問題 | No.2642 Don't cut line! |
ユーザー | abap34 |
提出日時 | 2024-02-15 01:20:31 |
言語 | Julia (1.10.2) |
結果 |
AC
|
実行時間 | 2,582 ms / 4,000 ms |
コード長 | 4,802 bytes |
コンパイル時間 | 171 ms |
コンパイル使用メモリ | 6,816 KB |
実行使用メモリ | 509,160 KB |
最終ジャッジ日時 | 2024-10-01 17:42:07 |
合計ジャッジ時間 | 73,606 ms |
ジャッジサーバーID (参考情報) |
judge3 / judge1 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 1,374 ms
321,184 KB |
testcase_01 | AC | 2,549 ms
503,084 KB |
testcase_02 | AC | 2,490 ms
505,892 KB |
testcase_03 | AC | 2,568 ms
503,916 KB |
testcase_04 | AC | 2,541 ms
505,268 KB |
testcase_05 | AC | 2,532 ms
509,160 KB |
testcase_06 | AC | 1,833 ms
376,336 KB |
testcase_07 | AC | 1,625 ms
377,328 KB |
testcase_08 | AC | 1,631 ms
376,456 KB |
testcase_09 | AC | 1,641 ms
375,932 KB |
testcase_10 | AC | 1,702 ms
376,192 KB |
testcase_11 | AC | 1,663 ms
376,548 KB |
testcase_12 | AC | 1,664 ms
377,296 KB |
testcase_13 | AC | 1,703 ms
376,112 KB |
testcase_14 | AC | 1,641 ms
376,068 KB |
testcase_15 | AC | 1,686 ms
377,124 KB |
testcase_16 | AC | 2,549 ms
503,620 KB |
testcase_17 | AC | 2,254 ms
461,336 KB |
testcase_18 | AC | 2,505 ms
475,832 KB |
testcase_19 | AC | 2,021 ms
429,928 KB |
testcase_20 | AC | 1,787 ms
370,156 KB |
testcase_21 | AC | 1,736 ms
377,056 KB |
testcase_22 | AC | 2,167 ms
441,880 KB |
testcase_23 | AC | 2,582 ms
498,476 KB |
testcase_24 | AC | 1,800 ms
372,252 KB |
testcase_25 | AC | 1,778 ms
374,860 KB |
testcase_26 | AC | 1,712 ms
376,240 KB |
testcase_27 | AC | 2,006 ms
423,372 KB |
testcase_28 | AC | 2,479 ms
475,800 KB |
testcase_29 | AC | 1,741 ms
378,148 KB |
testcase_30 | AC | 1,800 ms
367,452 KB |
testcase_31 | AC | 2,144 ms
442,764 KB |
testcase_32 | AC | 1,740 ms
370,516 KB |
testcase_33 | AC | 1,328 ms
321,316 KB |
testcase_34 | AC | 1,322 ms
320,960 KB |
testcase_35 | AC | 1,321 ms
320,812 KB |
ソースコード
const INF = 10^16 struct UnionFind par::Vector{Int} size::Vector{Int} UnionFind(N) = new(collect(1:N), collect(1:N)) end function root!(uf::UnionFind, x::Int) if uf.par[x] == x return x else return uf.par[x] = root!(uf, uf.par[x]) end end function issame!(uf::UnionFind, x::Int, y::Int) return root!(uf, x) == root!(uf, y) end function unite!(uf::UnionFind, x::Int, y::Int) x = root!(uf, x) y = root!(uf, y) (x == y) && (return true) if (uf.size[x] < uf.size[y]) uf.par[x] = y uf.size[y] += uf.size[x] else uf.par[y] = x uf.size[x] += uf.size[y] end return true end struct Edge from::Int to::Int weight::Int profit::Int end function reverse(edge::Edge) return Edge(edge.to, edge.from, edge.weight, edge.profit) end struct Graph edges::Vector{Vector{Edge}} function Graph(N) edges = [Vector{Edge}() for _ in 1:N] new(edges) end end function all_edges(graph::Graph)::Vector{Edge} return collect(Iterators.flatten(graph.edges)) end function add!(graph::Graph, edge::Edge; has_dir=false) push!(graph.edges[edge.from], edge) if !has_dir rev_edge = reverse(edge) push!(graph.edges[edge.to], rev_edge) end end function as_rooted_tree(graph::Graph, root::Int)::Graph N = length(graph.edges) tree = Graph(N) function dfs(v, p) for u in graph.edges[v] if u.to != p add!(tree, u, has_dir=true) dfs(u.to, v) end end end dfs(root, -1) return tree end function kruskal(graph::Graph; by=identity, rev=false)::Tuple{Graph,Int,Int} N = length(graph.edges) edges = sort(all_edges(graph), by=by, rev=rev) uf = UnionFind(length(graph.edges)) graph = Graph(N) cost = 0 profit = -1 for edge in edges if !issame!(uf, edge.from, edge.to) unite!(uf, edge.from, edge.to) add!(graph, edge) cost += edge.weight profit = max(profit, edge.profit) end end return graph, cost, profit end struct LCA parent::Vector{Vector{Int}} dist::Vector{Vector{Int}} depth::Vector{Int} K::Int V::Int function LCA(graph::Graph; root=1) V = length(graph.edges) K = ceil(Int, log2(V)) tree = as_rooted_tree(graph, root) parent = [fill(-1, V) for _ in 1:K] dist = [fill(-1, V) for _ in 1:K] depth = fill(-1, V) function dfs(v, p, d) depth[v] = d parent[1][v] = p for u in tree.edges[v] if u.to != p dist[1][u.to] = u.weight dfs(u.to, v, d + 1) end end end dfs(root, -1, 0) for k in 1:K-1 for v in 1:V if parent[k][v] == -1 continue else parent[k+1][v] = parent[k][parent[k][v]] dist[k+1][v] = max(dist[k][v], dist[k][parent[k][v]]) end end end return new(parent, dist, depth, K, V) end end function max_cost(lca::LCA, u::Int, v::Int)::Int cost = -INF if lca.depth[u] < lca.depth[v] u, v = v, u end diff = lca.depth[u] - lca.depth[v] for k in 0:lca.K-1 if (diff >> k) & 1 == 1 cost = max(cost, lca.dist[k+1][u]) u = lca.parent[k+1][u] end end if u == v return cost end for k in lca.K:-1:1 if lca.parent[k][u] != lca.parent[k][v] cost = max(cost, lca.dist[k][u]) cost = max(cost, lca.dist[k][v]) u = lca.parent[k][u] v = lca.parent[k][v] end end cost = max(cost, lca.dist[1][u]) cost = max(cost, lca.dist[1][v]) return cost end function main() N, K, C = parse.(Int, split(readline())) graph = Graph(N) for _ in 1:K u, v, w, p = parse.(Int, split(readline())) edge = Edge(u, v, w, p) add!(graph, edge) end mst, C_m, P_m = kruskal(graph, by=(x -> x.weight)) mst_edges_set = Set{Edge}(all_edges(mst)) mst_tree = as_rooted_tree(mst, 1) lca = LCA(mst_tree) cand_edges = all_edges(graph) sort!(cand_edges, by=(x -> x.profit), rev=true) for edge in cand_edges P = edge.profit if (edge in mst_edges_set) C_add = 0 C_rem = 0 else u, v, C_add = edge.from, edge.to, edge.weight C_rem = max_cost(lca, u, v) end if C_m + C_add - C_rem <= C println(P) return end end println(-1) end main()