osg 八叉树可视化

2023-11-09 05:12
文章标签 可视化 osg 八叉树

本文主要是介绍osg 八叉树可视化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

什么是八叉树

八叉树算法过程

八叉树的计算原理

八叉树c++实现

使用osg可视化八叉树


什么是八叉树

在描述三维场景的过程中常常用到一种名为八叉树的数据结构。描述三维空间的八叉树和描述二维空间的四叉树有相似之处,二维空间中正方形可以被分为四个相同形状的正方形,而三维空间中正方体可以被分为八个形状相同的正方体。

八叉树的每个结点表示一个正方体的体积元素,每一个结点有八个子节点,这种用于描述三维空间的树装结构叫做八叉树。

八叉树算法过程

八叉树的计算原理

1. 设定最大递归深度

2. 找出场景的最大尺寸,并以此尺寸建立第一个立方体

3. 依序将单位元元素丢入能包含且没有子节点的立方体

4. 若没达到最大递归深度,就进行细分八等份,再讲该立方体所装的单位元元素全部分组给八个子立方体

5. 若发现子立方体所分配到的单位元元素数量不为零且跟父立方体一样,则该子立方体停止细分

6. 重复3,知道到达最大递归深度

八叉树c++实现

#include <iostream>    using namespace std;    
//定义八叉树节点类    
template<class T>    
struct OctreeNode    
{    T data; //节点数据    T xmin,xmax; //节点坐标,即六面体个顶点的坐标    T ymin,ymax;    T zmin,zmax;    OctreeNode <T>*top_left_front,*top_left_back; //该节点的个子结点    OctreeNode <T>*top_right_front,*top_right_back;    OctreeNode <T>*bottom_left_front,*bottom_left_back;    OctreeNode <T>*bottom_right_front,*bottom_right_back;    OctreeNode //节点类    (T nodeValue = T(),    T xminValue = T(),T xmaxValue = T(),    T yminValue = T(),T ymaxValue = T(),    T zminValue = T(),T zmaxValue = T(),    OctreeNode<T>*top_left_front_Node = NULL,    OctreeNode<T>*top_left_back_Node = NULL,    OctreeNode<T>*top_right_front_Node = NULL,    OctreeNode<T>*top_right_back_Node = NULL,    OctreeNode<T>*bottom_left_front_Node = NULL,    OctreeNode<T>*bottom_left_back_Node = NULL,    OctreeNode<T>*bottom_right_front_Node = NULL,    OctreeNode<T>*bottom_right_back_Node = NULL )    :data(nodeValue),    xmin(xminValue),xmax(xmaxValue),    ymin(yminValue),ymax(ymaxValue),    zmin(zminValue),zmax(zmaxValue),    top_left_front(top_left_front_Node),    top_left_back(top_left_back_Node),    top_right_front(top_right_front_Node),    top_right_back(top_right_back_Node),    bottom_left_front(bottom_left_front_Node),    bottom_left_back(bottom_left_back_Node),    bottom_right_front(bottom_right_front_Node),    bottom_right_back(bottom_right_back_Node){}    
};    
//创建八叉树    
template <class T>    
void createOctree(OctreeNode<T> * &root,int maxdepth,double xmin,double xmax,double ymin,double ymax,double zmin,double zmax)    
{    //cout<<"处理中,请稍候……"<<endl;    maxdepth=maxdepth-1; //每递归一次就将最大递归深度-1    if(maxdepth>=0)    {    root=new OctreeNode<T>();    cout<<"请输入节点值:";    //root->data =9;//为节点赋值,可以存储节点信息,如物体可见性。由于是简单实现八叉树功能,简单赋值为9。    cin>>root->data;  //为节点赋值    root->xmin=xmin; //为节点坐标赋值    root->xmax=xmax;    root->ymin=ymin;    root->ymax=ymax;    root->zmin=zmin;    root->zmax=zmax;    double xm=(xmax-xmin)/2;//计算节点个维度上的半边长    double ym=(ymax-ymin)/2;    double zm=(ymax-ymin)/2;    //递归创建子树,根据每一个节点所处(是几号节点)的位置决定其子结点的坐标。    createOctree(root->top_left_front,maxdepth,xmin,xmax-xm,ymax-ym,ymax,zmax-zm,zmax);    createOctree(root->top_left_back,maxdepth,xmin,xmax-xm,ymin,ymax-ym,zmax-zm,zmax);    createOctree(root->top_right_front,maxdepth,xmax-xm,xmax,ymax-ym,ymax,zmax-zm,zmax);    createOctree(root->top_right_back,maxdepth,xmax-xm,xmax,ymin,ymax-ym,zmax-zm,zmax);    createOctree(root->bottom_left_front,maxdepth,xmin,xmax-xm,ymax-ym,ymax,zmin,zmax-zm);    createOctree(root->bottom_left_back,maxdepth,xmin,xmax-xm,ymin,ymax-ym,zmin,zmax-zm);    createOctree(root->bottom_right_front,maxdepth,xmax-xm,xmax,ymax-ym,ymax,zmin,zmax-zm);    createOctree(root->bottom_right_back,maxdepth,xmax-xm,xmax,ymin,ymax-ym,zmin,zmax-zm);    }    
}    
int i=1;    
//先序遍历八叉树    
template <class T>    
void preOrder( OctreeNode<T> * & p)    
{    if(p)    {    cout<<i<<".当前节点的值为:"<<p->data<<"\n坐标为:";    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    i+=1;    cout<<endl;    preOrder(p->top_left_front);    preOrder(p->top_left_back);    preOrder(p->top_right_front);    preOrder(p->top_right_back);    preOrder(p->bottom_left_front);    preOrder(p->bottom_left_back);    preOrder(p->bottom_right_front);    preOrder(p->bottom_right_back);    cout<<endl;    }    
}    
//求八叉树的深度    
template<class T>    
int depth(OctreeNode<T> *& p)    
{    if(p == NULL)    return -1;    int h =depth(p->top_left_front);    return h+1;    
}    
//计算单位长度,为查找点做准备    
int cal(int num)    
{    int result=1;    if(1==num)    result=1;    else    {    for(int i=1;i<num;i++)    result=2*result;    }    return result;    
}    
//查找点    
int maxdepth=0;    
int times=0;    
static double xmin=0,xmax=0,ymin=0,ymax=0,zmin=0,zmax=0;    
int tmaxdepth=0;    
double txm=1,tym=1,tzm=1;    
template<class T>    
void find(OctreeNode<T> *& p,double x,double y,double z)    
{    double xm=(p->xmax-p->xmin)/2;    double ym=(p->ymax-p->ymin)/2;    double zm=(p->ymax-p->ymin)/2;    times++;    if(x>xmax || x<xmin|| y>ymax || y<ymin || z>zmax || z<zmin)    {    cout<<"该点不在场景中!"<<endl;    return;    }    if(x<=p->xmin+txm&& x>=p->xmax-txm && y<=p->ymin+tym &&y>=p->ymax-tym && z<=p->zmin+tzm &&z>=p->zmax-tzm )    {    cout<<endl<<"找到该点!"<<"该点位于"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<"节点内!"<<endl;    cout<<"共经过"<<times<<"次递归!"<<endl;    }    else if(x<(p->xmax-xm) && y<(p->ymax-ym) &&z<(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->bottom_left_back,x,y,z);    }    else if(x<(p->xmax-xm) && y<(p->ymax-ym) &&z>(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->top_left_back,x,y,z);    }    else if(x>(p->xmax-xm) && y<(p->ymax-ym) &&z<(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->bottom_right_back,x,y,z);    }    else if(x>(p->xmax-xm) && y<(p->ymax-ym) &&z>(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->top_right_back,x,y,z);    }    else if(x<(p->xmax-xm) && y>(p->ymax-ym) &&z<(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->bottom_left_front,x,y,z);    }    else if(x<(p->xmax-xm) && y>(p->ymax-ym) &&z>(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->top_left_front,x,y,z);    }    else if(x>(p->xmax-xm) && y>(p->ymax-ym) &&z<(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->bottom_right_front,x,y,z);    }    else if(x>(p->xmax-xm) && y>(p->ymax-ym) &&z>(p->zmax-zm))    {    cout<<"当前经过节点坐标:"<<endl;    cout<<"xmin: "<<p->xmin<<" xmax: "<<p->xmax;    cout<<"ymin: "<<p->ymin<<" ymax: "<<p->ymax;    cout<<"zmin: "<<p->zmin<<" zmax: "<<p->zmax;    cout<<endl;    find(p->top_right_front,x,y,z);    }    
}    
//main函数    
int main ()    
{    OctreeNode<double> *rootNode = NULL;    int choiced = 0;    while(true)    {    system("cls");    cout<<"请选择操作:\n";    cout<<"1.创建八叉树 2.先序遍历八叉树\n";    cout<<"3.查看树深度 4.查找节点   \n";    cout<<"0.退出\n\n";    cin>>choiced;    if(choiced == 0)    return 0;    else if(choiced == 1)    {    system("cls");    cout<<"请输入最大递归深度:"<<endl;    cin>>maxdepth;    cout<<"请输入外包盒坐标,顺序如下:xmin,xmax,ymin,ymax,zmin,zmax"<<endl;    cin>>xmin>>xmax>>ymin>>ymax>>zmin>>zmax;    if(maxdepth>=0|| xmax>xmin || ymax>ymin || zmax>zmin || xmin>0 || ymin>0||zmin>0)    {    tmaxdepth=cal(maxdepth);    txm=(xmax-xmin)/tmaxdepth;    tym=(ymax-ymin)/tmaxdepth;    tzm=(zmax-zmin)/tmaxdepth;    createOctree(rootNode,maxdepth,xmin,xmax,ymin,ymax,zmin,zmax);    }    else    {    cout<<"输入错误!";    return 0;    }    }    else if(choiced == 2)    {    system("cls");    cout<<"先序遍历八叉树结果:/n";    i=1;    preOrder(rootNode);    cout<<endl;    system("pause");    }    else if(choiced == 3)    {    system("cls");    int dep =depth(rootNode);    cout<<"此八叉树的深度为"<<dep+1<<endl;    system("pause");    }    else if(choiced == 4)    {    system("cls");    cout<<"请输入您希望查找的点的坐标,顺序如下:x,y,z\n";    double x,y,z;    cin>>x>>y>>z;    times=0;    cout<<endl<<"开始搜寻该点……"<<endl;    find(rootNode,x,y,z);    system("pause");    }    else    {    system("cls");    cout<<"\n\n错误选择!\n";    system("pause");    }    }    
}  

