/*
Problem: Minimum Cost to Satisfy Daily Demand Using Buy and Repair Options
Company Tag: Walmart
Problem Statement:
A factory works for N days.
On day i, it needs exactly R[i] widgets.
A widget can be obtained in three ways:
1. Buy a new widget
Cost = P
2. Send a used widget for quick repair
It returns after M days
Cost = F
3. Send a used widget for slow repair
It returns after Q days
Cost = S
We need to satisfy demand every day with minimum total cost.
Constraints:
1 <= N <= 2000
1 <= R[i] <= 10^7
1 <= P, M, F, Q, S <= 10^4
Q > M
Simple Brute Force:
For every day, try all choices:
buy new,
quick repair,
slow repair,
store,
discard.
Why Brute Force Fails:
The number of possible schedules becomes huge.
Better Idea:
Model this as a minimum cost flow problem.
Each widget is one unit of flow.
For each day:
- clean widgets are used to satisfy demand
- after use, they become dirty
- dirty widgets can be repaired and reused later
- new widgets can be bought when needed
We force exactly R[i] widgets to be used on day i.
Dry Run:
N = 3
R = [1, 1, 1]
P = 10
M = 1, F = 3
Q = 2, S = 1
Best plan:
Day 1: buy 1 widget cost 10
Day 2: quick repaired one cost 3
Day 3: quick repaired one cost 3
Total cost = 16
Time Complexity:
Around O(flow augmentations * E log V)
Space Complexity:
O(N)
*/
#include <bits/stdc++.h>
using namespace std;
struct Edge {
int to;
int reverseIndex;
long long capacity;
long long cost;
};
class MinCostFlow {
public:
int nodeCount;
vector<vector<Edge>> graph;
MinCostFlow(int n) {
nodeCount = n;
graph.assign(n, {});
}
void addEdge(int from, int to, long long capacity, long long cost) {
Edge forward = {to, (int)graph[to].size(), capacity, cost};
Edge backward = {from, (int)graph[from].size(), 0, -cost};
graph[from].push_back(forward);
graph[to].push_back(backward);
}
pair<long long, long long> minCostMaxFlow(int source, int sink, long long requiredFlow) {
const long long INF = (1LL << 60);
long long totalFlow = 0;
long long totalCost = 0;
vector<long long> potential(nodeCount, 0);
while (totalFlow < requiredFlow) {
vector<long long> distance(nodeCount, INF);
vector<int> parentNode(nodeCount, -1);
vector<int> parentEdge(nodeCount, -1);
priority_queue<
pair<long long, int>,
vector<pair<long long, int>>,
greater<pair<long long, int>>
> pq;
distance[source] = 0;
pq.push({0, source});
while (!pq.empty()) {
pair<long long, int> current = pq.top();
pq.pop();
long long currentDistance = current.first;
int node = current.second;
if (currentDistance != distance[node]) {
continue;
}
for (int i = 0; i < (int)graph[node].size(); i++) {
Edge &edge = graph[node][i];
if (edge.capacity <= 0) {
continue;
}
long long newDistance =
distance[node] + edge.cost + potential[node] - potential[edge.to];
if (newDistance < distance[edge.to]) {
distance[edge.to] = newDistance;
parentNode[edge.to] = node;
parentEdge[edge.to] = i;
pq.push({newDistance, edge.to});
}
}
}
if (distance[sink] == INF) {
break;
}
for (int i = 0; i < nodeCount; i++) {
if (distance[i] < INF) {
potential[i] += distance[i];
}
}
long long addFlow = requiredFlow - totalFlow;
int node = sink;
while (node != source) {
int previous = parentNode[node];
int edgeIndex = parentEdge[node];
addFlow = min(addFlow, graph[previous][edgeIndex].capacity);
node = previous;
}
node = sink;
while (node != source) {
int previous = parentNode[node];
int edgeIndex = parentEdge[node];
Edge &edge = graph[previous][edgeIndex];
edge.capacity -= addFlow;
graph[node][edge.reverseIndex].capacity += addFlow;
totalCost += addFlow * edge.cost;
node = previous;
}
totalFlow += addFlow;
}
return {totalFlow, totalCost};
}
};
int main() {
int N;
cin >> N;
vector<long long> R(N);
for (int i = 0; i < N; i++) {
cin >> R[i];
}
long long P, M, F, Q, S;
cin >> P >> M >> F >> Q >> S;
const long long INF_CAP = (1LL << 55);
int cleanStart = 0;
int dirtyStart = N;
int source = 2 * N;
int sink = 2 * N + 1;
int superSource = 2 * N + 2;
int superSink = 2 * N + 3;
int totalNodes = 2 * N + 4;
MinCostFlow flow(totalNodes);
vector<long long> balance(totalNodes, 0);
auto addLowerBoundEdge = [&](int from, int to, long long lower, long long upper, long long cost) {
/*
Lower bound trick:
We force at least 'lower' flow through this edge
by adjusting the balance of both endpoints.
*/
flow.addEdge(from, to, upper - lower, cost);
balance[from] -= lower;
balance[to] += lower;
};
for (int day = 0; day < N; day++) {
int cleanNode = cleanStart + day;
int dirtyNode = dirtyStart + day;
// Buy a new clean widget for this day.
addLowerBoundEdge(source, cleanNode, 0, INF_CAP, P);
// Exactly R[day] widgets must be used on this day.
addLowerBoundEdge(cleanNode, dirtyNode, R[day], R[day], 0);
// After use, we may discard the dirty widget.
addLowerBoundEdge(dirtyNode, sink, 0, INF_CAP, 0);
// Clean widgets can be stored for the next day.
if (day + 1 < N) {
addLowerBoundEdge(cleanNode, cleanStart + day + 1, 0, INF_CAP, 0);
}
// Dirty widgets can also be kept and repaired later.
if (day + 1 < N) {
addLowerBoundEdge(dirtyNode, dirtyStart + day + 1, 0, INF_CAP, 0);
}
// Quick repair makes a dirty widget clean after M days.
if (day + M < N) {
addLowerBoundEdge(dirtyNode, cleanStart + day + M, 0, INF_CAP, F);
}
// Slow repair makes a dirty widget clean after Q days.
if (day + Q < N) {
addLowerBoundEdge(dirtyNode, cleanStart + day + Q, 0, INF_CAP, S);
}
}
/*
This edge converts the network into a circulation problem.
It lets the flow balance out properly.
*/
addLowerBoundEdge(sink, source, 0, INF_CAP, 0);
long long requiredFlow = 0;
for (int i = 0; i < totalNodes; i++) {
if (balance[i] > 0) {
flow.addEdge(superSource, i, balance[i], 0);
requiredFlow += balance[i];
} else if (balance[i] < 0) {
flow.addEdge(i, superSink, -balance[i], 0);
}
}
pair<long long, long long> result =
flow.minCostMaxFlow(superSource, superSink, requiredFlow);
cout << result.second << endl;
return 0;
}
/*
    Problem: Minimum Cost to Satisfy Daily Demand Using Buy and Repair Options
    Company Tag: Walmart

    Problem Statement:
        A factory works for N days.

        On day i, it needs exactly R[i] widgets.

        A widget can be obtained in three ways:

            1. Buy a new widget
               Cost = P

            2. Send a used widget for quick repair
               It returns after M days
               Cost = F

            3. Send a used widget for slow repair
               It returns after Q days
               Cost = S

        We need to satisfy demand every day with minimum total cost.

    Constraints:
        1 <= N <= 2000
        1 <= R[i] <= 10^7
        1 <= P, M, F, Q, S <= 10^4
        Q > M

    Simple Brute Force:
        For every day, try all choices:
            buy new,
            quick repair,
            slow repair,
            store,
            discard.

    Why Brute Force Fails:
        The number of possible schedules becomes huge.

    Better Idea:
        Model this as a minimum cost flow problem.

        Each widget is one unit of flow.

        For each day:
            - clean widgets are used to satisfy demand
            - after use, they become dirty
            - dirty widgets can be repaired and reused later
            - new widgets can be bought when needed

        We force exactly R[i] widgets to be used on day i.

    Dry Run:
        N = 3
        R = [1, 1, 1]
        P = 10
        M = 1, F = 3
        Q = 2, S = 1

        Best plan:
            Day 1: buy 1 widget        cost 10
            Day 2: quick repaired one  cost 3
            Day 3: quick repaired one  cost 3

        Total cost = 16

    Time Complexity:
        Around O(flow augmentations * E log V)

    Space Complexity:
        O(N)
*/

