ferinの競プロ帳

競プロについてのメモ

AOJ1347 Shopping

問題ページ
Shopping | Aizu Online Judge

概要

n個の店が並んだ商店街がありi番目の店は位置iにある。ある店dを訪れてからある店cを訪れなければならないというm個の制約の元で位置0から位置n+1まで歩くときの最小距離を求めろ。

n <= 10000, m <= 500

考えたこと

  • 区間だし終端でソートする
  • 前の区間と交差しないか包含するか交差するの3パターンになりそう f:id:ferin_tech:20180224035225p:plain
  • 交差するか包含する場合は引き返すのをまとめてやった方がよさそう
  • どこまでまとめて引き返すべきかを計算する
  • 出すと落ちる
  • 一旦交差しない区間が来た後、交差する区間が来るパターンを見落としていた
  • 直すと通った

区間グラフ系の問題苦手だ…

#include <bits/stdc++.h>

using namespace std;
using ll = long long;
#define int ll
using VI = vector<int>;
using VVI = vector<VI>;
using PII = pair<int, int>;

#define FOR(i, a, n) for (ll i = (ll)a; i < (ll)n; ++i)
#define REP(i, n) FOR(i, 0, n)
#define ALL(x) x.begin(), x.end()
#define PB push_back

const ll LLINF = (1LL<<60);
const int INF = (1LL<<30);
const int MOD = 1000000007;

template <typename T> T &chmin(T &a, const T &b) { return a = min(a, b); }
template <typename T> T &chmax(T &a, const T &b) { return a = max(a, b); }
template <typename T> bool IN(T a, T b, T x) { return a<=x&&x<b; }
template<typename T> T ceil(T a, T b) { return a/b + !!(a%b); }
template<class S,class T>
ostream &operator <<(ostream& out,const pair<S,T>& a){
  out<<'('<<a.first<<','<<a.second<<')';
  return out;
}
template<class T>
ostream &operator <<(ostream& out,const vector<T>& a){
  out<<'[';
  REP(i, a.size()) {out<<a[i];if(i!=a.size()-1)out<<',';}
  out<<']';
  return out;
}

int dx[] = {0, 1, 0, -1}, dy[] = {1, 0, -1, 0};

signed main(void)
{
  int n, m;
  cin >> n >> m;
  vector<PII> v;
  REP(i, m) {
    int c, d;
    cin >> c >> d;
    if(c < d) v.PB({d, c});
  }
  sort(ALL(v));

  if(v.size() == 0) {
    cout << n+1 << endl;
    return 0;
  }

  int ret = n+1;
  for(int i=0; i<m; ++i) {
    // cout << "i:" << i << endl;
    int l = v[i].second, r = v[i].first;
    int idx = i;
    for(int j=i+1; j<m; ++j) {
      if(v[j].second <= r) {
        r = v[j].first;
        chmin(l, v[j].second);
        idx = j;
      }
      // cout << l << " " << r << " " << idx << endl;
    }
    ret += (r-l)*2;
    i = idx;
  }

  cout << ret << endl;

  return 0;
}