結果

問題 No.703 ゴミ拾い Easy
ユーザー zaki_johozaki_joho
提出日時 2019-10-24 00:17:08
言語 C++14
(gcc 12.3.0 + boost 1.83.0)
結果
WA  
実行時間 -
コード長 3,787 bytes
コンパイル時間 1,792 ms
コンパイル使用メモリ 174,396 KB
実行使用メモリ 12,852 KB
最終ジャッジ日時 2023-09-21 20:35:50
合計ジャッジ時間 9,752 ms
ジャッジサーバーID
(参考情報)
judge13 / judge14
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 WA -
testcase_01 WA -
testcase_02 WA -
testcase_03 WA -
testcase_04 WA -
testcase_05 WA -
testcase_06 WA -
testcase_07 WA -
testcase_08 WA -
testcase_09 WA -
testcase_10 WA -
testcase_11 WA -
testcase_12 WA -
testcase_13 WA -
testcase_14 WA -
testcase_15 WA -
testcase_16 WA -
testcase_17 WA -
testcase_18 WA -
testcase_19 WA -
testcase_20 WA -
testcase_21 WA -
testcase_22 WA -
testcase_23 WA -
testcase_24 WA -
testcase_25 WA -
testcase_26 WA -
testcase_27 WA -
testcase_28 WA -
testcase_29 WA -
testcase_30 WA -
testcase_31 WA -
testcase_32 WA -
testcase_33 WA -
testcase_34 WA -
testcase_35 WA -
testcase_36 WA -
testcase_37 WA -
testcase_38 WA -
testcase_39 WA -
testcase_40 WA -
testcase_41 WA -
testcase_42 WA -
testcase_43 WA -
testcase_44 WA -
testcase_45 WA -
testcase_46 WA -
testcase_47 WA -
testcase_48 WA -
testcase_49 WA -
権限があれば一括ダウンロードができます

ソースコード

diff #

#include "bits/stdc++.h"

using namespace std;

// https://github.com/satanic0258/Cpp_snippet/blob/master/src/technique/ConvexHullTrick.cpp
using CHT_TYPE = long long;
class ConvexHullTrickDynamic
{
private:
	// 直線 **************************************************************
	struct Line
	{
		CHT_TYPE a, b;								  // y = ax + b
		mutable std::function<const Line *()> getSuc; // 次の直線へのポインタ (ソートで用いる)

		bool operator<(const Line &rhs) const
		{
			// 取得クエリでは次の直線との差分でソート
			if (rhs.b == IS_QUERY)
			{
				const Line *suc = getSuc();
				if (suc == nullptr)
					return false;
				const CHT_TYPE &x = rhs.a;
				return (suc->a - a) * x + suc->b - b > 0;
			}
			if (b == IS_QUERY)
			{
				const Line *suc = rhs.getSuc();
				if (suc == nullptr)
					return true;
				const CHT_TYPE &x = a;
				return (suc->a - rhs.a) * x + suc->b - rhs.b < 0;
			}

			// 通常の直線どうしは傾きソート
			return a < rhs.a;
		}
	};

	// 直線集合 **********************************************************
	class LinesSet : public std::multiset<Line>
	{
	private:
		// true -> 最小値クエリ, false -> 最大値クエリ
		bool flagMin;

	public:
		// コンストラクタ ( 第一引数falseで最大値クエリ,デフォルトで最小値クエリ )
		LinesSet(bool flagMin = false) : flagMin(flagMin){};

		// 直線lが不必要であるかどうか
		inline bool isBad(iterator l)
		{
			const auto &&nel = std::next(l);
			if (l == begin())
			{ // lが傾き最小のとき
				if (nel == end())
					return false; // lしかないなら必要
				return l->a == nel->a && l->b <= nel->b;
			}
			else
			{
				const auto &&prl = std::prev(l);
				if (nel == end())
					return l->a == prl->a && l->b <= prl->b;
				return (prl->b - l->b) * (nel->a - l->a) >= (nel->b - l->b) * (prl->a - l->a);
			}
		}

		// 直線y=ax+bを追加する
		inline void add(CHT_TYPE a, CHT_TYPE b)
		{
			if (flagMin)
				a = -a, b = -b;
			auto &&it = insert({a, b});
			it->getSuc = [=] { return (std::next(it) == end() ? nullptr : &*std::next(it)); };
			if (isBad(it))
			{
				erase(it);
				return;
			}
			while (std::next(it) != end() && isBad(std::next(it)))
				erase(std::next(it));
			while (it != begin() && isBad(std::prev(it)))
				erase(std::prev(it));
		}

		// 直線群の中でxの時に最小(最大)となる値を返す
		inline CHT_TYPE get(CHT_TYPE x)
		{
			auto &&l = *lower_bound(Line{x, IS_QUERY});
			if (flagMin)
				return -l.a * x - l.b;
			else
				return l.a * x + l.b;
		}
	};

	static const CHT_TYPE IS_QUERY = std::numeric_limits<CHT_TYPE>::lowest();
	LinesSet linesSet;

public:
	// コンストラクタ ( 第一引数falseで最大値クエリ,デフォルトで最小値クエリ )
	ConvexHullTrickDynamic(bool flagMin = false) : linesSet(flagMin) {
		linesSet.add(0, -1e15);
	}
	// 直線y=ax+bを追加する
	inline void add(CHT_TYPE a, CHT_TYPE b) { linesSet.add(a, b); }
	// あるxのときの直線集合での最小値を求める
	inline CHT_TYPE get(CHT_TYPE x) { return linesSet.get(x); }
};

void solve_yukicoder_703()
{
    using ll = long long;
    const ll inf = 1e15;
    int n;
    cin >> n;
    vector<ll> a(n), x(n), y(n);
    for (int i = 0; i < n; i++)
    {
        cin >> a[i];
    }
    for (int i = 0; i < n; i++)
    {
        cin >> x[i];
    }
    for (int i = 0; i < n; i++)
    {
        cin >> y[i];
    }
    ConvexHullTrickDynamic cht(100010);
    vector<ll> dp(n + 1, inf);
    dp[0] = 0;
    for (int i = 0; i < n; i++)
    {
        dp[i + 1] = (a[i] - x[i]) * (a[i] - x[i]) + y[i] * y[i];
        dp[i + 1] = min(dp[i + 1], cht.get(a[i]) + a[i] * a[i]);
    }
    cout << dp[n] << endl;
}

int main()
{
    solve_yukicoder_703();
}
0