使用osg可视化八叉树

 main函数

// osgPro227.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。#include <osg/Group>
#include <osgDB/ReadFile>
#include <osgUtil/PrintVisitor>
#include <osgViewer/ViewerEventHandlers>
#include <osgViewer/Viewer>
#include <osgViewer/ViewerEventHandlers> //事件监听
#include <osgGA/StateSetManipulator> //事件响应类,对渲染状态进行控制#include "OctreeBuilder.h"#pragma comment(lib, "OpenThreadsd.lib")
#pragma comment(lib, "osgd.lib")
#pragma comment(lib, "osgDBd.lib")
#pragma comment(lib, "osgUtild.lib")
#pragma comment(lib, "osgGAd.lib")
#pragma comment(lib, "osgViewerd.lib")
#pragma comment(lib, "osgTextd.lib")float randomValue(float min, float max)
{return (min + (float)rand() / (RAND_MAX + 1.0f) * (max - min));
}osg::Vec3 randomVector(float min, float max)
{return osg::Vec3(randomValue(min, max),randomValue(min, max),randomValue(min, max));
}
class PrintNameVisitor : public osgUtil::PrintVisitor
{
public:PrintNameVisitor(std::ostream& out) : osgUtil::PrintVisitor(out) {}void apply(osg::Node& node){if (!node.getName().empty()){output() << node.getName() << std::endl;enter();traverse(node);leave();}else osgUtil::PrintVisitor::apply(node);}
};int main(int argc, char** argv)
{osg::BoundingBox globalBound;std::vector<OctreeBuilder::ElementInfo> globalElements;for (unsigned int i = 0; i < 5000; ++i){osg::Vec3 pos = randomVector(-500.0f, 500.0f);float radius = randomValue(0.5f, 20.0f);std::stringstream ss; ss << "Ball-" << i + 1;osg::Vec3 min = pos - osg::Vec3(radius, radius, radius);osg::Vec3 max = pos + osg::Vec3(radius, radius, radius);osg::BoundingBox region(min, max);globalBound.expandBy(region);globalElements.push_back(OctreeBuilder::ElementInfo(ss.str(), region));}OctreeBuilder octree;osg::ref_ptr<osg::Group> root = octree.build(0, globalBound, globalElements);std::ofstream out("octree_output.txt");PrintNameVisitor printer(out);root->accept(printer);osg::ref_ptr <osgViewer::Viewer> viewer = new osgViewer::Viewer;viewer->setSceneData(root.get());viewer->addEventHandler(new osgGA::StateSetManipulator(viewer->getCamera()->getOrCreateStateSet()));viewer->addEventHandler(new osgViewer::StatsHandler());//实现状态信息统计viewer->addEventHandler(new osgViewer::WindowSizeHandler());return viewer->run();
}

