From ee7a7e0bcf5ee667a89af3bee9e39426537d8980 Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Tue, 3 Jan 2023 10:54:28 -0500 Subject: [PATCH 1/6] New test with two modes --- gtsam/hybrid/tests/TinyHybridExample.h | 48 +++++++---- .../tests/testHybridGaussianFactorGraph.cpp | 83 ++++++++++++++++--- 2 files changed, 102 insertions(+), 29 deletions(-) diff --git a/gtsam/hybrid/tests/TinyHybridExample.h b/gtsam/hybrid/tests/TinyHybridExample.h index ba04263f87..3ff9bec859 100644 --- a/gtsam/hybrid/tests/TinyHybridExample.h +++ b/gtsam/hybrid/tests/TinyHybridExample.h @@ -33,17 +33,21 @@ const DiscreteKey mode{M(0), 2}; /** * Create a tiny two variable hybrid model which represents * the generative probability P(z,x,mode) = P(z|x,mode)P(x)P(mode). + * numMeasurements is the number of measurements of the continuous variable x0. + * If manyModes is true, then we introduce one mode per measurement. */ -inline HybridBayesNet createHybridBayesNet(int num_measurements = 1) { +inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, + bool manyModes = false) { HybridBayesNet bayesNet; // Create Gaussian mixture z_i = x0 + noise for each measurement. - for (int i = 0; i < num_measurements; i++) { + for (int i = 0; i < numMeasurements; i++) { const auto conditional0 = boost::make_shared( GaussianConditional::FromMeanAndStddev(Z(i), I_1x1, X(0), Z_1x1, 0.5)); const auto conditional1 = boost::make_shared( GaussianConditional::FromMeanAndStddev(Z(i), I_1x1, X(0), Z_1x1, 3)); - GaussianMixture gm({Z(i)}, {X(0)}, {mode}, {conditional0, conditional1}); + const auto mode_i = manyModes ? DiscreteKey{M(i), 2} : mode; + GaussianMixture gm({Z(i)}, {X(0)}, {mode_i}, {conditional0, conditional1}); bayesNet.emplaceMixture(gm); // copy :-( } @@ -53,8 +57,10 @@ inline HybridBayesNet createHybridBayesNet(int num_measurements = 1) { bayesNet.emplaceGaussian(prior_on_x0); // copy :-( // Add prior on mode. - bayesNet.emplaceDiscrete(mode, "4/6"); - + const size_t nrModes = manyModes ? numMeasurements : 1; + for (int i = 0; i < nrModes; i++) { + bayesNet.emplaceDiscrete(DiscreteKey{M(i), 2}, "4/6"); + } return bayesNet; } @@ -64,14 +70,21 @@ inline HybridBayesNet createHybridBayesNet(int num_measurements = 1) { inline HybridGaussianFactorGraph convertBayesNet( const HybridBayesNet& bayesNet, const VectorValues& measurements) { HybridGaussianFactorGraph fg; - int num_measurements = bayesNet.size() - 2; - for (int i = 0; i < num_measurements; i++) { - auto conditional = bayesNet.atMixture(i); - auto factor = conditional->likelihood({{Z(i), measurements.at(Z(i))}}); - fg.push_back(factor); + // For all nodes in the Bayes net, if its frontal variable is in measurements, + // replace it by a likelihood factor: + for (const HybridConditional::shared_ptr& conditional : bayesNet) { + if (measurements.exists(conditional->firstFrontalKey())) { + if (auto gc = conditional->asGaussian()) + fg.push_back(gc->likelihood(measurements)); + else if (auto gm = conditional->asMixture()) + fg.push_back(gm->likelihood(measurements)); + else { + throw std::runtime_error("Unknown conditional type"); + } + } else { + fg.push_back(conditional); + } } - fg.push_back(bayesNet.atGaussian(num_measurements)); - fg.push_back(bayesNet.atDiscrete(num_measurements + 1)); return fg; } @@ -79,15 +92,18 @@ inline HybridGaussianFactorGraph convertBayesNet( * Create a tiny two variable hybrid factor graph which represents a discrete * mode and a continuous variable x0, given a number of measurements of the * continuous variable x0. If no measurements are given, they are sampled from - * the generative Bayes net model HybridBayesNet::Example(num_measurements) + * the generative Bayes net model HybridBayesNet::Example(numMeasurements) */ inline HybridGaussianFactorGraph createHybridGaussianFactorGraph( - int num_measurements = 1, - boost::optional measurements = boost::none) { - auto bayesNet = createHybridBayesNet(num_measurements); + int numMeasurements = 1, + boost::optional measurements = boost::none, + bool manyModes = false) { + auto bayesNet = createHybridBayesNet(numMeasurements, manyModes); if (measurements) { + // Use the measurements to create a hybrid factor graph. return convertBayesNet(bayesNet, *measurements); } else { + // Sample from the generative model to create a hybrid factor graph. return convertBayesNet(bayesNet, bayesNet.sample().continuous()); } } diff --git a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp index fa371cf163..6697e50842 100644 --- a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp +++ b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp @@ -619,44 +619,51 @@ TEST(HybridGaussianFactorGraph, ErrorAndProbPrimeTree) { // assignment. TEST(HybridGaussianFactorGraph, assembleGraphTree) { using symbol_shorthand::Z; - const int num_measurements = 1; + const int numMeasurements = 1; auto fg = tiny::createHybridGaussianFactorGraph( - num_measurements, VectorValues{{Z(0), Vector1(5.0)}}); + numMeasurements, VectorValues{{Z(0), Vector1(5.0)}}); EXPECT_LONGS_EQUAL(3, fg.size()); - auto sum = fg.assembleGraphTree(); + // Assemble graph tree: + auto actual = fg.assembleGraphTree(); + + // Create expected decision tree with two factor graphs: // Get mixture factor: auto mixture = boost::dynamic_pointer_cast(fg.at(0)); - using GF = GaussianFactor::shared_ptr; + CHECK(mixture); // Get prior factor: - const GF prior = - boost::dynamic_pointer_cast(fg.at(1))->inner(); + const auto gf = boost::dynamic_pointer_cast(fg.at(1)); + CHECK(gf); + using GF = GaussianFactor::shared_ptr; + const GF prior = gf->asGaussian(); + CHECK(prior); // Create DiscreteValues for both 0 and 1: DiscreteValues d0{{M(0), 0}}, d1{{M(0), 1}}; // Expected decision tree with two factor graphs: // f(x0;mode=0)P(x0) and f(x0;mode=1)P(x0) - GaussianFactorGraphTree expectedSum{ + GaussianFactorGraphTree expected{ M(0), {GaussianFactorGraph(std::vector{mixture->factor(d0), prior}), mixture->constant(d0)}, {GaussianFactorGraph(std::vector{mixture->factor(d1), prior}), mixture->constant(d1)}}; - EXPECT(assert_equal(expectedSum(d0), sum(d0), 1e-5)); - EXPECT(assert_equal(expectedSum(d1), sum(d1), 1e-5)); + EXPECT(assert_equal(expected(d0), actual(d0), 1e-5)); + EXPECT(assert_equal(expected(d1), actual(d1), 1e-5)); } /* ****************************************************************************/ // Check that eliminating tiny net with 1 measurement yields correct result. TEST(HybridGaussianFactorGraph, EliminateTiny1) { using symbol_shorthand::Z; - const int num_measurements = 1; + const int numMeasurements = 1; auto fg = tiny::createHybridGaussianFactorGraph( - num_measurements, VectorValues{{Z(0), Vector1(5.0)}}); + numMeasurements, VectorValues{{Z(0), Vector1(5.0)}}); + EXPECT_LONGS_EQUAL(3, fg.size()); // Create expected Bayes Net: HybridBayesNet expectedBayesNet; @@ -687,10 +694,11 @@ TEST(HybridGaussianFactorGraph, EliminateTiny1) { TEST(HybridGaussianFactorGraph, EliminateTiny2) { // Create factor graph with 2 measurements such that posterior mean = 5.0. using symbol_shorthand::Z; - const int num_measurements = 2; + const int numMeasurements = 2; auto fg = tiny::createHybridGaussianFactorGraph( - num_measurements, + numMeasurements, VectorValues{{Z(0), Vector1(4.0)}, {Z(1), Vector1(6.0)}}); + EXPECT_LONGS_EQUAL(4, fg.size()); // Create expected Bayes Net: HybridBayesNet expectedBayesNet; @@ -716,6 +724,55 @@ TEST(HybridGaussianFactorGraph, EliminateTiny2) { EXPECT(assert_equal(expectedBayesNet, *posterior, 0.01)); } +/* ****************************************************************************/ +// Test eliminating tiny net with 1 mode per measurement. +TEST(HybridGaussianFactorGraph, EliminateTiny22) { + // Create factor graph with 2 measurements such that posterior mean = 5.0. + using symbol_shorthand::Z; + const int numMeasurements = 2; + const bool manyModes = true; + + // Create Bayes net and convert to factor graph. + auto bn = tiny::createHybridBayesNet(numMeasurements, manyModes); + const VectorValues measurements{{Z(0), Vector1(4.0)}, {Z(1), Vector1(6.0)}}; + auto fg = tiny::convertBayesNet(bn, measurements); + EXPECT_LONGS_EQUAL(5, fg.size()); + + // Test elimination + Ordering ordering; + ordering.push_back(X(0)); + ordering.push_back(M(0)); + ordering.push_back(M(1)); + const auto posterior = fg.eliminateSequential(ordering); + + // Compute the log-ratio between the Bayes net and the factor graph. + auto compute_ratio = [&](HybridValues *sample) -> double { + // update sample with given measurements: + sample->update(measurements); + return bn.evaluate(*sample) / posterior->evaluate(*sample); + }; + + // Set up sampling + std::mt19937_64 rng(42); + + // The error evaluated by the factor graph and the Bayes net should differ by + // the normalizing term computed via the Bayes net determinant. + HybridValues sample = bn.sample(&rng); + double expected_ratio = compute_ratio(&sample); + // regression + EXPECT_DOUBLES_EQUAL(0.018253037966018862, expected_ratio, 1e-6); + + // 3. Do sampling + constexpr int num_samples = 100; + for (size_t i = 0; i < num_samples; i++) { + // Sample from the bayes net + HybridValues sample = bn.sample(&rng); + + // Check that the ratio is constant. + EXPECT_DOUBLES_EQUAL(expected_ratio, compute_ratio(&sample), 1e-6); + } +} + /* ************************************************************************* */ int main() { TestResult tr; From e30813e81ec267bca72810dc7c1b253ffb32b4fc Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Wed, 4 Jan 2023 19:52:37 -0800 Subject: [PATCH 2/6] toFactorGraph method in HybridBayesNet --- gtsam/hybrid/HybridBayesNet.cpp | 23 ++++++++++++++ gtsam/hybrid/HybridBayesNet.h | 6 ++++ gtsam/hybrid/HybridConditional.h | 10 +++++++ gtsam/hybrid/tests/TinyHybridExample.h | 30 ++----------------- .../tests/testHybridGaussianFactorGraph.cpp | 2 +- gtsam/linear/GaussianConditional.cpp | 2 +- 6 files changed, 44 insertions(+), 29 deletions(-) diff --git a/gtsam/hybrid/HybridBayesNet.cpp b/gtsam/hybrid/HybridBayesNet.cpp index 4404ccfdc8..e01fcbdcf2 100644 --- a/gtsam/hybrid/HybridBayesNet.cpp +++ b/gtsam/hybrid/HybridBayesNet.cpp @@ -377,4 +377,27 @@ AlgebraicDecisionTree HybridBayesNet::probPrime( return error_tree.apply([](double error) { return exp(-error); }); } +/* ************************************************************************* */ +HybridGaussianFactorGraph HybridBayesNet::toFactorGraph( + const VectorValues &measurements) const { + HybridGaussianFactorGraph fg; + + // For all nodes in the Bayes net, if its frontal variable is in measurements, + // replace it by a likelihood factor: + for (auto &&conditional : *this) { + if (conditional->frontalsIn(measurements)) { + if (auto gc = conditional->asGaussian()) + fg.push_back(gc->likelihood(measurements)); + else if (auto gm = conditional->asMixture()) + fg.push_back(gm->likelihood(measurements)); + else { + throw std::runtime_error("Unknown conditional type"); + } + } else { + fg.push_back(conditional); + } + } + return fg; +} + } // namespace gtsam diff --git a/gtsam/hybrid/HybridBayesNet.h b/gtsam/hybrid/HybridBayesNet.h index dcdf3a8e5c..de0a38271c 100644 --- a/gtsam/hybrid/HybridBayesNet.h +++ b/gtsam/hybrid/HybridBayesNet.h @@ -229,6 +229,12 @@ class GTSAM_EXPORT HybridBayesNet : public BayesNet { AlgebraicDecisionTree probPrime( const VectorValues &continuousValues) const; + /** + * Convert a hybrid Bayes net to a hybrid Gaussian factor graph by converting + * all conditionals with instantiated measurements into likelihood factors. + */ + HybridGaussianFactorGraph toFactorGraph( + const VectorValues &measurements) const; /// @} private: diff --git a/gtsam/hybrid/HybridConditional.h b/gtsam/hybrid/HybridConditional.h index 021ca13614..6c2f4a65bd 100644 --- a/gtsam/hybrid/HybridConditional.h +++ b/gtsam/hybrid/HybridConditional.h @@ -178,6 +178,16 @@ class GTSAM_EXPORT HybridConditional /// Return the error of the underlying conditional. double error(const HybridValues& values) const override; + /// Check if VectorValues `measurements` contains all frontal keys. + bool frontalsIn(const VectorValues& measurements) const { + for (Key key : frontals()) { + if (!measurements.exists(key)) { + return false; + } + } + return true; + } + /// @} private: diff --git a/gtsam/hybrid/tests/TinyHybridExample.h b/gtsam/hybrid/tests/TinyHybridExample.h index 3ff9bec859..2e1a66a8e9 100644 --- a/gtsam/hybrid/tests/TinyHybridExample.h +++ b/gtsam/hybrid/tests/TinyHybridExample.h @@ -37,7 +37,7 @@ const DiscreteKey mode{M(0), 2}; * If manyModes is true, then we introduce one mode per measurement. */ inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, - bool manyModes = false) { + bool manyModes = false) { HybridBayesNet bayesNet; // Create Gaussian mixture z_i = x0 + noise for each measurement. @@ -64,30 +64,6 @@ inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, return bayesNet; } -/** - * Convert a hybrid Bayes net to a hybrid Gaussian factor graph. - */ -inline HybridGaussianFactorGraph convertBayesNet( - const HybridBayesNet& bayesNet, const VectorValues& measurements) { - HybridGaussianFactorGraph fg; - // For all nodes in the Bayes net, if its frontal variable is in measurements, - // replace it by a likelihood factor: - for (const HybridConditional::shared_ptr& conditional : bayesNet) { - if (measurements.exists(conditional->firstFrontalKey())) { - if (auto gc = conditional->asGaussian()) - fg.push_back(gc->likelihood(measurements)); - else if (auto gm = conditional->asMixture()) - fg.push_back(gm->likelihood(measurements)); - else { - throw std::runtime_error("Unknown conditional type"); - } - } else { - fg.push_back(conditional); - } - } - return fg; -} - /** * Create a tiny two variable hybrid factor graph which represents a discrete * mode and a continuous variable x0, given a number of measurements of the @@ -101,10 +77,10 @@ inline HybridGaussianFactorGraph createHybridGaussianFactorGraph( auto bayesNet = createHybridBayesNet(numMeasurements, manyModes); if (measurements) { // Use the measurements to create a hybrid factor graph. - return convertBayesNet(bayesNet, *measurements); + return bayesNet.toFactorGraph(*measurements); } else { // Sample from the generative model to create a hybrid factor graph. - return convertBayesNet(bayesNet, bayesNet.sample().continuous()); + return bayesNet.toFactorGraph(bayesNet.sample().continuous()); } } diff --git a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp index 6697e50842..bfaf6d264e 100644 --- a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp +++ b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp @@ -735,7 +735,7 @@ TEST(HybridGaussianFactorGraph, EliminateTiny22) { // Create Bayes net and convert to factor graph. auto bn = tiny::createHybridBayesNet(numMeasurements, manyModes); const VectorValues measurements{{Z(0), Vector1(4.0)}, {Z(1), Vector1(6.0)}}; - auto fg = tiny::convertBayesNet(bn, measurements); + auto fg = bn.toFactorGraph(measurements); EXPECT_LONGS_EQUAL(5, fg.size()); // Test elimination diff --git a/gtsam/linear/GaussianConditional.cpp b/gtsam/linear/GaussianConditional.cpp index 39a21a6172..9fd217abf8 100644 --- a/gtsam/linear/GaussianConditional.cpp +++ b/gtsam/linear/GaussianConditional.cpp @@ -263,7 +263,7 @@ double GaussianConditional::evaluate(const VectorValues& x) const { Vector frontalVec = gy.vector(KeyVector(beginFrontals(), endFrontals())); frontalVec = R().transpose().triangularView().solve(frontalVec); - // Check for indeterminant solution + // Check for indeterminate solution if (frontalVec.hasNaN()) throw IndeterminantLinearSystemException(this->keys().front()); for (const_iterator it = beginParents(); it!= endParents(); it++) From cd0164bd8bd6cac5226bec718d1ca918099b62e4 Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Wed, 4 Jan 2023 20:25:48 -0800 Subject: [PATCH 3/6] Add forwarding template for mean-stddev variants --- gtsam/linear/GaussianConditional.h | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/gtsam/linear/GaussianConditional.h b/gtsam/linear/GaussianConditional.h index d25efb2e1a..05b8b86b88 100644 --- a/gtsam/linear/GaussianConditional.h +++ b/gtsam/linear/GaussianConditional.h @@ -100,6 +100,12 @@ namespace gtsam { const Matrix& A2, Key parent2, const Vector& b, double sigma); + /// Create shared pointer by forwarding arguments to fromMeanAndStddev. + template + static shared_ptr sharedMeanAndStddev(Args&&... args) { + return boost::make_shared(FromMeanAndStddev(std::forward(args)...)); + } + /** Combine several GaussianConditional into a single dense GC. The conditionals enumerated by * \c first and \c last must be in increasing order, meaning that the parents of any * conditional may not include a conditional coming before it. From 9eb4ac684832c9cff5502e00b234e01835833ced Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Wed, 4 Jan 2023 21:02:54 -0800 Subject: [PATCH 4/6] Test elimination of a switching network with one mode per measurement. --- .../tests/testHybridGaussianFactorGraph.cpp | 99 +++++++++++++++++++ 1 file changed, 99 insertions(+) diff --git a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp index bfaf6d264e..0fd94b300e 100644 --- a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp +++ b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp @@ -54,8 +54,10 @@ using namespace gtsam; using gtsam::symbol_shorthand::D; using gtsam::symbol_shorthand::M; +using gtsam::symbol_shorthand::N; using gtsam::symbol_shorthand::X; using gtsam::symbol_shorthand::Y; +using gtsam::symbol_shorthand::Z; /* ************************************************************************* */ TEST(HybridGaussianFactorGraph, Creation) { @@ -773,6 +775,103 @@ TEST(HybridGaussianFactorGraph, EliminateTiny22) { } } +/* ****************************************************************************/ +// Test elimination of a switching network with one mode per measurement. +TEST(HybridGaussianFactorGraph, EliminateSwitchingNetwork) { + // Create a switching network with one mode per measurement. + HybridBayesNet bn; + + // NOTE: we add reverse topological so we can sample from the Bayes net.: + + // Add measurements: + for (size_t t : {0, 1, 2}) { + // Create Gaussian mixture on Z(t) conditioned on X(t) and mode N(t): + const auto noise_mode_t = DiscreteKey{N(t), 2}; + GaussianMixture gm({Z(t)}, {X(t)}, {noise_mode_t}, + {GaussianConditional::sharedMeanAndStddev( + Z(t), I_1x1, X(t), Z_1x1, 0.5), + GaussianConditional::sharedMeanAndStddev( + Z(t), I_1x1, X(t), Z_1x1, 3.0)}); + bn.emplaceMixture(gm); // copy :-( + + // Create prior on discrete mode M(t): + bn.emplaceDiscrete(noise_mode_t, "20/80"); + } + + // Add motion models: + for (size_t t : {2, 1}) { + // Create Gaussian mixture on X(t) conditioned on X(t-1) and mode M(t-1): + const auto motion_model_t = DiscreteKey{M(t), 2}; + GaussianMixture gm({X(t)}, {X(t - 1)}, {motion_model_t}, + {GaussianConditional::sharedMeanAndStddev( + X(t), I_1x1, X(t - 1), Z_1x1, 0.2), + GaussianConditional::sharedMeanAndStddev( + X(t), I_1x1, X(t - 1), I_1x1, 0.2)}); + bn.emplaceMixture(gm); // copy :-( + + // Create prior on motion model M(t): + bn.emplaceDiscrete(motion_model_t, "40/60"); + } + + // Create Gaussian prior on continuous X(0) using sharedMeanAndStddev: + bn.addGaussian(GaussianConditional::sharedMeanAndStddev(X(0), Z_1x1, 0.1)); + + // Make sure we an sample from the Bayes net: + EXPECT_LONGS_EQUAL(6, bn.sample().continuous().size()); + + // Create measurements consistent with moving right every time: + const VectorValues measurements{ + {Z(0), Vector1(0.0)}, {Z(1), Vector1(1.0)}, {Z(2), Vector1(2.0)}}; + const auto fg = bn.toFactorGraph(measurements); + + // Create ordering that eliminates in time order, then discrete modes: + Ordering ordering; + ordering.push_back(X(2)); + ordering.push_back(X(1)); + ordering.push_back(X(0)); + ordering.push_back(N(0)); + ordering.push_back(N(1)); + ordering.push_back(N(2)); + ordering.push_back(M(1)); + ordering.push_back(M(2)); + + // Test elimination result has correct size: + const auto posterior = fg.eliminateSequential(ordering); + // GTSAM_PRINT(*posterior); + + // Test elimination result has correct size: + EXPECT_LONGS_EQUAL(8, posterior->size()); + + // TODO(dellaert): below is copy/pasta from above, refactor + + // Compute the log-ratio between the Bayes net and the factor graph. + auto compute_ratio = [&](HybridValues *sample) -> double { + // update sample with given measurements: + sample->update(measurements); + return bn.evaluate(*sample) / posterior->evaluate(*sample); + }; + + // Set up sampling + std::mt19937_64 rng(42); + + // The error evaluated by the factor graph and the Bayes net should differ by + // the normalizing term computed via the Bayes net determinant. + HybridValues sample = bn.sample(&rng); + double expected_ratio = compute_ratio(&sample); + // regression + EXPECT_DOUBLES_EQUAL(0.0094526745785019472, expected_ratio, 1e-6); + + // 3. Do sampling + constexpr int num_samples = 100; + for (size_t i = 0; i < num_samples; i++) { + // Sample from the bayes net + HybridValues sample = bn.sample(&rng); + + // Check that the ratio is constant. + EXPECT_DOUBLES_EQUAL(expected_ratio, compute_ratio(&sample), 1e-6); + } +} + /* ************************************************************************* */ int main() { TestResult tr; From 381087f48db0fbcd9306ee00a12ae2d06e6b5eb3 Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Wed, 4 Jan 2023 21:10:48 -0800 Subject: [PATCH 5/6] use sharedMeanAndStddev --- gtsam/hybrid/tests/TinyHybridExample.h | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/gtsam/hybrid/tests/TinyHybridExample.h b/gtsam/hybrid/tests/TinyHybridExample.h index 2e1a66a8e9..c9633ec55b 100644 --- a/gtsam/hybrid/tests/TinyHybridExample.h +++ b/gtsam/hybrid/tests/TinyHybridExample.h @@ -42,19 +42,18 @@ inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, // Create Gaussian mixture z_i = x0 + noise for each measurement. for (int i = 0; i < numMeasurements; i++) { - const auto conditional0 = boost::make_shared( - GaussianConditional::FromMeanAndStddev(Z(i), I_1x1, X(0), Z_1x1, 0.5)); - const auto conditional1 = boost::make_shared( - GaussianConditional::FromMeanAndStddev(Z(i), I_1x1, X(0), Z_1x1, 3)); const auto mode_i = manyModes ? DiscreteKey{M(i), 2} : mode; - GaussianMixture gm({Z(i)}, {X(0)}, {mode_i}, {conditional0, conditional1}); + GaussianMixture gm({Z(i)}, {X(0)}, {mode_i}, + {GaussianConditional::sharedMeanAndStddev( + Z(i), I_1x1, X(0), Z_1x1, 0.5), + GaussianConditional::sharedMeanAndStddev( + Z(i), I_1x1, X(0), Z_1x1, 3)}); bayesNet.emplaceMixture(gm); // copy :-( } // Create prior on X(0). - const auto prior_on_x0 = - GaussianConditional::FromMeanAndStddev(X(0), Vector1(5.0), 0.5); - bayesNet.emplaceGaussian(prior_on_x0); // copy :-( + bayesNet.addGaussian( + GaussianConditional::sharedMeanAndStddev(X(0), Vector1(5.0), 0.5)); // Add prior on mode. const size_t nrModes = manyModes ? numMeasurements : 1; From 26575921b0d47cea4d9bf805a868e82e8bc51516 Mon Sep 17 00:00:00 2001 From: Frank Dellaert Date: Wed, 4 Jan 2023 21:16:28 -0800 Subject: [PATCH 6/6] Reverted variable name change to ease reviewing --- gtsam/hybrid/tests/TinyHybridExample.h | 14 +++++++------- .../tests/testHybridGaussianFactorGraph.cpp | 16 ++++++++-------- 2 files changed, 15 insertions(+), 15 deletions(-) diff --git a/gtsam/hybrid/tests/TinyHybridExample.h b/gtsam/hybrid/tests/TinyHybridExample.h index c9633ec55b..a427d20429 100644 --- a/gtsam/hybrid/tests/TinyHybridExample.h +++ b/gtsam/hybrid/tests/TinyHybridExample.h @@ -33,15 +33,15 @@ const DiscreteKey mode{M(0), 2}; /** * Create a tiny two variable hybrid model which represents * the generative probability P(z,x,mode) = P(z|x,mode)P(x)P(mode). - * numMeasurements is the number of measurements of the continuous variable x0. + * num_measurements is the number of measurements of the continuous variable x0. * If manyModes is true, then we introduce one mode per measurement. */ -inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, +inline HybridBayesNet createHybridBayesNet(int num_measurements = 1, bool manyModes = false) { HybridBayesNet bayesNet; // Create Gaussian mixture z_i = x0 + noise for each measurement. - for (int i = 0; i < numMeasurements; i++) { + for (int i = 0; i < num_measurements; i++) { const auto mode_i = manyModes ? DiscreteKey{M(i), 2} : mode; GaussianMixture gm({Z(i)}, {X(0)}, {mode_i}, {GaussianConditional::sharedMeanAndStddev( @@ -56,7 +56,7 @@ inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, GaussianConditional::sharedMeanAndStddev(X(0), Vector1(5.0), 0.5)); // Add prior on mode. - const size_t nrModes = manyModes ? numMeasurements : 1; + const size_t nrModes = manyModes ? num_measurements : 1; for (int i = 0; i < nrModes; i++) { bayesNet.emplaceDiscrete(DiscreteKey{M(i), 2}, "4/6"); } @@ -67,13 +67,13 @@ inline HybridBayesNet createHybridBayesNet(int numMeasurements = 1, * Create a tiny two variable hybrid factor graph which represents a discrete * mode and a continuous variable x0, given a number of measurements of the * continuous variable x0. If no measurements are given, they are sampled from - * the generative Bayes net model HybridBayesNet::Example(numMeasurements) + * the generative Bayes net model HybridBayesNet::Example(num_measurements) */ inline HybridGaussianFactorGraph createHybridGaussianFactorGraph( - int numMeasurements = 1, + int num_measurements = 1, boost::optional measurements = boost::none, bool manyModes = false) { - auto bayesNet = createHybridBayesNet(numMeasurements, manyModes); + auto bayesNet = createHybridBayesNet(num_measurements, manyModes); if (measurements) { // Use the measurements to create a hybrid factor graph. return bayesNet.toFactorGraph(*measurements); diff --git a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp index 0fd94b300e..cc45718751 100644 --- a/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp +++ b/gtsam/hybrid/tests/testHybridGaussianFactorGraph.cpp @@ -621,9 +621,9 @@ TEST(HybridGaussianFactorGraph, ErrorAndProbPrimeTree) { // assignment. TEST(HybridGaussianFactorGraph, assembleGraphTree) { using symbol_shorthand::Z; - const int numMeasurements = 1; + const int num_measurements = 1; auto fg = tiny::createHybridGaussianFactorGraph( - numMeasurements, VectorValues{{Z(0), Vector1(5.0)}}); + num_measurements, VectorValues{{Z(0), Vector1(5.0)}}); EXPECT_LONGS_EQUAL(3, fg.size()); // Assemble graph tree: @@ -662,9 +662,9 @@ TEST(HybridGaussianFactorGraph, assembleGraphTree) { // Check that eliminating tiny net with 1 measurement yields correct result. TEST(HybridGaussianFactorGraph, EliminateTiny1) { using symbol_shorthand::Z; - const int numMeasurements = 1; + const int num_measurements = 1; auto fg = tiny::createHybridGaussianFactorGraph( - numMeasurements, VectorValues{{Z(0), Vector1(5.0)}}); + num_measurements, VectorValues{{Z(0), Vector1(5.0)}}); EXPECT_LONGS_EQUAL(3, fg.size()); // Create expected Bayes Net: @@ -696,9 +696,9 @@ TEST(HybridGaussianFactorGraph, EliminateTiny1) { TEST(HybridGaussianFactorGraph, EliminateTiny2) { // Create factor graph with 2 measurements such that posterior mean = 5.0. using symbol_shorthand::Z; - const int numMeasurements = 2; + const int num_measurements = 2; auto fg = tiny::createHybridGaussianFactorGraph( - numMeasurements, + num_measurements, VectorValues{{Z(0), Vector1(4.0)}, {Z(1), Vector1(6.0)}}); EXPECT_LONGS_EQUAL(4, fg.size()); @@ -731,11 +731,11 @@ TEST(HybridGaussianFactorGraph, EliminateTiny2) { TEST(HybridGaussianFactorGraph, EliminateTiny22) { // Create factor graph with 2 measurements such that posterior mean = 5.0. using symbol_shorthand::Z; - const int numMeasurements = 2; + const int num_measurements = 2; const bool manyModes = true; // Create Bayes net and convert to factor graph. - auto bn = tiny::createHybridBayesNet(numMeasurements, manyModes); + auto bn = tiny::createHybridBayesNet(num_measurements, manyModes); const VectorValues measurements{{Z(0), Vector1(4.0)}, {Z(1), Vector1(6.0)}}; auto fg = bn.toFactorGraph(measurements); EXPECT_LONGS_EQUAL(5, fg.size());