結果
| 問題 |
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]); }
}