Browse Source

Changes:

* included small consensus example
* made backward-transition generation more beautiful and versatile
* included Dijkstra search for most probable paths
* included first rough scheduler-guessing (there's room for improvement though)

Former-commit-id: db795fa1bf
tempestpy_adaptions
dehnert 12 years ago
parent
commit
15542d46da
  1. 8
      examples/mdp/consensus/coin.pctl
  2. 49
      src/modelchecker/prctl/SparseMdpPrctlModelChecker.h
  3. 24
      src/models/AbstractDeterministicModel.h
  4. 49
      src/models/AbstractModel.h
  5. 24
      src/models/AbstractNondeterministicModel.h
  6. 9
      src/solver/GmmxxLinearEquationSolver.h
  7. 72
      src/storage/SparseMatrix.h
  8. 2
      src/utility/ConstTemplates.h
  9. 70
      src/utility/graph.h

8
examples/mdp/consensus/coin.pctl

@ -11,10 +11,10 @@ Pmin=? [ F finished & all_coins_equal_1 ]
Pmax=? [ F finished & !agree ]
// Min/max probability of finishing within k steps
Pmin=? [ F<=50 finished ]
Pmax=? [ F<=50 finished ]
// Pmin=? [ F<=50 finished ]
// Pmax=? [ F<=50 finished ]
// Min/max expected steps to finish
Rmin=? [ F finished ]
Rmax=? [ F finished ]
// Rmin=? [ F finished ]
// Rmax=? [ F finished ]

49
src/modelchecker/prctl/SparseMdpPrctlModelChecker.h

@ -279,10 +279,6 @@ namespace storm {
storm::storage::BitVector statesWithProbability0 = std::move(statesWithProbability01.first);
storm::storage::BitVector statesWithProbability1 = std::move(statesWithProbability01.second);
// Delete sub-results that are obsolete now.
delete leftStates;
delete rightStates;
storm::storage::BitVector maybeStates = ~(statesWithProbability0 | statesWithProbability1);
LOG4CPLUS_INFO(logger, "Found " << statesWithProbability0.getNumberOfSetBits() << " 'no' states.");
LOG4CPLUS_INFO(logger, "Found " << statesWithProbability1.getNumberOfSetBits() << " 'yes' states.");
@ -317,7 +313,7 @@ namespace storm {
std::vector<Type> b = this->getModel().getTransitionMatrix().getConstrainedRowSumVector(maybeStates, this->getModel().getNondeterministicChoiceIndices(), statesWithProbability1, submatrix.getRowCount());
// Create vector for results for maybe states.
std::vector<Type> x = this->getInitialValueIterationValues(submatrix, subNondeterministicChoiceIndices, b);
std::vector<Type> x = this->getInitialValueIterationValues(minimize, submatrix, subNondeterministicChoiceIndices, b, statesWithProbability1, maybeStates);
// Solve the corresponding system of equations.
if (linearEquationSolver != nullptr) {
@ -339,6 +335,9 @@ namespace storm {
this->computeTakenChoices(this->minimumOperatorStack.top(), false, *result, *scheduler, this->getModel().getNondeterministicChoiceIndices());
}
// Delete sub-results that are obsolete now.
delete leftStates;
delete rightStates;
return result;
}
@ -523,7 +522,7 @@ namespace storm {
}
// Create vector for results for maybe states.
std::vector<Type> x = this->getInitialValueIterationValues(submatrix, subNondeterministicChoiceIndices, b);
std::vector<Type> x = this->getInitialValueIterationValues(minimize, submatrix, subNondeterministicChoiceIndices, b, *targetStates, maybeStates);
// Solve the corresponding system of equations.
if (linearEquationSolver != nullptr) {
@ -645,11 +644,20 @@ namespace storm {
*
* @param submatrix The matrix that will be used for value iteration later.
*/
std::vector<Type> getInitialValueIterationValues(storm::storage::SparseMatrix<Type> const& submatrix, std::vector<uint_fast64_t> const& subNondeterministicChoiceIndices, std::vector<Type> const& rightHandSide) const {
std::vector<Type> getInitialValueIterationValues(bool minimize, storm::storage::SparseMatrix<Type> const& submatrix,
std::vector<uint_fast64_t> const& subNondeterministicChoiceIndices,
std::vector<Type> const& rightHandSide,
storm::storage::BitVector const& targetStates,
storm::storage::BitVector const& maybeStates) const {
storm::settings::Settings* s = storm::settings::instance();
double precision = s->get<double>("precision");
if (s->get<bool>("use-heuristic-presolve")) {
std::vector<uint_fast64_t> scheduler(submatrix.getColumnCount());
std::pair<std::vector<Type>, std::vector<uint_fast64_t>> distancesAndPredecessorsPair = storm::utility::graph::performDijkstra(this->getModel(),
this->getModel().template getBackwardTransitions<Type>([](Type const& value) -> Type { return value; }),
minimize? ~(maybeStates | targetStates) : targetStates, &maybeStates);
std::vector<uint_fast64_t> scheduler = this->convertShortestPathsToScheduler(maybeStates, distancesAndPredecessorsPair.second);
std::vector<Type> result(scheduler.size(), Type(0.5));
std::vector<Type> b(scheduler.size());
storm::utility::vector::selectVectorValues(b, scheduler, subNondeterministicChoiceIndices, rightHandSide);
@ -672,6 +680,31 @@ namespace storm {
}
}
std::vector<uint_fast64_t> convertShortestPathsToScheduler(storm::storage::BitVector const& maybeStates, std::vector<uint_fast64_t> const& shortestPathSuccessors) const {
std::vector<uint_fast64_t> scheduler(maybeStates.getNumberOfSetBits());
Type maxProbability = 0;
uint_fast64_t currentStateIndex = 0;
for (auto state : maybeStates) {
maxProbability = 0;
for (uint_fast64_t row = 0, rowEnd = this->getModel().getNondeterministicChoiceIndices()[state + 1] - this->getModel().getNondeterministicChoiceIndices()[state]; row < rowEnd; ++row) {
typename storm::storage::SparseMatrix<Type>::Rows currentRow = this->getModel().getTransitionMatrix().getRows(this->getModel().getNondeterministicChoiceIndices()[state] + row, this->getModel().getNondeterministicChoiceIndices()[state] + row);
for (auto& transition : currentRow) {
if (transition.column() == shortestPathSuccessors[state] && transition.value() > maxProbability) {
maxProbability = transition.value();
scheduler[currentStateIndex] = row;
}
}
}
++currentStateIndex;
}
return scheduler;
}
// An object that is used for solving linear equations and performing matrix-vector multiplication.
std::unique_ptr<storm::solver::AbstractNondeterministicLinearEquationSolver<Type>> linearEquationSolver;

24
src/models/AbstractDeterministicModel.h

@ -68,24 +68,16 @@ class AbstractDeterministicModel: public AbstractModel<T> {
// Intentionally left empty.
}
/*!
* Returns an iterator to the successors of the given state.
*
* @param state The state for which to return the iterator.
* @return An iterator to the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorBegin(uint_fast64_t state) const {
return this->transitionMatrix.constColumnIteratorBegin(state);
virtual typename storm::storage::SparseMatrix<T>::Rows getRows(uint_fast64_t state) const override {
return this->transitionMatrix.getRows(state, state);
}
/*!
* Returns an iterator pointing to the element past the successors of the given state.
*
* @param state The state for which to return the iterator.
* @return An iterator pointing to the element past the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorEnd(uint_fast64_t state) const {
return this->transitionMatrix.constColumnIteratorEnd(state);
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorBegin(uint_fast64_t state) const override {
return this->transitionMatrix.begin(state);
}
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorEnd(uint_fast64_t state) const override {
return this->transitionMatrix.end(state);
}
/*!

49
src/models/AbstractModel.h

@ -158,8 +158,9 @@ class AbstractModel: public std::enable_shared_from_this<AbstractModel<T>> {
// Now, we determine the blocks which are reachable (in one step) from the current block.
std::set<uint_fast64_t> allTargetBlocks;
for (auto state : block) {
for (typename storm::storage::SparseMatrix<T>::ConstIndexIterator succIt = this->constStateSuccessorIteratorBegin(state), succIte = this->constStateSuccessorIteratorEnd(state); succIt != succIte; ++succIt) {
uint_fast64_t targetBlock = stateToBlockMap[*succIt];
typename storm::storage::SparseMatrix<T>::Rows rows = this->getRows(state);
for (auto& transition : rows) {
uint_fast64_t targetBlock = stateToBlockMap[transition.column()];
// We only need to consider transitions that are actually leaving the SCC.
if (targetBlock != currentBlockIndex) {
@ -185,18 +186,32 @@ class AbstractModel: public std::enable_shared_from_this<AbstractModel<T>> {
*
* @return A sparse matrix that represents the backward transitions of this model.
*/
virtual storm::storage::SparseMatrix<bool> getBackwardTransitions() const {
storm::storage::SparseMatrix<bool> getBackwardTransitions() const {
return getBackwardTransitions<bool>([](T const& value) -> bool { return value != 0; });
}
/*!
* Retrieves the backward transition relation of the model, i.e. a set of transitions
* between states that correspond to the reversed transition relation of this model.
*
* @return A sparse matrix that represents the backward transitions of this model.
*/
template <typename TransitionType>
storm::storage::SparseMatrix<TransitionType> getBackwardTransitions(std::function<TransitionType(T const&)> const& selectionFunction) const {
uint_fast64_t numberOfStates = this->getNumberOfStates();
uint_fast64_t numberOfTransitions = this->getNumberOfTransitions();
std::vector<uint_fast64_t> rowIndications(numberOfStates + 1);
std::vector<uint_fast64_t> columnIndications(numberOfTransitions);
std::vector<bool> values(numberOfTransitions, true);
std::vector<TransitionType> values(numberOfTransitions, selectionFunction(0));
// First, we need to count how many backward transitions each state has.
for (uint_fast64_t i = 0; i < numberOfStates; ++i) {
for (auto rowIt = this->constStateSuccessorIteratorBegin(i), rowIte = this->constStateSuccessorIteratorEnd(i); rowIt != rowIte; ++rowIt) {
rowIndications[*rowIt + 1]++;
typename storm::storage::SparseMatrix<T>::Rows rows = this->getRows(i);
for (auto& transition : rows) {
if (transition.value() > 0) {
++rowIndications[transition.column() + 1];
}
}
}
@ -218,12 +233,16 @@ class AbstractModel: public std::enable_shared_from_this<AbstractModel<T>> {
// Now we are ready to actually fill in the list of predecessors for
// every state. Again, we start by considering all but the last row.
for (uint_fast64_t i = 0; i < numberOfStates; ++i) {
for (auto rowIt = this->constStateSuccessorIteratorBegin(i), rowIte = this->constStateSuccessorIteratorEnd(i); rowIt != rowIte; ++rowIt) {
columnIndications[nextIndices[*rowIt]++] = i;
typename storm::storage::SparseMatrix<T>::Rows rows = this->getRows(i);
for (auto& transition : rows) {
if (transition.value() > 0) {
values[nextIndices[transition.column()]] = selectionFunction(transition.value());
columnIndications[nextIndices[transition.column()]++] = i;
}
}
}
storm::storage::SparseMatrix<bool> backwardTransitionMatrix(numberOfStates, numberOfStates,
storm::storage::SparseMatrix<TransitionType> backwardTransitionMatrix(numberOfStates, numberOfStates,
numberOfTransitions,
std::move(rowIndications),
std::move(columnIndications),
@ -232,13 +251,21 @@ class AbstractModel: public std::enable_shared_from_this<AbstractModel<T>> {
return backwardTransitionMatrix;
}
/*!
* Returns an object representing the matrix rows associated with the given state.
*
* @param state The state for which to retrieve the rows.
* @return An object representing the matrix rows associated with the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::Rows getRows(uint_fast64_t state) const = 0;
/*!
* Returns an iterator to the successors of the given state.
*
* @param state The state for which to return the iterator.
* @return An iterator to the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorBegin(uint_fast64_t state) const = 0;
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorBegin(uint_fast64_t state) const = 0;
/*!
* Returns an iterator pointing to the element past the successors of the given state.
@ -246,7 +273,7 @@ class AbstractModel: public std::enable_shared_from_this<AbstractModel<T>> {
* @param state The state for which to return the iterator.
* @return An iterator pointing to the element past the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorEnd(uint_fast64_t state) const = 0;
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorEnd(uint_fast64_t state) const = 0;
/*!
* Returns the state space size of the model.

24
src/models/AbstractNondeterministicModel.h

@ -108,24 +108,16 @@ class AbstractNondeterministicModel: public AbstractModel<T> {
return nondeterministicChoiceIndices;
}
/*!
* Returns an iterator to the successors of the given state.
*
* @param state The state for which to return the iterator.
* @return An iterator to the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorBegin(uint_fast64_t state) const {
return this->transitionMatrix.constColumnIteratorBegin(nondeterministicChoiceIndices[state]);
virtual typename storm::storage::SparseMatrix<T>::Rows getRows(uint_fast64_t state) const override {
return this->transitionMatrix.getRows(nondeterministicChoiceIndices[state], nondeterministicChoiceIndices[state + 1] - 1);
}
/*!
* Returns an iterator pointing to the element past the successors of the given state.
*
* @param state The state for which to return the iterator.
* @return An iterator pointing to the element past the successors of the given state.
*/
virtual typename storm::storage::SparseMatrix<T>::ConstIndexIterator constStateSuccessorIteratorEnd(uint_fast64_t state) const {
return this->transitionMatrix.constColumnIteratorEnd(nondeterministicChoiceIndices[state + 1] - 1);
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorBegin(uint_fast64_t state) const override {
return this->transitionMatrix.begin(nondeterministicChoiceIndices[state]);
}
virtual typename storm::storage::SparseMatrix<T>::ConstRowIterator rowIteratorEnd(uint_fast64_t state) const override {
return this->transitionMatrix.end(nondeterministicChoiceIndices[state + 1] - 1);
}
/*!

9
src/solver/GmmxxLinearEquationSolver.h

@ -177,7 +177,7 @@ namespace storm {
std::vector<Type>* xCurrent = &x;
// Target vector for precision calculation.
std::vector<Type>* residuum = new std::vector<Type>(x.size());
std::vector<Type> tmpX(x.size());
// Set up additional environment variables.
uint_fast64_t iterationCount = 0;
@ -185,11 +185,11 @@ namespace storm {
while (!converged && iterationCount < maxIterations) {
// R * x_k (xCurrent is x_k) -> xNext
gmm::mult(*gmmxxLU, *xCurrent, *xNext);
gmm::mult(*gmmxxLU, *xCurrent, tmpX);
// b - R * x_k (xNext contains R * x_k) -> xNext
gmm::add(b, gmm::scaled(*xNext, -1.0), *xNext);
gmm::add(b, gmm::scaled(tmpX, -1.0), tmpX);
// D^-1 * (b - R * x_k) -> xNext
gmm::mult(*gmmxxDiagonalInverted, *xNext, *xNext);
gmm::mult(*gmmxxDiagonalInverted, tmpX, *xNext);
// Swap xNext with xCurrent so that the next iteration can use xCurrent again without having to copy the
// vector.
@ -214,7 +214,6 @@ namespace storm {
delete xCopy;
// Also delete the other dynamically allocated variables.
delete residuum;
delete gmmxxDiagonalInverted;
delete gmmxxLU;

72
src/storage/SparseMatrix.h

@ -96,6 +96,16 @@ public:
return *this;
}
/*!
* Dereferences the iterator by returning a reference to itself. This is needed for making use of the range-based
* for loop over transitions.
*
* @return A reference to itself.
*/
ConstIterator& operator*() {
return *this;
}
/*!
* Compares the two iterators for inequality.
*
@ -144,34 +154,34 @@ public:
};
/*!
* This class represents a single row of the matrix.
* This class represents a number of consecutive rows of the matrix.
*/
class Row {
class Rows {
public:
/*!
* Constructs a row from the given parameters.
*
* @param valuePtr A pointer to the value of the first non-zero element of the row.
* @param columnPtr A pointer to the column of the first non-zero element of the row.
* @param entryCount The number of non-zero elements of this row.
* @param valuePtr A pointer to the value of the first non-zero element of the rows.
* @param columnPtr A pointer to the column of the first non-zero element of the rows.
* @param entryCount The number of non-zero elements of the rows.
*/
Row(T const* valuePtr, uint_fast64_t const* columnPtr, uint_fast64_t entryCount) : valuePtr(valuePtr), columnPtr(columnPtr), entryCount(entryCount) {
Rows(T const* valuePtr, uint_fast64_t const* columnPtr, uint_fast64_t entryCount) : valuePtr(valuePtr), columnPtr(columnPtr), entryCount(entryCount) {
// Intentionally left empty.
}
/*!
* Retrieves an iterator that points to the beginning of the row.
* Retrieves an iterator that points to the beginning of the rows.
*
* @return An iterator that points to the beginning of the row.
* @return An iterator that points to the beginning of the rows.
*/
ConstIterator begin() const {
return ConstIterator(valuePtr, columnPtr);
}
/*!
* Retrieves an iterator that points past the last element of the row.
* Retrieves an iterator that points past the last element of the rows.
*
* @return An iterator that points past the last element of the row.
* @return An iterator that points past the last element of the rows.
*/
ConstIterator end() const {
return ConstIterator(valuePtr + entryCount, columnPtr + entryCount);
@ -184,7 +194,7 @@ public:
// The pointer to the column of the first element.
uint_fast64_t const* columnPtr;
// The number of non-zero entries in this row.
// The number of non-zero entries in the rows.
uint_fast64_t entryCount;
};
@ -214,6 +224,13 @@ public:
return *this;
}
/*!
* Compares the iterator to the given row iterator.
*/
bool operator!=(ConstRowIterator const& other) const {
return this->rowPtr != other.rowPtr;
}
/*!
* Retrieves an iterator that points to the beginning of the current row.
*
@ -1124,9 +1141,6 @@ public:
#else
ConstRowIterator rowIt = this->begin();
// std::cout << this->toString() << std::endl;
// std::cout << vector << std::endl;
for (auto resultIt = result.begin(), resultIte = result.end(); resultIt != resultIte; ++resultIt, ++rowIt) {
*resultIt = storm::utility::constGetZero<T>();
@ -1134,7 +1148,6 @@ public:
*resultIt += elementIt.value() * vector[elementIt.column()];
}
}
// std::cout << result << std::endl;
#endif
}
@ -1154,18 +1167,25 @@ public:
return size;
}
void moveToRow(ConstIterator& it, uint_fast64_t rowOffset) const {
// std::cout << rowOffset << " / " << this->nonZeroEntryCount << std::endl;
it.moveTo(this->valueStorage.data() + rowOffset, this->columnIndications.data() + rowOffset);
/*!
* Returns an object representing the consecutive rows given by the parameters.
*
* @param The starting row.
* @param The ending row (which is included in the result).
* @return An object representing the consecutive rows given by the parameters.
*/
Rows getRows(uint_fast64_t startRow, uint_fast64_t endRow) const {
return Rows(this->valueStorage.data() + this->rowIndications[startRow], this->columnIndications.data() + this->rowIndications[startRow], this->rowIndications[endRow + 1] - this->rowIndications[startRow]);
}
/*!
* Returns a const iterator to the rows of the matrix.
*
* @param initialRow The initial row to which this iterator points.
* @return A const iterator to the rows of the matrix.
*/
ConstRowIterator begin() const {
return ConstRowIterator(this->valueStorage.data(), this->columnIndications.data(), this->rowIndications.data());
ConstRowIterator begin(uint_fast64_t initialRow = 0) const {
return ConstRowIterator(this->valueStorage.data(), this->columnIndications.data(), this->rowIndications.data() + initialRow);
}
/*!
@ -1174,7 +1194,17 @@ public:
* @return A const iterator that points to past the last row of the matrix
*/
ConstRowIterator end() const {
return ConstRowIterator(this->valueStorage.data() + nonZeroEntryCount, this->columnIndications.data() + nonZeroEntryCount, this->rowIndications.data() + rowCount);
return ConstRowIterator(this->valueStorage.data(), this->columnIndications.data(), this->rowIndications.data() + rowCount);
}
/*!
* Returns a const iterator that points past the given row.
*
* @param row The row past which this iterator points.
* @return A const iterator that points past the given row.
*/
ConstRowIterator end(uint_fast64_t row) const {
return ConstRowIterator(this->valueStorage.data(), this->columnIndications.data(), this->rowIndications.data() + row + 1);
}
/*!

2
src/utility/ConstTemplates.h

@ -160,7 +160,7 @@ inline int_fast32_t constGetInfinity() {
template<>
inline uint_fast64_t constGetInfinity() {
throw storm::exceptions::InvalidArgumentException() << "Integer has no infinity.";
return std::numeric_limits<int_fast64_t>::max();
return std::numeric_limits<uint_fast64_t>::max();
}
/*!

70
src/utility/graph.h

@ -8,8 +8,12 @@
#ifndef STORM_UTILITY_GRAPH_H_
#define STORM_UTILITY_GRAPH_H_
#include <set>
#include <limits>
#include "src/models/AbstractDeterministicModel.h"
#include "src/models/AbstractNondeterministicModel.h"
#include "ConstTemplates.h"
#include "src/exceptions/InvalidArgumentException.h"
#include "log4cplus/logger.h"
@ -757,6 +761,72 @@ namespace graph {
return topologicalSort;
}
template<typename T>
struct DistanceCompare {
bool operator()(std::pair<T, uint_fast64_t> const& lhs, std::pair<T, uint_fast64_t> const& rhs) const {
return lhs.first < rhs.first || (lhs.first == rhs.first && lhs.second < rhs.second);
}
};
template <typename T>
std::pair<std::vector<T>, std::vector<uint_fast64_t>> performDijkstra(storm::models::AbstractModel<T> const& model,
storm::storage::SparseMatrix<T> const& transitions,
storm::storage::BitVector const& startingStates,
// FIXME: The case in which there are target states given should be handled properly.
// storm::storage::BitVector* targetStates = nullptr,
storm::storage::BitVector const* filterStates = nullptr) {
LOG4CPLUS_INFO(logger, "Performing Dijkstra search.");
const uint_fast64_t noPredecessorValue = std::numeric_limits<uint_fast64_t>::max();
std::vector<T> distances(model.getNumberOfStates(), storm::utility::constGetInfinity<T>());
std::vector<uint_fast64_t> predecessors(model.getNumberOfStates(), noPredecessorValue);
// Set the distance to 0 for all starting states.
std::set<std::pair<T, uint_fast64_t>, DistanceCompare<T>> distanceStateSet;
for (auto state : startingStates) {
distanceStateSet.emplace(storm::utility::constGetZero<T>(), state);
distances[state] = 0;
}
while (!distanceStateSet.empty()) {
// Get the state with the least distance from the set and remove it.
std::pair<T, uint_fast64_t> distanceStatePair = *distanceStateSet.begin();
distanceStateSet.erase(distanceStateSet.begin());
// Now check the new distances for all successors of the current state.
typename storm::storage::SparseMatrix<T>::Rows row = transitions.getRows(distanceStatePair.second, distanceStatePair.second);
for (auto& transition : row) {
// Only follow the transition if it lies within the filtered states.
if (filterStates != nullptr && filterStates->get(transition.column())) {
// Calculate the distance we achieve when we take the path to the successor via the current state.
T newDistance = distanceStatePair.first + std::log(storm::utility::constGetOne<T>() / transition.value());
// We found a cheaper way to get to the target state of the transition.
if (newDistance < distances[transition.column()]) {
// Remove the old distance.
if (distances[transition.column()] != noPredecessorValue) {
distanceStateSet.erase(std::make_pair(distances[transition.column()], transition.column()));
}
// Set and add the new distance.
distances[transition.column()] = newDistance;
predecessors[transition.column()] = distanceStatePair.second;
distanceStateSet.insert(std::make_pair(newDistance, transition.column()));
}
}
}
}
std::pair<std::vector<T>, std::vector<uint_fast64_t>> result;
result.first = std::move(distances);
result.second = std::move(predecessors);
LOG4CPLUS_INFO(logger, "Done performing Dijkstra search.");
return result;
}
} // namespace graph
} // namespace utility

Loading…
Cancel
Save