OctreeBuilder.h 头文件

#ifndef H_COOKBOOK_CH8_OCTREEBUILDER
#define H_COOKBOOK_CH8_OCTREEBUILDER#include <osg/Geode>
#include <osg/LOD>class OctreeBuilder
{
public:OctreeBuilder() : _maxChildNumber(16), _maxTreeDepth(8), _maxLevel(0) {}int getMaxLevel() const { return _maxLevel; }void setMaxChildNumber( int max ) { _maxChildNumber = max; }int getMaxChildNumber() const { return _maxChildNumber; }void setMaxTreeDepth( int max ) { _maxTreeDepth = max; }int getMaxTreeDepth() const { return _maxTreeDepth; }typedef std::pair<std::string, osg::BoundingBox> ElementInfo;osg::Group* build( int depth, const osg::BoundingBox& total,std::vector<ElementInfo>& elements );protected:osg::LOD* createNewLevel( int level, const osg::Vec3& center, float radius );osg::Node* createElement( const std::string& id, const osg::Vec3& center, float radius );osg::Geode* createBoxForDebug( const osg::Vec3& max, const osg::Vec3& min );int _maxChildNumber;int _maxTreeDepth;int _maxLevel;
};#endif

OctreeBuilder.cpp 

#include <windows.h>
#include <iostream>
#include <fstream>
#include <sstream>
using namespace std;#include <osg/ShapeDrawable>
#include <osg/Geometry>
#include <osg/PolygonMode>
#include "OctreeBuilder.h"osg::Group* OctreeBuilder::build( int depth, const osg::BoundingBox& total,std::vector<ElementInfo>& elements )
{int s[3];  // axis sides (0 or 1)//存放当前包围盒的最大、中间、最小点,以为划分八叉树做准备osg::Vec3 extentSet[3] = {total._min,(total._max + total._min) * 0.5f,total._max};std::vector<ElementInfo> childData;//遍历父结点的所有孩子,让包含在当前盒子里的,不完全包含但是中点在盒子里的,都压入到当前盒子的子结点for ( unsigned int i=0; i<elements.size(); ++i ){const ElementInfo& obj = elements[i];if ( total.contains(obj.second._min) && total.contains(obj.second._max) )childData.push_back( obj );else if ( total.intersects(obj.second) ){osg::Vec3 center = (obj.second._max + obj.second._min) * 0.5f;if (total.contains(center)){childData.push_back(obj);}}}//如果当前结点的孩子数量已经达标,或者层数已经达标,则认为是叶结点bool isLeafNode = false;if ((int)childData.size() <= _maxChildNumber || depth > _maxTreeDepth){isLeafNode = true;}//当前八叉树根osg::ref_ptr<osg::Group> group = new osg::Group;//如果不是叶结点,继续分,把空间一分为八if ( !isLeafNode ){osg::ref_ptr<osg::Group> childNodes[8];//空间一分为八2*2*2for ( s[0]=0; s[0]<2; ++s[0] ) //x{for ( s[1]=0; s[1]<2; ++s[1] ) //y{for ( s[2]=0; s[2]<2; ++s[2] ) //z{// Calculate the child extent//extentSet 0是最小,1是中间,2是最大//下面这个小算法有点磨人,分别求出min和max的x, y, z自己好好推几个试试osg::Vec3 min, max;for ( int a=0; a<3; ++a ){min[a] = (extentSet[s[a] + 0])[a];max[a] = (extentSet[s[a] + 1])[a];}//这么求id是为了确保唯一性int id = s[0] + (2 * s[1]) + (4 * s[2]);childNodes[id] = build( depth+1, osg::BoundingBox(min, max), childData );}}}//八个子结点构建完毕后,加入到根结点当中for ( unsigned int i=0; i<8; ++i ){if ( childNodes[i] && childNodes[i]->getNumChildren() )group->addChild( childNodes[i] );}}else //找到叶结点,递归就结束了{for ( unsigned int i=0; i<childData.size(); ++i ){const ElementInfo& obj = childData[i];osg::Vec3 center = (obj.second._max + obj.second._min) * 0.5;float radius = (obj.second._max - obj.second._min).length() * 0.5f;//创建一个球group->addChild( createElement(obj.first, center, radius) );}}osg::Vec3 center = (total._max + total._min) * 0.5;float radius = (total._max - total._min).length() * 0.5f;//最后创建一个LOD,离的远了显示调试盒子,离的近了显示分的组osg::LOD* level = createNewLevel( depth, center, radius );level->insertChild( 0, createBoxForDebug(total._max, total._min) );  // For debug uselevel->insertChild( 1, group.get() );return level;
}osg::LOD* OctreeBuilder::createNewLevel( int level, const osg::Vec3& center, float radius )
{osg::ref_ptr<osg::LOD> lod = new osg::LOD;lod->setCenterMode( osg::LOD::USER_DEFINED_CENTER );lod->setCenter( center );lod->setRadius( radius );lod->setRange( 0, radius * 5.0f, FLT_MAX );lod->setRange( 1, 0.0f, radius * 5.0f );if ( _maxLevel<level ) _maxLevel = level;return lod.release();
}osg::Node* OctreeBuilder::createElement( const std::string& id, const osg::Vec3& center, float radius )
{osg::ref_ptr<osg::Geode> geode = new osg::Geode;geode->addDrawable( new osg::ShapeDrawable(new osg::Sphere(center, radius)) );geode->setName( id );return geode.release();
}osg::Geode* OctreeBuilder::createBoxForDebug( const osg::Vec3& max, const osg::Vec3& min )
{osg::Vec3 dir = max - min;osg::ref_ptr<osg::Vec3Array> va = new osg::Vec3Array(10);(*va)[0] = min + osg::Vec3(0.0f, 0.0f, 0.0f);(*va)[1] = min + osg::Vec3(0.0f, 0.0f, dir[2]);(*va)[2] = min + osg::Vec3(dir[0], 0.0f, 0.0f);(*va)[3] = min + osg::Vec3(dir[0], 0.0f, dir[2]);(*va)[4] = min + osg::Vec3(dir[0], dir[1], 0.0f);(*va)[5] = min + osg::Vec3(dir[0], dir[1], dir[2]);(*va)[6] = min + osg::Vec3(0.0f, dir[1], 0.0f);(*va)[7] = min + osg::Vec3(0.0f, dir[1], dir[2]);(*va)[8] = min + osg::Vec3(0.0f, 0.0f, 0.0f);(*va)[9] = min + osg::Vec3(0.0f, 0.0f, dir[2]);osg::ref_ptr<osg::Geometry> geom = new osg::Geometry;geom->setVertexArray( va.get() );geom->addPrimitiveSet( new osg::DrawArrays(GL_QUAD_STRIP, 0, 10) );osg::ref_ptr<osg::Geode> geode = new osg::Geode;geode->addDrawable( geom.get() );geode->getOrCreateStateSet()->setAttribute(new osg::PolygonMode(osg::PolygonMode::FRONT_AND_BACK, osg::PolygonMode::LINE) );geode->getOrCreateStateSet()->setMode( GL_LIGHTING, osg::StateAttribute::OFF );return geode.release();
}

