結果
問題 | No.230 Splarraay スプラレェーイ |
ユーザー | 明智重蔵 |
提出日時 | 2021-11-07 09:49:17 |
言語 | C# (.NET 8.0.203) |
結果 |
AC
|
実行時間 | 694 ms / 5,000 ms |
コード長 | 9,764 bytes |
コンパイル時間 | 12,566 ms |
コンパイル使用メモリ | 167,136 KB |
実行使用メモリ | 214,348 KB |
最終ジャッジ日時 | 2024-11-08 20:28:44 |
合計ジャッジ時間 | 18,605 ms |
ジャッジサーバーID (参考情報) |
judge5 / judge1 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 59 ms
30,072 KB |
testcase_01 | AC | 61 ms
30,336 KB |
testcase_02 | AC | 61 ms
30,336 KB |
testcase_03 | AC | 58 ms
30,464 KB |
testcase_04 | AC | 59 ms
30,328 KB |
testcase_05 | AC | 63 ms
30,336 KB |
testcase_06 | AC | 84 ms
32,896 KB |
testcase_07 | AC | 61 ms
30,720 KB |
testcase_08 | AC | 65 ms
30,720 KB |
testcase_09 | AC | 339 ms
51,328 KB |
testcase_10 | AC | 268 ms
61,824 KB |
testcase_11 | AC | 218 ms
42,880 KB |
testcase_12 | AC | 335 ms
51,584 KB |
testcase_13 | AC | 98 ms
33,664 KB |
testcase_14 | AC | 224 ms
65,280 KB |
testcase_15 | AC | 432 ms
65,664 KB |
testcase_16 | AC | 535 ms
68,980 KB |
testcase_17 | AC | 694 ms
70,272 KB |
testcase_18 | AC | 354 ms
69,504 KB |
testcase_19 | AC | 434 ms
214,348 KB |
コンパイルメッセージ
復元対象のプロジェクトを決定しています... /home/judge/data/code/main.csproj を復元しました (91 ms)。 MSBuild のバージョン 17.9.6+a4ecab324 (.NET) main -> /home/judge/data/code/bin/Release/net8.0/main.dll main -> /home/judge/data/code/bin/Release/net8.0/publish/
ソースコード
using System; using System.Collections.Generic; using System.Linq; // https://yukicoder.me/problems/no/230 class Program { static string InputPattern = "InputX"; static List<string> GetInputList() { var WillReturn = new List<string>(); if (InputPattern == "Input1") { WillReturn.Add("5"); WillReturn.Add("2"); WillReturn.Add("1 0 3"); WillReturn.Add("2 2 4"); //2 3 } else if (InputPattern == "Input2") { WillReturn.Add("5"); WillReturn.Add("3"); WillReturn.Add("1 0 4"); WillReturn.Add("2 1 3"); WillReturn.Add("0 4 4"); //3 3 } else if (InputPattern == "Input3") { WillReturn.Add("6"); WillReturn.Add("6"); WillReturn.Add("1 1 2"); WillReturn.Add("2 4 5"); WillReturn.Add("0 2 4"); WillReturn.Add("2 0 3"); WillReturn.Add("1 1 5"); WillReturn.Add("0 0 5"); //10 1 } else { string wkStr; while ((wkStr = Console.ReadLine()) != null) WillReturn.Add(wkStr); } return WillReturn; } static void Main() { List<string> InputList = GetInputList(); int N = int.Parse(InputList[0]); var InsLazySegmentTreeVal = new LazySegmentTree(N); var InsLazySegmentTreeZeroCnt = new LazySegmentTree(N); InsLazySegmentTreeZeroCnt.RangeUpdate(0, N, 1, 0); int[] wkArr = { }; Action<string> SplitAct = pStr => wkArr = pStr.Split(' ').Select(pX => int.Parse(pX)).ToArray(); long ScoreA = 0; long ScoreB = 0; foreach (string EachStr in InputList.Skip(2)) { SplitAct(EachStr); int X = wkArr[0]; int L = wkArr[1]; int R = wkArr[2]; if (X == 0) { long RangeSum = InsLazySegmentTreeVal.Query(L, R, 0); long RangeNonZeroCnt = InsLazySegmentTreeZeroCnt.Query(L, R, 0); long RangeCnt = R - L + 1; RangeCnt -= RangeNonZeroCnt; long RangePlusCnt, RangeMinusCnt; SolveRenritu(RangeCnt, RangeSum, out RangePlusCnt, out RangeMinusCnt); if (RangePlusCnt > RangeMinusCnt) ScoreA += RangePlusCnt; if (RangeMinusCnt > RangePlusCnt) ScoreB += RangeMinusCnt; } if (X == 1) { InsLazySegmentTreeVal.RangeUpdate(L, R, 1, 0); InsLazySegmentTreeZeroCnt.RangeUpdate(L, R, 0, 0); } if (X == 2) { InsLazySegmentTreeVal.RangeUpdate(L, R, -1, 0); InsLazySegmentTreeZeroCnt.RangeUpdate(L, R, 0, 0); } } long AllSum = InsLazySegmentTreeVal.Query(0, N, 0); long AllNonZeroCnt = InsLazySegmentTreeZeroCnt.Query(0, N, 0); long AllRangeCnt = N + 1; AllRangeCnt -= AllNonZeroCnt; long AllPlusCnt, AllMinusCnt; SolveRenritu(AllRangeCnt, AllSum, out AllPlusCnt, out AllMinusCnt); ScoreA += AllPlusCnt; ScoreB += AllMinusCnt; Console.WriteLine("{0} {1}", ScoreA, ScoreB); } // 下記の連立方程式を加減法で解く // A+B = Cnt // A-B = Sum static void SolveRenritu(long pCnt, long pSum, out long pA, out long pB) { pA = (pCnt + pSum) / 2; pB = pCnt - pA; } } // LazySegmentTreeクラス internal class LazySegmentTree { private int[] mTreeNodeArr; private int UB; // 木のノードの配列のUB private int mLeafCnt; // 葉ノードの数 private int?[] mLazyArr; // 遅延配列 // ノードの添字を引数とし、範囲の開始添字と終了添字を持つ配列 private struct RangeInfoDef { internal int StaInd; internal int EndInd; } private RangeInfoDef[] mRangeInfo; // コンストラクタ internal LazySegmentTree(int pLeafCnt) { // 簡単のため、葉ノード数を2のべき乗に int ArrLength = 0; for (int I = 1; I < int.MaxValue; I *= 2) { ArrLength += I; mLeafCnt = I; if (pLeafCnt < mLeafCnt) break; } // すべての値を0に UB = ArrLength - 1; mTreeNodeArr = new int[UB + 1]; for (int I = 0; I <= UB; I++) { mTreeNodeArr[I] = 0; } // 遅延配列を初期化 mLazyArr = new int?[UB + 1]; // ノードの添字を引数とし、範囲の開始添字と終了添字を持つ配列の作成 mRangeInfo = new RangeInfoDef[UB + 1]; for (int I = 0; I <= UB; I++) { if (I == 0) { RangeInfoDef WillSet1; WillSet1.StaInd = 0; WillSet1.EndInd = mLeafCnt - 1; mRangeInfo[I] = WillSet1; continue; } int ParentNode = DeriveParentNode(I); RangeInfoDef ParentRangeInfo = mRangeInfo[ParentNode]; RangeInfoDef WillSet2; int Mid = (ParentRangeInfo.StaInd + ParentRangeInfo.EndInd) / 2; if (I % 2 == 1) { // 奇数ノードの場合 WillSet2.StaInd = ParentRangeInfo.StaInd; WillSet2.EndInd = Mid; } else { // 偶数ノードの場合 WillSet2.StaInd = Mid + 1; WillSet2.EndInd = ParentRangeInfo.EndInd; } mRangeInfo[I] = WillSet2; } } // 親ノードの添字を取得 private int DeriveParentNode(int pTarget) { return (pTarget - 1) / 2; } // 子ノードの添字(小さいほう)を取得 private int DeriveChildNode(int pTarget) { return pTarget * 2 + 1; } // カレントノードを引数として、遅延評価を行う void LazyEval(int pCurrNode) { // 遅延配列が空なら何もしない if (mLazyArr[pCurrNode].HasValue == false) return; int CurrNodeStaInd = mRangeInfo[pCurrNode].StaInd; int CurrNodeEndInd = mRangeInfo[pCurrNode].EndInd; // 遅延配列の値を反映する //mTreeNodeArr[pCurrNode] += mLazyArr[pCurrNode].Value; mTreeNodeArr[pCurrNode] = mLazyArr[pCurrNode].Value * (CurrNodeEndInd - CurrNodeStaInd + 1); int ChildNode1 = DeriveChildNode(pCurrNode); int ChildNode2 = ChildNode1 + 1; //if (ChildNode1 <= UB) mLazyArr[ChildNode1] += mLazyArr[pCurrNode] / 2; //if (ChildNode2 <= UB) mLazyArr[ChildNode2] += mLazyArr[pCurrNode] / 2; if (ChildNode1 <= UB) mLazyArr[ChildNode1] = mLazyArr[pCurrNode]; if (ChildNode2 <= UB) mLazyArr[ChildNode2] = mLazyArr[pCurrNode]; // 伝播が終わったので、自ノードの遅延配列を空にする mLazyArr[pCurrNode] = null; } // 開始添字と終了添字とカレントノードを引数として、区間更新を行う internal void RangeUpdate(int pSearchStaInd, int pSearchEndInd, int pUpdateVal, int pCurrNode) { // カレントノードの遅延評価を行う LazyEval(pCurrNode); int CurrNodeStaInd = mRangeInfo[pCurrNode].StaInd; int CurrNodeEndInd = mRangeInfo[pCurrNode].EndInd; // OverLapしてなければ、何もしない if (CurrNodeEndInd < pSearchStaInd || pSearchEndInd < CurrNodeStaInd) { return; } // 完全に含んでいれば、遅延配列に値を入れた後に評価 if (pSearchStaInd <= CurrNodeStaInd && CurrNodeEndInd <= pSearchEndInd) { //mLazyArr[pCurrNode] += pUpdateVal * (CurrNodeEndInd - CurrNodeStaInd + 1); mLazyArr[pCurrNode] = pUpdateVal; LazyEval(pCurrNode); return; } // そうでなければ、2つの区間に再帰呼出し int ChildNode1 = DeriveChildNode(pCurrNode); int ChildNode2 = ChildNode1 + 1; RangeUpdate(pSearchStaInd, pSearchEndInd, pUpdateVal, ChildNode1); RangeUpdate(pSearchStaInd, pSearchEndInd, pUpdateVal, ChildNode2); mTreeNodeArr[pCurrNode] = mTreeNodeArr[ChildNode1] + mTreeNodeArr[ChildNode2]; } // 開始添字と終了添字とカレントノードを引数として、最小値を返す internal int Query(int pSearchStaInd, int pSearchEndInd, int pCurrNode) { // 該当ノードを遅延評価する LazyEval(pCurrNode); int CurrNodeStaInd = mRangeInfo[pCurrNode].StaInd; int CurrNodeEndInd = mRangeInfo[pCurrNode].EndInd; // OverLapしてなければ、0 if (CurrNodeEndInd < pSearchStaInd || pSearchEndInd < CurrNodeStaInd) return 0; // 完全に含んでいれば、このノードの値 if (pSearchStaInd <= CurrNodeStaInd && CurrNodeEndInd <= pSearchEndInd) return mTreeNodeArr[pCurrNode]; // そうでなければ、2つの子のSum int ChildNode1 = DeriveChildNode(pCurrNode); int ChildNode2 = ChildNode1 + 1; int ChildVal1 = Query(pSearchStaInd, pSearchEndInd, ChildNode1); int ChildVal2 = Query(pSearchStaInd, pSearchEndInd, ChildNode2); return ChildVal1 + ChildVal2; } internal void DebugPrint() { for (int I = 0; I <= UB; I++) { if (mLazyArr[I].HasValue) { Console.WriteLine("mTreeNodeArr[{0}] = {1} , mLazyArr[{0}] = {2}", I, mTreeNodeArr[I], mLazyArr[I]); } else { Console.WriteLine("mTreeNodeArr[{0}] = {1}", I, mTreeNodeArr[I]); } } } }