• 根据面积均分球面 & c++示意代码


    参考文献:Malkin Z. A new method to subdivide a spherical surface into equal-area cells[J]. arXiv preprint arXiv:1612.03467, 2016.

    一种根据面积均分球面的方法是根据经纬度来分,即将直角坐标 ( x , y , z ) (x,y,z) (x,y,z)转换成极坐标 ( θ , ϕ , r ) (\theta, \phi, r) (θ,ϕ,r),然后将 θ \theta θ ϕ \phi ϕ均分。均分 ϕ \phi ϕ(经度)确实能让分出来的小块面积相等,但 θ \theta θ(维度)分块后是不均匀的,如下图
    在这里插入图片描述
    网上找到一篇博客介绍了两种均分方法:正多面体(柏拉图体regular polyhedron)、Hierarchical Equal Area isoLatitude Pixelization(这个方法甚至专门有个c++库
    在这里插入图片描述在这里插入图片描述
    但是这两种方法感觉在原理和实现上都很复杂,于是经过一番搜寻找到了论文16年的一篇论文《A new method to subdivide a spherical surface into equal-area cells》

    方法

    这篇论文的方法跟开头介绍的错误分法都是从经纬度( θ \theta θ ϕ \phi ϕ)下手,但它并不是均分。主要步骤分两步:

    • 首先将纬度划分成 x x x
      • 相邻块之间的宽度不完全一样,但是大致相同,而且整个划分关于赤道对称
    • 将每一个纬度带 i ( i = 1 , 2 , . . . , x ) i(i=1,2,...,x) i(i=1,2,...,x),均分成 y i y_i yi块。
      • 不同纬度带划分的块数不一样。纬度越高块数越少,赤道附近的块接近正方形
      • 同样关于赤道对称
      • 要求划分完成后每一块的面积相等

    划分的方法论文中并没有给,只给了将球面均分成46、130、406块的划分方式
    在这里插入图片描述
    在这里插入图片描述
    由于我手边工作的需要,这些划分数量太多了,我就自己手推了 x = 4 x=4 x=4 x = 5 x=5 x=5的划分方式,而且我并没有要求“赤道附近的块接近正方形”

    x = 4 x=4 x=4的划分方式

    • 因为关于赤道对称,上半球有两个纬度带,最高纬度带对应角度为 θ \theta θ,第二个为 π / 2 − θ \pi/2-\theta π/2θ
    • 最高纬度带的面积可以用球冠面积公式得到
      S 1 = 2 π r h = 2 π r ( 1 − c o s θ ) r = 2 π r 2 ( 1 − c o s θ ) S_1=2\pi rh=2\pi r (1-cos\theta)r=2\pi r^2 (1-cos\theta) S1=2πrh=2πr(1cosθ)r=2πr2(1cosθ)
    • 根据半球面积可以计算得到第二个纬度带的面积
      S 2 = S 半 球 − S 1 = 2 π r 2 − 2 π r 2 ( 1 − c o s θ ) = 2 π r 2 c o s θ S_2 = S_{半球}-S_1 = 2\pi r^2 - 2\pi r^2 (1-cos\theta) =2 \pi r^2 cos\theta S2=SS1=2πr22πr2(1cosθ)=2πr2cosθ
    • 假设第一个纬度带分 m m m块,第二个纬度带分 n n n块,则要求
      2 π r 2 ( 1 − c o s θ ) m = 2 π r 2 c o s θ n ( 1 − c o s θ ) c o s θ = m n \frac{2\pi r^2 (1-cos\theta)}{m}=\frac{2\pi r^2 cos\theta }{n}\\ \frac{(1-cos\theta)}{cos\theta }=\frac{m}{n} m2πr2(1cosθ)=n2πr2cosθcosθ(1cosθ)=nm
    • (可选)如果想要让赤道附近的块尽量接近正方形,我们可以大致估算一下 n n n的范围
      • 赤道附近的每个块的高度(纬度带的宽度)为 h = r ∗ ( π / 2 − θ ) h=r*(\pi/2-\theta) h=r(π/2θ)
      • 赤道附近的纬度带分 n n n块,则我们想要上下边的长度尽量接近高度,即
        l 上 = 2 π r s i n θ n ≈ r ∗ ( π / 2 − θ ) → n = 2 π s i n θ π / 2 − θ l 下 = 2 π r n ≈ r ∗ ( π / 2 − θ ) → n = 2 π π / 2 − θ l_{上}=\frac{2\pi r sin\theta}{n}\approx r*(\pi/2-\theta)\rightarrow n = \frac{2\pi sin\theta}{\pi/2-\theta}\\ l_{下}=\frac{2\pi r }{n} \approx r*(\pi/2-\theta)\rightarrow n = \frac{2\pi}{\pi/2-\theta} l=n2πrsinθr(π/2θ)n=π/2θ2πsinθl=n2πrr(π/2θ)n=π/2θ2π
      • 将这两条线画出来可以看到 n n n越大,越接近正方形在这里插入图片描述
    • (可选)如果想要纬度带宽度大致相同,即要求 θ ≈ π / 4 \theta\approx\pi/4 θπ/4,带入之前的公式得
      m n = ( 1 − c o s θ ) c o s θ ≈ ( 1 − c o s π / 4 ) c o s π / 4 ≈ 0.414 \frac{m}{n}=\frac{(1-cos\theta)}{cos\theta }\approx \frac{(1-cos\pi/4)}{cos\pi/4 }\approx0.414 nm=cosθ(1cosθ)cosπ/4(1cosπ/4)0.414
    • 接下来只要按照想要的划分方式给 m m m n n n赋整数,就能求解 θ \theta θ例如
      • m = 2 , n = 3 , θ = 53.130 ° m=2, n=3, \theta=53.130\degree m=2,n=3,θ=53.130°
      • m = 3 , n = 4 , θ = 55.150 ° m=3, n=4, \theta=55.150\degree m=3,n=4,θ=55.150°
      • m = 3 , n = 5 , θ = 51.318 ° m=3, n=5, \theta=51.318\degree m=3,n=5,θ=51.318°
      • m = 4 , n = 5 , θ = 56.217 ° m=4, n=5, \theta=56.217\degree m=4,n=5,θ=56.217°
      • m = 3 , n = 6 , θ = 48.190 ° m=3, n=6, \theta=48.190\degree m=3,n=6,θ=48.190°
      • m = 3 , n = 7 , θ = 45.573 ° m=3, n=7, \theta=45.573\degree m=3,n=7,θ=45.573°
    Total CellCellsLatitudinalLongitudinal
    102
    3
    3
    2
    0-53.13°
    53.13°-90°
    90°-126.87°
    126.87°-180°
    180°
    120°
    120°
    180°
    143
    4
    4
    3
    0-55.15°
    55.15°-90°
    90°-124.85°
    124.85°-180°
    120°
    90°
    90°
    120°
    184
    5
    5
    4
    0-56.22°
    56.22°-90°
    90°-123.78°
    123.78°-180°
    90°
    72°
    72°
    90°
    203
    7
    7
    3
    0-45.573° / 0 - 0.7954
    45.573°-90° / 0.7954 - 1.5708
    90°-134.427° / 1.5708 - 2.3462
    134.427°-180° / 2.3462 - 3.1416
    120° / 2.0944
    51.43° / 0.8976
    51.43° / 0.8976
    120° / 2.0944

    x = 5 x=5 x=5的划分方式

    • 虽然 x = 5 x=5 x=5为奇数,但是因为关于赤道对称,这里算上半球有2.5个纬度带,即第三个纬度带只算一半。三个纬度带从高到底对应的角度分别为 θ , ϕ , π / 2 − θ − ϕ \theta, \phi, \pi/2-\theta-\phi θ,ϕ,π/2θϕ
    • 最高纬度带的面积可以用球冠面积公式得到
      S 1 = 2 π r h = 2 π r ( 1 − c o s θ ) r = 2 π r 2 ( 1 − c o s θ ) S_1=2\pi rh=2\pi r (1-cos\theta)r=2\pi r^2 (1-cos\theta) S1=2πrh=2πr(1cosθ)r=2πr2(1cosθ)
    • 根据球冠面积同样可以计算得到第二个纬度带的面积
      S 2 = 2 π r 2 ( 1 − c o s ( θ + ϕ ) ) − 2 π r 2 ( 1 − c o s θ ) = 2 π r 2 ( c o s θ − c o s ( θ + ϕ ) ) S_2 = 2\pi r^2(1-cos(\theta+\phi)) - 2\pi r^2 (1-cos\theta) =2 \pi r^2 ( cos\theta - cos(\theta+\phi)) S2=2πr2(1cos(θ+ϕ))2πr2(1cosθ)=2πr2(cosθcos(θ+ϕ))
    • 根据半球面积可以计算得到最后半个纬度带的面积
      S 2.5 = S 半 球 − ( S 1 + S 2 ) = 2 π r 2 − 2 π r 2 ( 1 − c o s ( θ + ϕ ) ) = 2 π r 2 c o s ( θ + ϕ ) S_{2.5} = S_{半球}-(S_1+S_2) = 2\pi r^2 - 2\pi r^2(1-cos(\theta+\phi)) =2 \pi r^2 cos(\theta+\phi) S2.5=S(S1+S2)=2πr22πr2(1cos(θ+ϕ))=2πr2cos(θ+ϕ)
    • 假设第一个纬度带分 m m m块,第二个纬度带分 n n n块,最后半个纬度带分 o o o块,则要求
      2 π r 2 ( 1 − c o s θ ) m = 2 π r 2 ( c o s θ − c o s ( θ + ϕ ) ) n = 2 ∗ 2 π r 2 c o s ( θ + ϕ ) o 1 − c o s θ m = c o s θ − c o s ( θ + ϕ ) n = 2 c o s ( θ + ϕ ) o \frac{2\pi r^2 (1-cos\theta)}{m}=\frac{2 \pi r^2 ( cos\theta - cos(\theta+\phi))}{n}=2*\frac{2 \pi r^2 cos(\theta+\phi)}{o}\\ \frac{1-cos\theta}{m}=\frac{ cos\theta - cos(\theta+\phi)}{n}=\frac{2cos(\theta+\phi)}{o} m2πr2(1cosθ)=n2πr2(cosθcos(θ+ϕ))=2o2πr2cos(θ+ϕ)m1cosθ=ncosθcos(θ+ϕ)=o2cos(θ+ϕ)
    • 接下来只要按照想要的划分方式给 m m m n n n o o o赋整数,就能求解 θ \theta θ ϕ \phi ϕ例如
      • m = 3 , n = 4 , o = 5 , θ = 46.826 ° , ϕ = 27.916 ° m=3, n=4, o=5, \theta=46.826\degree, \phi=27.916\degree m=3,n=4,o=5,θ=46.826°,ϕ=27.916°
      • m = 3 , n = 4 , o = 6 , θ = 45.573 ° , ϕ = 26.969 ° m=3, n=4, o=6, \theta=45.573\degree, \phi=26.969\degree m=3,n=4,o=6,θ=45.573°,ϕ=26.969°
      • m = 4 , n = 5 , o = 6 , θ = 48.190 ° , ϕ = 27.333 ° m=4, n=5, o=6, \theta=48.190\degree, \phi=27.333\degree m=4,n=5,o=6,θ=48.190°,ϕ=27.333°
    Total CellCellsLatitudinalLongitudinal
    193
    4
    5
    4
    3
    0-46.826°
    46.826°-74.743°
    74.743°-105.257°
    105.257°- 133.174°
    133.174°-180°
    120°
    90°
    72°
    90°
    120°
    244
    5
    6
    5
    4
    0-48.190°
    48.190°-75.522°
    75.522°-104.478°
    104.478°- 131.810°
    131.810°-180°
    90°
    72°
    60°
    72°
    90°

    c++代码

    这里以“统计球形点云落在每个块中的数量”为例,划分方式为 x = 4 , m = 3 , n = 7 x=4, m=3, n=7 x=4,m=3,n=7

    /**
     * 统计球形点云落在每个块中的数量
     * @param[in] points n*3的矩阵,存储三维点云
     * @param[out] nums  统计结果,存储顺序北极->南极 & 经度0°->360°
     */
    void ClassifyCentroidVector(const Eigen::MatrixXf &points, vector<int> &nums) {
      direct.resize(20);
      Eigen::Vector4f block_theta(0., 0.7954, 1.5708, 2.3462);
      Eigen::Vector4f block_phi(2.0944, 0.8976, 0.8976, 2.0944);
      Eigen::Vector4i offset(0, 3, 10, 17);
      for (int k = 0; k < points.rows(); k++) {
        //笛卡尔坐标系转换为球坐标系
        const float x = points(k, 0), y = points(k, 1), z = points(k, 2);
        float r = sqrt(x * x + y * y + z * z);
        float theta = acos(z / r);
        float fai = atan2(y, x) + M_PI; // 取值范围0-2π
        
        int coord_theta = -1;
        for(int i = 0; i < block_theta.size(); i++)
          if(theta > block_theta(i)){
            coord_theta = i;
            break;
          }
        assert(coord_theta != -1);
        int coord_phi = int(fai/block_phi(coord_theta));
        direct[offset(coord_theta)+coord_phi] += 1;
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
  • 相关阅读:
    Vue笔记3(计算属性,监视属性,绑定样式)
    vue3嵌套路由keep-alive问题
    YOLOv8优化策略:全新的聚焦式线性注意力模块Focused Linear Attention | ICCV2023
    搭建自己的以图搜图系统(二):深入优化搭建生产级别的图搜系统
    SpringCloud - Spring Cloud Netflix 之 Zuul网关;路由(十一)
    多态的详解
    python实现FINS协议的TCP服务端(篇一)
    py 控制台输入参数
    JUL日志讲解
    国内DNS首选
  • 原文地址:https://blog.csdn.net/OTZ_2333/article/details/125408236