#include <bits/stdc++.h>
using namespace std;

struct Edge {
    int to;
    int reverseIndex;
    long long capacity;
    long long cost;
};

class MinCostFlow {
public:
    int nodeCount;
    vector<vector<Edge>> graph;

    MinCostFlow(int n) {
        nodeCount = n;
        graph.assign(n, {});
    }

    void addEdge(int from, int to, long long capacity, long long cost) {
        Edge forward = {to, (int)graph[to].size(), capacity, cost};
        Edge backward = {from, (int)graph[from].size(), 0, -cost};

        graph[from].push_back(forward);
        graph[to].push_back(backward);
    }

    pair<long long, long long> minCostMaxFlow(int source, int sink, long long requiredFlow) {
        const long long INF = (1LL << 60);

        long long totalFlow = 0;
        long long totalCost = 0;

        vector<long long> potential(nodeCount, 0);

        while (totalFlow < requiredFlow) {
            vector<long long> distance(nodeCount, INF);
            vector<int> parentNode(nodeCount, -1);
            vector<int> parentEdge(nodeCount, -1);

            priority_queue<
                pair<long long, int>,
                vector<pair<long long, int>>,
                greater<pair<long long, int>>
            > pq;

            distance[source] = 0;
            pq.push({0, source});

            while (!pq.empty()) {
                pair<long long, int> current = pq.top();
                pq.pop();

                long long currentDistance = current.first;
                int node = current.second;

                if (currentDistance != distance[node]) {
                    continue;
                }

                for (int i = 0; i < (int)graph[node].size(); i++) {
                    Edge &edge = graph[node][i];

                    if (edge.capacity <= 0) {
                        continue;
                    }

                    long long newDistance =
                        distance[node] + edge.cost + potential[node] - potential[edge.to];

                    if (newDistance < distance[edge.to]) {
                        distance[edge.to] = newDistance;
                        parentNode[edge.to] = node;
                        parentEdge[edge.to] = i;
                        pq.push({newDistance, edge.to});
                    }
                }
            }

            if (distance[sink] == INF) {
                break;
            }

            for (int i = 0; i < nodeCount; i++) {
                if (distance[i] < INF) {
                    potential[i] += distance[i];
                }
            }

            long long addFlow = requiredFlow - totalFlow;
            int node = sink;

            while (node != source) {
                int previous = parentNode[node];
                int edgeIndex = parentEdge[node];

                addFlow = min(addFlow, graph[previous][edgeIndex].capacity);
                node = previous;
            }

            node = sink;

            while (node != source) {
                int previous = parentNode[node];
                int edgeIndex = parentEdge[node];

                Edge &edge = graph[previous][edgeIndex];

                edge.capacity -= addFlow;
                graph[node][edge.reverseIndex].capacity += addFlow;

                totalCost += addFlow * edge.cost;

                node = previous;
            }

            totalFlow += addFlow;
        }

        return {totalFlow, totalCost};
    }
};

