使用 Assimp 库加载 3D 模型
前言
要想让自己的 3D 之旅多一点乐趣,肯定得想办法找一些有意思一点的 3D 模型。3D 模型有各种各样的格式,obj的,stl的,fbx的等等不一而足。特别是 obj 格式的 3D 模型,完全是纯文本格式,网络上很多高手都喜欢自己写程序对 obj 格式的模型文件进行解析。我自己收集了一些 3D 模型,有几个 obj 格式的,也有几个 stl 格式的,为我以后的学习做准备。当我需要查看这些模型的时候,我首选是使用 Blender。在我的程序中使用的时候,我首选 Assimp 库。在我之前的随笔中,已经对 Assimp 库做了介绍。见 为什么说使用 Linux 系统学习 OpenGL 更方便。
使用 Assimp 加载 3D 模型
Assimp 的使用是非常简单的,直接参考 Assimp 的文档,依葫芦画瓢即可。这里我只简单介绍一下 Assimp 中的数据组织结构。Assimp 读入模型后,先有一个 aiScene 对象,在这个对象中,有一个根节点,根节点又有子结点,子结点还可能有子结点,形成一个树状的结构。节点的类型是 aiNode。每一个节点包含一个或多个 aiMesh,而 aiMesh 又包含顶点数据和索引数据,索引数据是储存在 aiFace 类型中的,一般来说,我们的每一个 aiFace 都应该是一个三角形。每一个 aiMesh 又有对应的材质信息,因为我这里还没有涉及到光照和贴图,所以暂不考虑材质信息。只需要递归地进行解析,我们就可以很容易获得模型中的所有顶点信息和索引信息。
在这里,我创建了一个 Model 类,同时利用了之前的 Mesh 类,一个 Model 类的对象中可以包含多个 Mesh 类的对象。然后,在 Model 类的 loadModel() 方法中,使用 Assimp 读取模型数据,遍历各 aiNode, 再遍历 aiMesh,将 aiMesh 中的数据拷贝到 Mesh 中,然后调用所有 Mesh 对象的 setup() 方法和 render() 方法,即可完成模型的渲染。Model 类的内容如下:
#ifndef __MODEL_HPP__
#define __MODEL_HPP__
#include "mesh.hpp"
#include <assimp/Importer.hpp>
#include <assimp/postprocess.h>
#include <assimp/scene.h>
class Model
{
private:
std::vector<Mesh> meshes;
public:
void loadModel(std::string filename)
{
Assimp::Importer importer;
const aiScene *scene = importer.ReadFile(filename,
aiProcess_Triangulate |
aiProcess_GenNormals |
aiProcess_FlipUVs);
processNode(scene->mRootNode, scene);
}
void processNode(aiNode *node, const aiScene *scene)
{
for (int i = 0; i < node->mNumMeshes; i++)
{
aiMesh *mesh = scene->mMeshes[node->mMeshes[i]];
Vertex tempVertex;
std::vector<Vertex> tempVertices;
std::vector<GLuint> tempIndices;
//先读取顶点信息
for (int j = 0; j < mesh->mNumVertices; j++)
{
if (mesh->HasPositions())
{
tempVertex.position.x = mesh->mVertices[j].x;
tempVertex.position.y = mesh->mVertices[j].y;
tempVertex.position.z = mesh->mVertices[j].z;
tempVertex.position.w = 1.0f;
}
if (mesh->HasNormals())
{
tempVertex.normal.x = mesh->mNormals[j].x;
tempVertex.normal.y = mesh->mNormals[j].y;
tempVertex.normal.z = mesh->mNormals[j].z;
}
if (mesh->HasTextureCoords(0))
{
tempVertex.texCoord.x = mesh->mTextureCoords[0][j].x;
tempVertex.texCoord.y = mesh->mTextureCoords[0][j].y;
}
tempVertices.push_back(tempVertex);
}
//再读取索引信息
for (int i = 0; i < mesh->mNumFaces; i++)
{
for (int j = 0; j < mesh->mFaces[i].mNumIndices; j++)
{
tempIndices.push_back(mesh->mFaces[i].mIndices[j]);
}
}
Mesh tempMesh;
tempMesh.setVertices(std::move(tempVertices));
tempMesh.setIndices(std::move(tempIndices));
tempMesh.setup();
this->meshes.push_back(std::move(tempMesh));
}
if (node->mNumChildren != 0)
{
for (int k = 0; k < node->mNumChildren; k++)
{
processNode(node->mChildren[k], scene);
}
}
return;
}
void render()
{
for (auto i = meshes.begin(); i != meshes.end(); i++)
{
i->render();
}
}
};
#endif
然后来测试一下我之前收集的那些模型文件,其主文件 LoadModel.cpp 内容如下:
#include "../include/app.hpp"
#include "../include/shader.hpp"
#include "../include/model.hpp"
#include <glm/glm.hpp>
#include <glm/gtc/matrix_transform.hpp>
class MyApp : public App {
private:
const GLfloat clearColor[4] = {0.2f, 0.3f, 0.3f, 1.0f};
Model allis;
Model buma;
Model saber;
Model cottage;
Model bugatti;
Shader* simpleShader;
public:
void init(){
ShaderInfo shaders[] = {
{GL_VERTEX_SHADER, "simpleShader.vert"},
{GL_FRAGMENT_SHADER, "simpleShader.frag"},
{GL_NONE, ""}
};
simpleShader = new Shader(shaders);
allis.loadModel( "allis.stl");
buma.loadModel("buma.stl");
saber.loadModel( "saber.stl");
bugatti.loadModel("bugatti/bugatti.obj");
glEnable(GL_DEPTH_TEST);
glDepthFunc(GL_LEQUAL);
glPolygonMode( GL_FRONT_AND_BACK, GL_LINE );
}
void display(){
glClearBufferfv(GL_COLOR, 0, clearColor);
glClear(GL_DEPTH_BUFFER_BIT);
glm::mat4 I(1.0f);
glm::vec3 X(1.0f, 0.0f, 0.0f);
glm::vec3 Y(0.0f, 1.0f, 0.0f);
glm::vec3 Z(0.0f, 0.0f, 1.0f);
float t = (float)glfwGetTime();
glm::mat4 view_matrix = glm::translate(I, glm::vec3(0.0f, 0.0f, -10.0f))
* glm::rotate(I, t, Y);
glm::mat4 projection_matrix = glm::perspective(glm::radians(45.0f), aspect, 1.0f, 100.0f);
glm::mat4 allis_model_matrix = glm::translate(I, glm::vec3(-3.0f, -1.5f, 0.0f))
* glm::scale(I, glm::vec3(1.0f, 1.0f, 1.0f)) * glm::rotate(I, glm::radians(-90.0f), X);
simpleShader->setModelMatrix(allis_model_matrix);
simpleShader->setViewMatrix(view_matrix);
simpleShader->setProjectionMatrix(projection_matrix);
simpleShader->setCurrent();
allis.render();
glm::mat4 buma_model_matrix = glm::translate(I, glm::vec3(-1.0f, -1.5f, 0.0f))
* glm::scale(I, glm::vec3(0.03f, 0.03f, 0.03f)) * glm::rotate(I, glm::radians(0.0f), X);
simpleShader->setModelMatrix(buma_model_matrix);
buma.render();
glm::mat4 saber_model_matrix = glm::translate(I, glm::vec3(7.5f, 0.6f, 1.0f))
* glm::scale(I, glm::vec3(0.03f, 0.03f, 0.03f)) * glm::rotate(I, glm::radians(-90.0f), X);
simpleShader->setModelMatrix(saber_model_matrix);
saber.render();
glm::mat4 bugatti_model_matrix = glm::translate(I, glm::vec3(5.0f, -1.0f, 0.0f))
* glm::scale(I, glm::vec3(0.2f, 0.2f, 0.2f)) * glm::rotate(I, glm::radians(0.0f), X);
simpleShader->setModelMatrix(bugatti_model_matrix);
bugatti.render();
}
~MyApp(){
if(simpleShader != NULL){
delete simpleShader;
}
}
};
DECLARE_MAIN(MyApp)
效果如下:

