2022-01-19 23:29:50 +08:00
|
|
|
/**
|
|
|
|
* @file
|
2023-07-20 04:53:16 +08:00
|
|
|
* @brief
|
|
|
|
* Implementation to calculate an estimate of the [number π
|
|
|
|
* (Pi)](https://en.wikipedia.org/wiki/File:Pi_30K.gif).
|
2022-01-19 23:29:50 +08:00
|
|
|
*
|
|
|
|
* @details
|
2023-07-20 04:53:16 +08:00
|
|
|
* We take a random point P with coordinates (x, y) such that 0 ≤ x ≤ 1 and 0 ≤
|
|
|
|
* y ≤ 1. If x² + y² ≤ 1, then the point is inside the quarter disk of radius 1,
|
|
|
|
* else the point is outside. We know that the probability of the point being
|
|
|
|
* inside the quarter disk is equal to π/4 double approx(vector<Point> &pts)
|
|
|
|
* which will use the points pts (drawn at random) to return an estimate of the
|
|
|
|
* number π
|
|
|
|
* @note This implementation is better than naive recursive or iterative
|
2022-01-19 23:29:50 +08:00
|
|
|
* approach.
|
|
|
|
*
|
|
|
|
* @author [Qannaf AL-SAHMI](https://github.com/Qannaf)
|
|
|
|
*/
|
|
|
|
|
2023-07-20 04:53:16 +08:00
|
|
|
#include <cassert> /// for assert
|
|
|
|
#include <cstdlib> /// for std::rand
|
2022-01-19 23:29:50 +08:00
|
|
|
#include <iostream> /// for IO operations
|
|
|
|
#include <vector> /// for std::vector
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @namespace math
|
|
|
|
* @brief Mathematical algorithms
|
|
|
|
*/
|
|
|
|
namespace math {
|
|
|
|
|
2023-07-20 04:53:16 +08:00
|
|
|
/**
|
|
|
|
* @brief structure of points containing two numbers, x and y, such that 0 ≤ x ≤
|
|
|
|
* 1 and 0 ≤ y ≤ 1.
|
|
|
|
*/
|
|
|
|
using Point = struct {
|
2022-01-19 23:29:50 +08:00
|
|
|
double x;
|
|
|
|
double y;
|
2023-07-20 04:53:16 +08:00
|
|
|
};
|
2022-01-19 23:29:50 +08:00
|
|
|
|
2023-07-20 04:53:16 +08:00
|
|
|
/**
|
|
|
|
* @brief This function uses the points in a given vector 'pts' (drawn at
|
|
|
|
* random) to return an approximation of the number π.
|
|
|
|
* @param pts Each item of pts contains a point. A point is represented by the
|
|
|
|
* point structure (coded above).
|
|
|
|
* @return an estimate of the number π.
|
|
|
|
*/
|
|
|
|
double approximate_pi(const std::vector<Point> &pts) {
|
|
|
|
double count = 0; // Points in circle
|
|
|
|
for (Point p : pts) {
|
|
|
|
if ((p.x * p.x) + (p.y * p.y) <= 1) {
|
|
|
|
count++;
|
2022-01-19 23:29:50 +08:00
|
|
|
}
|
|
|
|
}
|
2023-07-20 04:53:16 +08:00
|
|
|
return 4.0 * count / static_cast<double>(pts.size());
|
|
|
|
}
|
2022-01-19 23:29:50 +08:00
|
|
|
} // namespace math
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Self-test implementations
|
|
|
|
* @returns void
|
|
|
|
*/
|
2023-07-20 04:53:16 +08:00
|
|
|
static void tests() {
|
|
|
|
std::vector<math::Point> rands;
|
2022-01-19 23:29:50 +08:00
|
|
|
for (std::size_t i = 0; i < 100000; i++) {
|
|
|
|
math::Point p;
|
2023-07-20 04:53:16 +08:00
|
|
|
p.x = rand() / static_cast<double>(RAND_MAX); // 0 <= x <= 1
|
|
|
|
p.y = rand() / static_cast<double>(RAND_MAX); // 0 <= y <= 1
|
2022-01-19 23:29:50 +08:00
|
|
|
rands.push_back(p);
|
|
|
|
}
|
2023-07-20 04:53:16 +08:00
|
|
|
assert(math::approximate_pi(rands) > 3.135);
|
|
|
|
assert(math::approximate_pi(rands) < 3.145);
|
|
|
|
|
|
|
|
std::cout << "All tests have successfully passed!" << std::endl;
|
2022-01-19 23:29:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Main function
|
|
|
|
* @returns 0 on exit
|
|
|
|
*/
|
2023-07-20 04:53:16 +08:00
|
|
|
int main() {
|
|
|
|
tests(); // run self-test implementations
|
2022-01-19 23:29:50 +08:00
|
|
|
return 0;
|
|
|
|
}
|