这篇关于osg 八叉树可视化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/374153

相关文章

Python:豆瓣电影商业数据分析-爬取全数据【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】

**爬取豆瓣电影信息,分析近年电影行业的发展情况** 本文是完整的数据分析展现,代码有完整版,包含豆瓣电影爬取的具体方式【附带爬虫豆瓣,数据处理过程,数据分析,可视化,以及完整PPT报告】   最近MBA在学习《商业数据分析》,大实训作业给了数据要进行数据分析,所以先拿豆瓣电影练练手,网络上爬取豆瓣电影TOP250较多,但对于豆瓣电影全数据的爬取教程很少,所以我自己做一版。 目

基于SSM+Vue+MySQL的可视化高校公寓管理系统

系统展示 管理员界面 宿管界面 学生界面 系统背景   当前社会各行业领域竞争压力非常大,随着当前时代的信息化,科学化发展,让社会各行业领域都争相使用新的信息技术,对行业内的各种相关数据进行科学化,规范化管理。这样的大环境让那些止步不前,不接受信息改革带来的信息技术的企业随时面临被淘汰,被取代的风险。所以当今,各个行业领域,不管是传统的教育行业

「大数据分析」图形可视化,如何选择大数据可视化图形?

​图形可视化技术,在大数据分析中,是一个非常重要的关键部分。我们前期通过数据获取,数据处理,数据分析,得出结果,这些过程都是比较抽象的。如果是非数据分析专业人员,很难清楚我们这些工作,到底做了些什么事情。即使是专业人员,在不清楚项目,不了解业务规则,不熟悉技术细节的情况下。要搞清楚我们的大数据分析,这一系列过程,也是比较困难的。 我们在数据处理和分析完成后,一般来说,都需要形成结论报告。怎样让大