int main() {
    int N;
    cin >> N;

    vector<long long> R(N);

    for (int i = 0; i < N; i++) {
        cin >> R[i];
    }

    long long P, M, F, Q, S;
    cin >> P >> M >> F >> Q >> S;

    const long long INF_CAP = (1LL << 55);

    int cleanStart = 0;
    int dirtyStart = N;

    int source = 2 * N;
    int sink = 2 * N + 1;

    int superSource = 2 * N + 2;
    int superSink = 2 * N + 3;

    int totalNodes = 2 * N + 4;

    MinCostFlow flow(totalNodes);

    vector<long long> balance(totalNodes, 0);

    auto addLowerBoundEdge = [&](int from, int to, long long lower, long long upper, long long cost) {
        /*
            Lower bound trick:
            We force at least 'lower' flow through this edge
            by adjusting the balance of both endpoints.
        */
        flow.addEdge(from, to, upper - lower, cost);

        balance[from] -= lower;
        balance[to] += lower;
    };

    for (int day = 0; day < N; day++) {
        int cleanNode = cleanStart + day;
        int dirtyNode = dirtyStart + day;

        // Buy a new clean widget for this day.
        addLowerBoundEdge(source, cleanNode, 0, INF_CAP, P);

        // Exactly R[day] widgets must be used on this day.
        addLowerBoundEdge(cleanNode, dirtyNode, R[day], R[day], 0);

        // After use, we may discard the dirty widget.
        addLowerBoundEdge(dirtyNode, sink, 0, INF_CAP, 0);

        // Clean widgets can be stored for the next day.
        if (day + 1 < N) {
            addLowerBoundEdge(cleanNode, cleanStart + day + 1, 0, INF_CAP, 0);
        }

        // Dirty widgets can also be kept and repaired later.
        if (day + 1 < N) {
            addLowerBoundEdge(dirtyNode, dirtyStart + day + 1, 0, INF_CAP, 0);
        }

        // Quick repair makes a dirty widget clean after M days.
        if (day + M < N) {
            addLowerBoundEdge(dirtyNode, cleanStart + day + M, 0, INF_CAP, F);
        }

        // Slow repair makes a dirty widget clean after Q days.
        if (day + Q < N) {
            addLowerBoundEdge(dirtyNode, cleanStart + day + Q, 0, INF_CAP, S);
        }
    }

    /*
        This edge converts the network into a circulation problem.
        It lets the flow balance out properly.
    */
    addLowerBoundEdge(sink, source, 0, INF_CAP, 0);

    long long requiredFlow = 0;

    for (int i = 0; i < totalNodes; i++) {
        if (balance[i] > 0) {
            flow.addEdge(superSource, i, balance[i], 0);
            requiredFlow += balance[i];
        } else if (balance[i] < 0) {
            flow.addEdge(i, superSink, -balance[i], 0);
        }
    }

    pair<long long, long long> result =
        flow.minCostMaxFlow(superSource, superSink, requiredFlow);

    cout << result.second << endl;

    return 0;
}