推导说明   

为便于公式推导,本文仅以球心在坐标原点时进行推导,需要说明的是球心不在原点时,只需把已知条件的坐标减去球心坐标,等完成求解后再加上球心坐标即可!

推导:

在前面博客,已经给出了三点画圆的代码,观察函数generateCircle可知,若要画圆,我们只需要知道圆的圆心,圆所在平面的x方向向量以及y方向向量。

接下来,我们把工作重点放到求上述三个量上面,假定球为球心在坐标原点的单位球R=1,我们需要在球上 rs=[0.75,-0.17,0.64] 的位置画一个 15° 的圆,则

由已知可知,该圆的半径为: r=R*sin(15/180*pi);

该圆所在平面的方向向量为:cc=rs=[0.75,-0.17,0.64]

那么,过原点且与该圆所在平面平行的平面方程为:rs(1)*x+rs(2)*y+rs(3)*z=0

那么,在该平面随便取一个向量作为x轴的方向向量,如:nx=[1,1,-(rs(1)+rs(2))/rs(3)]

那么,根据叉乘规则,y轴的方向向量为:ny=cross(rs,nx);

此外,我们知道,圆心所在的位置在rs与圆心连线的直线上,根据球的参数方程可得,圆心所在的球面半径为:rr=R*cos(15/180*pi)

绘制球面的代码:

function drawsphere(a,b,c,R)
%% 绘制球面
% 以(a,b,c)为球心,R为半径

    % 生成数据
    [x,y,z] = sphere(20);

    % 调整半径
    x = R*x; 
    y = R*y;
    z = R*z;

    % 调整球心
    x = x+a;
    y = y+b;
    z = z+c;

    % 使用mesh绘制
%     figure;
%     axis equal;
    mesh(x,y,z);

    % 使用surf绘制
%     figure;
%     axis equal;
%     surf(x,y,z);
end

生成圆上点的代码

%% 生成圆上的点
% cp 圆心,R 圆半径,a圆平面的x轴方向向量,b圆平面的y轴方向向量,num 插值点个数
function xyz=generateCircle(cp,R,a,b,num)
    xyz=[];
    i=linspace(0,2*pi,num);
    x=cp(1) + R*cos(i)*a(1) +R*sin(i)*b(1);
    y=cp(2) + R*cos(i)*a(2) +R*sin(i)*b(2);
    z=cp(3) + R*cos(i)*a(3) +R*sin(i)*b(3);
    xyz=[x',y',z'];
end

直角坐标到球面坐标的转换

function xyz=sp2dikaer(azimuth,elevation,r)
    x = r .* cos(elevation) .* cos(azimuth);
    y = r .* cos(elevation) .* sin(azimuth);
    z = r .* sin(elevation);
    xyz=[x,y,z];
end

参考自matlab官网

测试代码:

clc;clear all;close all;
rs=[0.75,-0.17,0.64];
a=0;
b=0;
c=0;
R=1;
flag=0;
drawsphere(a,b,c,R);
hold on;
plot3(rs(1),rs(2),rs(3),'rp');

%% 圆的半径
r=R*sin(15/180*pi);
th=0:0.01:2*pi;
azimuth=atan2(rs(2),rs(1));
elevation=atan2(rs(3),sqrt(rs(2)^2+rs(1)^1));
% xyz=sp2dikaer(azimuth,elevation,R)

nx=[1,1,-(rs(1)+rs(2))/rs(3)];
ny=cross(rs,nx);
nx=nx/norm(nx);
ny=ny/norm(ny);
rs1=sp2dikaer(atan2(rs(2),rs(1)),atan2(rs(3),sqrt(sum(rs(1:2).^2))),R*cos(15/180*pi));
xyz=generateCircle(rs1,r,nx,ny,100);
plot3(xyz(:,1),xyz(:,2),xyz(:,3),'r', 'LineWidth',2);