Processing math: 100%
(Analysis by Spencer Compton)

We begin by considering the structure of optimal solutions for a particular query q where you must end at mana pool eq after sq seconds. Let us characterize a journey by an ordered list (p1,t2),(p2,t2),,(pk,tk), where (pi,ti) represents Bessie last visiting pool pi at time ti and collecting all of its available mana. By definition, a valid journey for query q must end at eq (meaning pk=eq) and end after sq seconds (meaning tk=sq). The mana collected in such a proposed journey is then ki=1mpiti. Of course, some journeys we describe are infeasible. More formally, we will consider a journey feasible if and only if t10 and ti+1ti+D[pi][pi+1] for all i{1,,k1}, and where D[a][b] represents the shortest path from pool a to pool b (note that D can be pre-computed with the Floyd-Warshall algorithm in O(N3) time). With this in mind, what is the optimal feasible and valid journey for any fixed p1,,pk? It can be shown with an inductive proof or exchange argument that it is optimal to wait as long as possible. More concretely, this means the optimal times t are:

Note how if t1<0 then no journey with pools p1,,pk is feasible. For the first subtask, it is sufficient to find the optimal solution by exhaustively enumerating over all O(N!) possible choices of p1,,pk and calculate the corresponding t1,,tk for each query in O(N!(N+Q)) total time.

For the second subtask, we can no longer afford to iterate over each query for each possible journey. We look for some structure of similarity that eliminates the need for this iteration. For simplicity of presentation, we will only consider journeys and queries ending at some fixed pool e, as we can handle other journey/queries separately. Based on our closed-form for t, the journey will be feasible and valid for some query if and only if k1i=1D[pi][pi+1]sq. Moreover, the total mana gained will be equal to ki=1mpiti=ki=1mpi(sqk1j=iD[pj][pj+1])=(ki=1mpi)sq(ki=1(mpik1j=iD[pj][pj+1])).

We see how this expression takes the form of a function linear in sq. In particular, we can determine that a journey J is feasible if and only if sqTJ, and then journey J gains mana fJ(sq)slopeJsq+addJ where:

This immediately implies that the answer for a query q is the maximum over a collection of linear functions maxJ s.t. TJsqfJ(sq). If not for the condition that TJsq, queries regarding the maximum of linear functions at a particular point can be answered efficiently with a data structure using Convex Hull Trick (CHT) or LineContainer. To handle the condition that TJsq, we can sort queries by sq and journeys by TJ, and incrementally add lines to our data structure such that when we answer query q, the journeys with TJsq are exactly the set of journeys in our data structure. Accordingly, we can enumerate over all O(N!) possible choices of J=p1,,pk, calculate the corresponding TJ and fJ, and then answer queries using our data structure in total time O(N!Nlog(N)+Qlog(Q)) for the second subtask.

For the remaining subtasks, it is too slow to enumerate over all O(N!) possibilities for J=p1,,pk. We make two structural observations:

First, the condition that TJsq is surprisingly unimportant, because maxJ s.t. TJsqfJ(sq)=maxJfJ(sq). We will show this with a proof by contradiction. Suppose, for sake of contradiction, that some job J with TJ>sq obtains the optimal amount of mana. Recall how fJ(sq)=ki=1mpi(sqk1j=iD[pj][pj+1]). If TJ>sq, then the summand corresponding to p1 must be negative, and thus the journey corresponding to p2,,pk would be strictly better, and we have proven a contradiction. Accordingly, we can ignore the TJ constraint and simply optimize for maxJfJ(sq).

Second, all journeys with the same unordered set of pools p1,,pk have the same slope (see the definition of slopeJ above). More concretely, let P{1,,N} denote some subset of pools to visit, and let PJ denote the unordered set of pools that a journey J visits. All journeys J with PJ=P satisfy slopeJ=pPmp.

Our first observation shows how the only aspects of a line that matter are slopeJ and addJ (not TJ). Our second observation notes that slopeJ is completely determined by the unordered set of pools J visits. Combining our two observations, we conclude that we can optimize over subsets of pools to visit, and for each fixed subset P the only meaningful degree of freedom is addJ, so we can just consider the line with the best addJ among those that visit P. More formally: maxJ s.t. TJsqfJ(sq)=maxJfJ(sq)=maxJslopeJsq+addJ=maxP(pPmp)sq+(maxJs.t. PJ=PaddJ).

This closed-form is also a function linear in sq. We can then define our optimization as over linear functions for each subset as gP(sq)slopePsq+addP. Where:

Our hope is to optimize over maxPgP(sq). All that remains is being able to compute each addP. We compute all addP by a dynamic programming solution where the state dp[now][rem] represents the set rem of remaining nodes to visit (going backwards in the ordering), and the current location now. Recall how addJ=ki=1(mpik1j=iD[pj][pj+1]). Accordingly, if we decide to transition to the next location nextrem, then the summands in addJ for each prem will increase by mpD[next][now]. This formulation is sufficient to design a dynamic program with O(2NN) states, O(N) transitions, and O(2NN2) total running time.

Combining all steps, for each query we output maxPgP(sq), where we compute the coefficients for the O(2NN) lines gP by dynamic programming in O(2NN2) time. We then use CHT to evaluate maxPgP(sq) for each query. There is some flexibility in implementation. For example, one could sort the lines and queries and then use a vector/deque for a total runtime of O(2NN2+Qlog(Q)) (as is done in the code below). Or, one could prepare the CHT data structure over the lines beforehand and then evaluate the queries without sorting them for a total runtime of O(2NN2+QN).