我这里只展示了三个美女和一个汽车,从左侧的文件列表中可以看出,我还收集了不少其它的模型,以后再逐渐与大家见面。
目前看起来还只是白茫茫一片,那是因为模型太精细了,所以看不清楚网格。下一步,我将在我的程序框架中添加改变窗口大小、在场景中漫游、线框模型和面模型之间切换等功能。今天就到这里吧。
版权申明
该随笔由京山游侠在2021年08月07日发布于博客园,引用请注明出处,转载或出版请联系博主。QQ邮箱:1841079@qq.com
使用 Assimp 库加载 3D 模型的更多相关文章
- cesium加载gltf模型点击以及列表点击定位弹窗
前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 之 ...
- Android OpenGL库加载过程源码分析
Android系统采用OpenGL绘制3D图形,使用skia来绘制二维图形:OpenGL源码位于: frameworks/native/opengl frameworks/base/opengl 本文 ...
- WPF动态加载3D 放大-旋转-平移
原文:WPF动态加载3D 放大-旋转-平移 WavefrontObjLoader.cs 第二步:ModelVisual3DWithName.cs public class ModelVisual3DW ...
- xBIM 实战01 在浏览器中加载IFC模型文件
系列目录 [已更新最新开发文章,点击查看详细] 一.创建Web项目 打开VS,新建Web项目,选择 .NET Framework 4.5 选择一个空的项目 新建完成后,项目结构如下: 二.添 ...
- cesium加载gltf模型
cesium加载gltf模型 一.采用vue-cesium:在项目里加载依赖包.命令如下: npm i --save vue-cesium 在main.js中加入如下代码: https://www.n ...
- WebGL three.js学习笔记 加载外部模型以及Tween.js动画
WebGL three.js学习笔记 加载外部模型以及Tween.js动画 本文的程序实现了加载外部stl格式的模型,以及学习了如何把加载的模型变为一个粒子系统,并使用Tween.js对该粒子系统进行 ...
- 安卓奇葩问题之.so库加载不了
真是哔了狗了. 今天突然遇到一个问题:之前用第三方的密码控件,给了一个.so库文件.然后我就放在了/jniLibs/armeabi目录下. 运行,一切都很OK. 然后重点来了.N天之后的今天,突然打包 ...
- LIB库加载方法-引用百度百科
LIB库加载方法,有三种,如下: 1.LIB文件直接加入到工程文件列表中 在VC中打开File View一页,选中工程名,单击鼠标右键,然后选中\"Add Files to Project\ ...
- linux动态库加载RPATH, RUNPATH
摘自http://gotowqj.iteye.com/blog/1926771 linux动态库加载RPATH, RUNPATH 链接动态库 如何程序在连接时使用了共享库,就必须在运行的时候能够找到共 ...
随机推荐
- 示例讲解PostgreSQL表分区的三种方式
我最新最全的文章都在南瓜慢说 www.pkslow.com,欢迎大家来喝茶! 1 简介 表分区是解决一些因单表过大引用的性能问题的方式,比如某张表过大就会造成查询变慢,可能分区是一种解决方案.一般建议 ...
- 用transform和rem哪个好
个人觉得电脑端的用transform好,毕竟电脑端的项目基本都会固定屏幕比列,16:9.28:9.32:9的 一个固定的设计稿就能很好的适配. 移动端用rem比较好,移动端的屏幕比列太杂,使用rem自 ...
- 关于WLS2中Ubuntu启用SSH远程登录功能,基于Xshell登录,支持Root
背景介绍 虽然WSL2提供了非常便利的访问Ubuntu目录的形式,但是仍然我们需要通过一个工具,比如XSHELL来实现对Ubuntu的SSH登录. 获取并安装Xshell 7 目前Xshell已经更新 ...
- 4.5 RNN循环神经网络(recurrent neural network)
自己开发了一个股票智能分析软件,功能很强大,需要的点击下面的链接获取: https://www.cnblogs.com/bclshuai/p/11380657.html 1.1 RNN循环神经网络 ...
- 打通“任督二脉”:Android 应用安装优化实战
疑问: (1)了解APK安装流程有什么好处 (2)了解APK安装流程可以解决什么问题 一.可以在安装流程里做什么 安装就分为下面三个阶段,每个阶段可以做些什么工作,可以帮助我们优化安装流程,解决安装后 ...
- 记一次Struts中文乱码
起因 最近公司一个智能家具的项目,需要开发后端,APP/WEB的所有请求通过HTPP发送到后台,后台通过socket连接到智能设备.公司只有一个Java技术栈的同事,而他负责设备方面,我只能赶鸭子上架 ...
- 全网唯一开源java开发的支持高扩展,高性能的Mqtt集群broker!
SMQTT是一款开源的MQTT消息代理Broker, SMQTT基于Netty开发,底层采用Reactor3反应堆模型,支持单机部署,支持容器化部署,具备低延迟,高吞吐量,支持百万TCP连接,同时支持 ...
- NOIP 模拟赛 day5 T2 水 故事题解
题目描述 有一块矩形土地被划分成 \(\small n×m\) 个正方形小块.这些小块高低不平,每一小块都有自己的高度.水流可以由任意一块地流向周围四个方向的四块地中,但是不能直接流入对角相连的小块中 ...
- python使用笔记006-函数+json操作
一.函数 函数:提高代码的复用性 1.1 函数的定义 1 def hello(): 2 print('hello') 3 print('fdsfjslkfs') 4 5 #函数不调用就不会执行 6 h ...
- 常用js代码积累
1,js判断进入可视区,参考(亲测不行):https://www.cnblogs.com/Marydon20170307/p/8830069.html 重点学习的话,可参考: js计算元素距离顶部的高 ...