11Python的Pandas:可视化

Pandas本身并没有直接的可视化功能,但它与其他Python库(如Matplotlib和Seaborn)无缝集成,允许你快速创建各种图表和可视化。这里是一些使用Pandas数据进行可视化的常见方法: 1. 使用Matplotlib Pandas中的plot()方法实际上是基于Matplotlib的,你可以使用它来绘制各种基本图表,例如折线图、柱状图、散点图等。 import pandas

【全网最全】2024年数学建模国赛A题30页完整建模文档+17页成品论文+保奖matla代码+可视化图表等(后续会更新)

您的点赞收藏是我继续更新的最大动力! 一定要点击如下的卡片,那是获取资料的入口! 【全网最全】2024年数学建模国赛A题30页完整建模文档+17页成品论文+保奖matla代码+可视化图表等(后续会更新)「首先来看看目前已有的资料,还会不断更新哦~一次购买,后续不会再被收费哦,保证是全网最全资源,随着后续内容更新,价格会上涨,越早购买,价格越低,让大家再也不需要到处买断片资料啦~💰💸👋」�

Python利用pyecharts实现数据可视化

小编会持续更新知识笔记,如果感兴趣可以三连支持。闲来无事,水文一篇,不过上手实践一下倒还是挺好玩的,这一块知识说不定以后真可以尝试拿来做数据库的报表显示。         有梦别怕苦,想赢别喊累。 目录 前言 JSON数据格式的转换 pyecharts简介和入门使用 前言       小编我今天闲来无事,打算学习一下py,结果你猜怎么着,竟然看到py可以将数据