It is possible to get the third subtask by utilizing the above dynamic program but enumerating over all queries (instead of using CHT) for a O(2N(N2+Q)) time algorithm. The fourth and fifth subtask of N=16 and N=17 exist to potentially give partial credit for slightly suboptimal solutions, or those needing constant factor optimization.

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
#define f first
#define s second
vector<ll> rates;
ll dist[19][19];
ll inf = 1e9+1;
ll inf2 = 2e18;
int N, Q;
ll maskDeriv[1<<18];
ll dp[18][1<<18];
vector<pair<ll, ll> > hull[18];
vector<ll> hull_pt;
 
ll go(int now, int rem){
    // computing optimal add_J that ends at now, and visits everything remaining in rem
    if(rem == 0){
        return 0LL;
    }
    if(dp[now][rem] <= 0LL){
        return dp[now][rem];
    }
    ll ret = -inf2;
    for(int i = 0; i<N; i++){
        if((rem&(1<<i))!=0){
            ret = max(ret,go(i,rem-(1<<i)) - maskDeriv[rem] * dist[i][now]);
        }
    }
    dp[now][rem] = ret;
    return ret;
}
 
ll overcome(pair<ll, ll> a, pair<ll, ll> b){
    // when does line "a" with slope a.f and add a.s, take over line "b" with slope b.f and add b.s
    assert((a.f > b.f) && (a.s < b.s));
    ll slopeDif = a.f - b.f;
    ll addDif = b.s - a.s;
    ll addOne = ((addDif%slopeDif)!=0)?1:0;
    return addDif/slopeDif + addOne;
}
 
void ins(int loc, ll slope, ll add){
    // insert a line to the hull
    // corresponds to a journey ending at loc
    auto cur = make_pair(slope,add);
    if(hull[loc].size()>0 && (hull[loc].back().f == slope) && hull[loc].back().s >= add){
        return;
    }
    while(true){
        int sz = (int)(hull[loc].size());
        if(sz==0){
            break;
        }
        auto bef = hull[loc][sz-1];
        // if the previous line has worse slope and add than the new line, delete it
        if(bef.s <= add){
            hull[loc].pop_back();
            continue;
        }
        if(sz == 1){
            break;
        }
        auto grandBef = hull[loc][sz-2];
        // if the new line overcomes the second-newest before the second-newest overcomes the third-newest
        // then the second-newest is never on the hull, so we delete it
        if(overcome(bef,grandBef) >= overcome(cur,bef)){
            hull[loc].pop_back();
            continue;
        }
        break;
    }
    hull[loc].push_back(cur);
}
 
ll query(int loc, ll t){
    // query the hull for ending location loc at time t
    int sz = (int)(hull[loc].size());
    while(hull_pt[loc]+1<sz){
        auto it = hull[loc].begin();
        auto val1 = hull[loc][hull_pt[loc]];
        it++;
        auto val2 = hull[loc][hull_pt[loc]+1];
        if(overcome(val2,val1) > t){
            break;
        }
        hull_pt[loc]++;
    }
    auto val = (hull[loc][hull_pt[loc]]);
    return t * val.f + val.s;
}
 
int main(){
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    int m;
    cin >> N >> m;
    hull_pt.resize(N);
    for(int i = 0; i<N; i++){
        ll x;
        cin >> x;
        rates.push_back(x);
    }
    for(int i = 0; i<=N; i++){
        for(int j = 0; j<=N; j++){
            dist[i][j] = inf;
        }
    }
    for(int i = 0; i<m; i++){
        int a, b;
        ll t;
        cin >> a >> b >> t;
        a--;
        b--;
        dist[a][b] = t;
    }
    // compute shortest paths
    for(int k = 0; k<N; k++){
        for(int i = 0; i<N; i++){
            for(int j = 0; j<N; j++){
                dist[i][j] = min(dist[i][j],dist[i][k] + dist[k][j]);
            }
        }
    }
    // compute the total rate of all masks
    maskDeriv[0] = 0L;
    for(int mask = 1; mask<(1<<N); mask++){
        for(int i = 0; i<N; i++){
            if((mask&(1<<i))!=0){
                maskDeriv[mask] = maskDeriv[mask - (1<<i)] + rates[i];
                break;
            }
        }
    }
    for(int i = 0; i<(1<<N); i++){
        for(int j = 0; j<N; j++){
            dp[j][i] = 1LL;
        }
    }
    int all = (1<<N) - 1;
    vector<pair<ll, pair<ll, int> > > lines;
    for(int mask = 0; mask<(1<<N); mask++){
        for(int j = 0; j<N; j++){
            if((mask&(1<<j))==0){
                // considers the line for a journey ending at location j
                // maskDeriv[mask|(1<<j)] is slope
                // go(j,mask) is add
                lines.push_back(make_pair(maskDeriv[mask|(1<<j)],make_pair(go(j,mask),j)));
            }
            
        }
    }
    sort(lines.begin(),lines.end());
    // insert lines into hull in order of slope
    for(auto x: lines){
        ins(x.s.s, x.f,x.s.f);
    }
    cin >> Q;
    vector<ll> ans(Q);
    vector<pair<ll, pair<int, int> > > queries;
    for(int i = 0; i<Q; i++){
        ll s;
        int e;
        cin >> s >> e;
        e--;
        // consider query i for ending at e after s seconds
        queries.push_back(make_pair(s,make_pair(e,i)));
    }
    // consider queries in order of time
    sort(queries.begin(),queries.end());
    for(int i = 0; i<Q; i++){
        ll t = queries[i].f;
        int loc = queries[i].s.f;
        int id = queries[i].s.s;
        // query the relevant hull at the relevant time
        ans[id] = query(loc,t);
    }
    for(int i = 0; i<Q; i++){
        cout << ans[i] << "\n";
    }
}