結果
問題 |
No.3238 Shadow
|
ユーザー |
|
提出日時 | 2025-08-15 22:47:11 |
言語 | C# (.NET 8.0.404) |
結果 |
AC
|
実行時間 | 405 ms / 2,000 ms |
コード長 | 3,255 bytes |
コンパイル時間 | 8,293 ms |
コンパイル使用メモリ | 169,308 KB |
実行使用メモリ | 251,936 KB |
最終ジャッジ日時 | 2025-08-15 22:47:27 |
合計ジャッジ時間 | 13,520 ms |
ジャッジサーバーID (参考情報) |
judge3 / judge4 |
(要ログイン)
ファイルパターン | 結果 |
---|---|
sample | AC * 3 |
other | AC * 17 |
コンパイルメッセージ
復元対象のプロジェクトを決定しています... /home/judge/data/code/main.csproj を復元しました (102 ミリ秒)。 main -> /home/judge/data/code/bin/Release/net8.0/main.dll main -> /home/judge/data/code/bin/Release/net8.0/publish/
ソースコード
#nullable enable #region var _input = Array.Empty<string>(); var _iter = 0; string String() { while (_iter >= _input.Length) (_input, _iter) = (Console.ReadLine()!.Split(' '), 0); return _input[_iter++]; } T I<T>() where T : IParsable<T> => T.Parse(String(), null); #endregion T[] Range<T>(int n, Func<T> F) => Enumerable.Range(0, n).Select(_ => F()).ToArray(); var n = I<int>(); var pz = Range(n, () => I<int>() - 1); var ans = new List<(int, int)>(); var rev = new int[n]; for (var i = 0; i < n; i++) rev[pz[i]] = i; var tree = SegmentTree.Build(pz, new Op()); var left = n; while (left > 0) { var r = n; var xm = n; var ym = n; while (true) { var my = tree.Product(0, r); if (my >= n) break; var i = rev[my]; ym = Math.Min(ym, my); xm = Math.Min(xm, i); r = i; tree.Set(rev[my], n); left--; } ans.Add((xm, ym)); } string Join<T>(IEnumerable<T> values, bool ws = false) => string.Join(ws ? " " : Environment.NewLine, values); Console.WriteLine(Join(ans.Select(e => (e.Item1 + 1) + " " + (e.Item2 + 1)))); readonly struct Op : ISegmentTreeOperator<int> { public int IdentityElement => int.MaxValue; public int Operation(int x, int y) => int.Min(x, y); } class SegmentTree { public static SegmentTree<TMonoid, TOperator> Build<TMonoid, TOperator>(TMonoid[] initial, TOperator op) where TOperator : struct, ISegmentTreeOperator<TMonoid> => new(initial, op); } interface ISegmentTreeOperator<TMonoid> { TMonoid IdentityElement { get; } TMonoid Operation(TMonoid x, TMonoid y); } partial class SegmentTree<TMonoid, TOperator> where TOperator : struct, ISegmentTreeOperator<TMonoid> { public int BaseSize { get; init; } protected int Height { get; init; } protected TOperator Op { get; init; } protected TMonoid[] Values { get; init; } public SegmentTree(TMonoid[] initial, TOperator op = default) { Op = op; (BaseSize, Height) = (1, 0); while (BaseSize < initial.Length) (BaseSize, Height) = (BaseSize << 1, Height + 1); var size = BaseSize << 1; Values = new TMonoid[size]; var values = Values.AsSpan(); for (var i = 0; i < size; i++) values[i] = op.IdentityElement; for (var i = 0; i < initial.Length; i++) values[i + BaseSize] = initial[i]; for (var i = BaseSize - 1; i > 0; i--) Update(values, i); } public TMonoid Product(int l, int r) { (l, r) = (l + BaseSize, r + BaseSize); var (resL, resR) = (Op.IdentityElement, Op.IdentityElement); var values = Values.AsSpan(); while (l < r) { if ((l & 1) == 1) resL = Op.Operation(resL, values[l++]); if ((r & 1) == 1) resR = Op.Operation(values[--r], resR); (l, r) = (l >> 1, r >> 1); } return Op.Operation(resL, resR); } public void Set(int index, TMonoid value) { var j = index + BaseSize; var values = Values.AsSpan(); values[j] = value; for (var i = 1; i <= Height; i++) Update(values, j >> i); } protected void Update(Span<TMonoid> values, int k) { values[k] = Op.Operation(values[k << 1], values[(k << 1) + 1]); } }