【mysql zeppelin】zeppelin 大数据可视化分析工具安装教程精要

Apache Zeppelin是一款大数据分析和可视化工具,可以让数据分析师在一个基于Web页面的笔记本中,使用不同的语言,对不同数据源中的数据进行交互式分析,并对分析结果进行可视化的工具。下面我们主要讲解如何安装和配置的精要部分。 一、zeppelin 安装和配置登录用户 官方网站: https://zeppelin.apache.org/ 下载地址: https://zeppelin

【R语言可视化】R语言画爱心图

数学系也可以很浪漫~~ rm(list=ls())library(grid)heart <- function(lcolor){t=seq(0, 2*pi, by=0.1)x=16*sin(t)^3y=13*cos(t)-5*cos(2*t)-2*cos(3*t)-cos(4*t)a=(x-min(x))/(max(x)-min(x))b=(y-min(y))/(max(y)-min(y))g

【R语言 可视化】R语言画雷达图

1、安装插件 github网址: https://github.com/ricardo-bion/ggradar devtools::install_github("ricardo-bion/ggradar")install.packages("knitr") 2、效果图 3、源代码 rm(list=ls())gc()library(ggradar)mydata<-ma

【python 可视化】python 一行代码画爱心图

一、效果 二、源代码 print'\n'.join([''.join([('LoveDaLin'[(x-y)%8]if((x*0.05)**2+(y*0.1)**2-1)**3-(x*0.05)**2*(y*0.1)**3<=0 else' ')for x in range(-30,30)])for y in range(15,-15,-1)])