首页 > 其他分享 >图形学学习(二):Shader输入输出及自制Shader类

图形学学习(二):Shader输入输出及自制Shader类

时间:2024-09-05 18:23:43浏览次数:18  
标签:ifstream int 输入输出 Shader 图形学 顶点 include 着色器

在LearnOpengl学了Shader的输入输出和自制Shader类,输入输出还是好理解的,自制Shader类的讲解我感觉还是用代码更直观一些(个人感觉),这两天看了一下秋招的面试面经,想了想明年找工作的问题,刺激,十分感慨,优秀的人还是太多了,不过最重要的还是得做好自己!!!


Shader的输入输出

总结有两种方式:一种是着色器之间关联的,一种是Uniform全局变量形式

如下即是着色器之间关联的输入输出方式:

顶点着色器:

layout (location = 0) in   vec3 aPos;//此为接收输入,输入要在变量前加前缀in

Out  vec4 vertexColor;//此为输出

顶点着色器输入的接收要用到我们之前用到glVertexAttribPointer函数传递,void main()中是我们对输入输出数据的处理

片段着色器:

out  vec4 FragColor;

In   vec4 vertexColor;//变量名一样才能接收到顶点着色器的输入

void main()中显然我们根据顶点着色器的输入进行了处理。

如下是Uniform的输入输出方式:

Uniform是全局的,我们不需要顶点着色器做中介,直接在代码中把数据传入就行,比较麻烦的是要调用函数去处理。
 

在渲染循环中添加就好了,然后解释一下这个
glUniform4f(vertexColorLocation, 0.0f, greenValue, 0.0f, 1.0f);

第一个参数vertexColorLocation在我看来是用指定的着色器程序找到这个Uniform变量的位置,后面是个才是参数值设置,也就是后面4个float参数,glUniform4f中的4f就是说4个float参数,这四个float代表的就是RGBA(red,green,blue,alpha)四个值的设置。

自制Shader类

 想了想先给出着色器类的代码和我遇到的问题吧,有些东西讲太冗余也没必要,我遇到的问题就是突然cpp文件无法链接到文档.h头文件了。

shader.h

#include<glad/glad.h>
#include<string>
#include<fstream>
#include<sstream>
#include<iostream>

using namespace std;

class Shader
{
public:
	//程序ID
	unsigned int ID;

	//构造器读取并构建着色器
	Shader(const char* vertexPath, const char* fragmentPath)
	{
		//从文件路径中获取顶点片段着色器
		string vertexCode;
		string fragmentCode;
		ifstream vShaderFile;
		ifstream fShaderFile;
		//保证ifstream对象可以抛出异常
		vShaderFile.exceptions(ifstream::failbit | ifstream::badbit);
		fShaderFile.exceptions(ifstream::failbit | ifstream::badbit);
		try
		{
			//打开文件
			vShaderFile.open(vertexPath);
			fShaderFile.open(fragmentPath);
			stringstream vShaderStream, fShaderStream;
			//读取文件的缓冲内容到数据流中
			vShaderStream << vShaderFile.rdbuf();
			fShaderStream << fShaderFile.rdbuf();
			//关闭文件处理器
			vShaderFile.close();
			fShaderFile.close();
			//转换数据流到string
			vertexCode = vShaderStream.str();
			fragmentCode = fShaderStream.str();
		}
		catch (ifstream::failure e)
		{
			cout << "ERROR::SHADER::FILE_NOT_SUCCESFULLY_READ" << endl;
		}

		const char* vShaderCode = vertexCode.c_str();
		const char* fShaderCode = fragmentCode.c_str();

		unsigned int vertex, fragment;

		vertex = glCreateShader(GL_VERTEX_SHADER);
		glShaderSource(vertex, 1, &vShaderCode, NULL);
		glCompileShader(vertex);
		checkCompileErrors(vertex, "VERTEX");
		// fragment Shader
		fragment = glCreateShader(GL_FRAGMENT_SHADER);
		glShaderSource(fragment, 1, &fShaderCode, NULL);
		glCompileShader(fragment);
		checkCompileErrors(fragment, "FRAGMENT");
		// shader Program
		ID = glCreateProgram();
		glAttachShader(ID, vertex);
		glAttachShader(ID, fragment);
		glLinkProgram(ID);
		checkCompileErrors(ID, "PROGRAM");
		// delete the shaders as they're linked into our program now and no longer necessary
		glDeleteShader(vertex);
		glDeleteShader(fragment);

	}
	//使用激活程序
	void use() 
	{
		glad_glUseProgram(ID);
	}
	//uniform工具函数
	void setBool(const string& name, bool value) const
	{
		glUniform1i(glGetUniformLocation(ID, name.c_str()), (int)value);
	}
	void setInt(const string& name, int value) const
	{
		glUniform1i(glGetUniformLocation(ID, name.c_str()), value);
	}
	void setFloat(const string& name, float value) const
	{
		glUniform1f(glGetUniformLocation(ID, name.c_str()), value);
	}

private:
	
	void checkCompileErrors(unsigned int shader, string type)
	{
		int success;
		char infoLog[1024];
		if (type != "PROGRAM")
		{
			glGetShaderiv(shader, GL_COMPILE_STATUS, &success);
			if (!success)
			{
				glGetShaderInfoLog(shader, 1024, NULL, infoLog);
				cout << "ERROR::SHADER_COMPILATION_ERROR of type: " << type << "\n" << infoLog << "\n -- "<<endl;
			}
		}
		else
		{
			glGetProgramiv(shader, GL_LINK_STATUS, &success);
			if (!success)
			{
				glGetProgramInfoLog(shader, 1024, NULL, infoLog);
				cout << "ERROR::PROGRAM_LINKING_ERROR of type: " << type << "\n" << infoLog << "\n -- - -- " << endl;
			}
		}
	}
};

shader.cpp

#include <glad/glad.h>
#include <GLFW/glfw3.h>

#include "shader.h"

#include <iostream>

void framebuffer_size_callback(GLFWwindow* window, int width, int height);
void processInput(GLFWwindow* window);

// settings
const unsigned int SCR_WIDTH = 800;
const unsigned int SCR_HEIGHT = 600;

int main()
{

    glfwInit();
    glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
    glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
    glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);

 
    GLFWwindow* window = glfwCreateWindow(SCR_WIDTH, SCR_HEIGHT, "LearnOpenGL", NULL, NULL);
    if (window == NULL)
    {
        std::cout << "Failed to create GLFW window" << std::endl;
        glfwTerminate();
        return -1;
    }
    glfwMakeContextCurrent(window);
    glfwSetFramebufferSizeCallback(window, framebuffer_size_callback);


    if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
    {
        std::cout << "Failed to initialize GLAD" << std::endl;
        return -1;
    }

    Shader ourShader("shader.vs", "shader.fs"); // you can name your shader files however you like


    float vertices[] = {
        // positions         // colors
         0.5f, -0.5f, 0.0f,  1.0f, 0.0f, 0.0f,  // bottom right
        -0.5f, -0.5f, 0.0f,  0.0f, 1.0f, 0.0f,  // bottom left
         0.0f,  0.5f, 0.0f,  0.0f, 0.0f, 1.0f   // top 
    };

    unsigned int VBO, VAO;
    glGenVertexArrays(1, &VAO);
    glGenBuffers(1, &VBO);
    // bind the Vertex Array Object first, then bind and set vertex buffer(s), and then configure vertex attributes(s).
    glBindVertexArray(VAO);

    glBindBuffer(GL_ARRAY_BUFFER, VBO);
    glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);

    // position attribute
    glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 6 * sizeof(float), (void*)0);
    glEnableVertexAttribArray(0);
    // color attribute
    glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, 6 * sizeof(float), (void*)(3 * sizeof(float)));
    glEnableVertexAttribArray(1);



    while (!glfwWindowShouldClose(window))
    {

        processInput(window);

 
        glClearColor(0.2f, 0.3f, 0.3f, 1.0f);
        glClear(GL_COLOR_BUFFER_BIT);

        // render the triangle
        ourShader.use();
        glBindVertexArray(VAO);
        glDrawArrays(GL_TRIANGLES, 0, 3);

  
        glfwSwapBuffers(window);
        glfwPollEvents();
    }


    glDeleteVertexArrays(1, &VAO);
    glDeleteBuffers(1, &VBO);


    glfwTerminate();
    return 0;
}


void processInput(GLFWwindow* window)
{
    if (glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS)
        glfwSetWindowShouldClose(window, true);
}


void framebuffer_size_callback(GLFWwindow* window, int width, int height)
{

    glViewport(0, 0, width, height);
}

问题解决:

在这里加上你的工程文件所在目录(如下图)就好了。

自制shader类把一下固定的前置代码都放在一个类中设置,而不用显示在main函数中一大串,也是为了方便。
也就是如下内容要放到外部文件或者Shader类中:

一、Shader代码

//顶点着色器

const char *vertexShaderSource = "#version 330 core\n"

"layout (location = 0) in vec3 aPos;\n"

"layout (location = 1) in vec3 aColor;\n"

"out vec3 ourColor;\n"

"void main()\n"

"{\n"

"   gl_Position = vec4(aPos.x, aPos.y, aPos.z, 1.0);\n"

"   ourColor = aColor;\n"  

"}\0";



//片段着色器

const char* fragmentShaderSource = "#version 330 core\n"

"out vec4 FragColor;\n"

"in vec3 ourColor;\n"

"void main()\n"

"{\n"

"   FragColor = vec4(ourColor,1.0f);\n"

"}\n\0";

二、顶点着色器、片段着色器、链接着色器的配置

 //创建顶点着色器

 unsigned int vertexShader = glCreateShader(GL_VERTEX_SHADER);

 //载入自写的着色器配置

 glShaderSource(vertexShader, 1, &vertexShaderSource, NULL);

 //编译运行顶点着色器

 glCompileShader(vertexShader);



 //检查是否成功

 int success;

 char infoLog[512];

 glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &success);

 if (!success)

 {

     glGetShaderInfoLog(vertexShader, 512, NULL, infoLog);

     std::cout << "ERROR::SHADER::VERTEX::COMPILATION_FAILED\n" << infoLog << std::endl;

 }



 //片段着色器

 unsigned int fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);

 glShaderSource(fragmentShader, 1, &fragmentShaderSource, NULL);

 glCompileShader(fragmentShader);



 glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, &success);

 if (!success)

 {

     glGetShaderInfoLog(fragmentShader, 512, NULL, infoLog);

     std::cout << "ERROR::SHADER::FRAGMENT::COMPILATION_FAILED\n" << infoLog << std::endl;

 }



 //链接着色器

 unsigned int shaderProgram = glCreateProgram();

 glAttachShader(shaderProgram, vertexShader);

 glAttachShader(shaderProgram, fragmentShader);

 glLinkProgram(shaderProgram);

 //检测链接成功没有

 glGetProgramiv(shaderProgram, GL_LINK_STATUS, &success);

 if (!success)

 {

     glGetProgramInfoLog(shaderProgram, 513, NULL, infoLog);

     std::cout << "ERROR::SHADER::PROGRAM::LINKING_FAILED\n" << infoLog << std::endl;

 }



 glDeleteShader(vertexShader);

 glDeleteShader(fragmentShader);

为达成以上目标:

首先我们将顶点着色器和片段着色器代码写在了外部(文件名及其后缀随意)

然后我也遇到一个问题就是文件中中双引号要不要写进去,发现其实不用写进去

然后它们长这样

之后就是着色器类的编写了,因为着色器写成了外部文件我们要记得去读取,然后我直接给出读取的代码了。

//从文件路径中获取顶点片段着色器

string vertexCode;

string fragmentCode;

ifstream vShaderFile;

ifstream fShaderFile;

//保证ifstream对象可以抛出异常

vShaderFile.exceptions(ifstream::failbit | ifstream::badbit);

fShaderFile.exceptions(ifstream::failbit | ifstream::badbit);

try

{

//打开文件

vShaderFile.open(vertexPath);

fShaderFile.open(fragmentPath);

stringstream vShaderStream, fShaderStream;

//读取文件的缓冲内容到数据流中

vShaderStream << vShaderFile.rdbuf();

fShaderStream << fShaderFile.rdbuf();

//关闭文件处理器

vShaderFile.close();

fShaderFile.close();

//转换数据流到string

vertexCode = vShaderStream.str();

fragmentCode = fShaderStream.str();

}

catch (ifstream::failure e)

{

cout << "ERROR::SHADER::FILE_NOT_SUCCESFULLY_READ" << endl;

}



const char* vShaderCode = vertexCode.c_str();

const char* fShaderCode = fragmentCode.c_str();

其他的就和向之前一样编写着色器的配置和链接就好了,不过这个ShaderProgram的程序ID这个变量要变成写在构造函数外面。

main中调用生成Shader对象就行了。

标签:ifstream,int,输入输出,Shader,图形学,顶点,include,着色器
From: https://blog.csdn.net/luoqingoo/article/details/141937245

相关文章

  • 图形学系列教程,带你从零开始入门图形学(包含配套代码)—— 你的第一个三角形
    图形学系列文章目录序章初探图形编程第1章你的第一个三角形第2章视图矩阵第3章顶点变换第4章纹理映射第5章透明度和深度第6章裁剪区域和模板缓冲区第7章场景图第8章场景管理第9章索引缓冲区第10章骨骼动画第11章后处理第12章实时光照(一)第13章实......
  • 两个月冲刺软考——存储器件有哪些?数据输入输出技术;计算后缀表达式(配合例题讲解+分析
    1.段页式存储管理根据段号、页号、页内地址进行计算“话术”:最多可有……个段,每个段最大允许有……个页,页的大小为……2.已知内存容量256K,求若用16K*4bit的存储器芯片构成该内存共需要多少片?16K*4bit的存储芯片,意味着每个芯片有16K个存储单元,每个存储单元可以存储4位(bit)的......
  • 集成电路学习:什么是I/O输入输出
    I/O:输入输出      I/O,全称Input/Output,即输入输出,是信息处理系统(如计算机)与外部世界(可能是人类或另一信息处理系统)之间的通信方式。具体来说,输入是系统接收的信号或数据,而输出则是系统从其发送的信号或数据。      在计算机系统中,I/O操作涉及数据在内部存储......
  • C++ 标准输入输出 -- <iostream>
    <iostream>库是C++标准库中用于输入输出操作的头文件。<iostream>定义了几个常用的流类和操作符,允许程序与标准输入输出设备(如键盘和屏幕)进行交互。以下是<iostream>库的详细使用说明,包括其主要类和常见用法示例。主要类std::istream:用于输入操作的抽象基类。std::ostre......
  • stdio.h及字符串输入输出
    这里只简单介绍常用的C语言常见的输入输出及字符串的输入输出,可以作为常用C语言字符串的速记收藏。#include<stdio.h>scanf //与空格,tab键及换行就阻断缓冲区printf //格式输入输出gets(数组名) //直到遇到换行键停止chararr[n];gets(arr);puts(数组名) //......
  • 12.流输入输出
    12.流输入输出12.1Introduction在C++程序中,首选C++样式的I/O而不是C样式的I/O。12.2流(Streams)C++I/Ooccursinstreams,whicharesequencesofbytes.C++providesboth“low-level”and“high-level”I/Ocapabilities.Low-levelI/Ocapabilities(i.e......
  • 云知声多模态模型:实时多模态输入输出;独立于 Siri ,苹果或开发新 AI 用于机器人丨 RTE
       开发者朋友们大家好: 这里是「RTE开发者日报」,每天和大家一起看新闻、聊八卦。我们的社区编辑团队会整理分享RTE(Real-TimeEngagement)领域内「有话题的新闻」、「有态度的观点」、「有意思的数据」、「有思考的文章」、「有看点的会议」,但内容仅代表编......
  • [我的C语言学习笔记(08)]C语言输入输出以及缓冲区概念
    查阅stdio.h标准库(https://cplusplus.com/reference/cstdio/),可以发现不少输入输出函数。这些是格式输入输出:这些是字符(包括字符串,也即字符数组)输入输出:这篇会介绍几个常用函数的用法,同时介绍缓冲区的概念。文章目录stream的概念输出printf函数putchar函数pu......
  • 山东大学计算机组成原理实验6七段译码设计(含原理图,引脚分配,实验结果输入输出)
    实验目的熟悉QuartusII的设计流程全过程,学习计数器的设计和硬件测试。掌握原理图的设计方法。实验原理4位计数器连接7段译码,多数码管进行显示控制。实验框图如图6所示。图6 原理图示意图其中,CNT4B采用74161计数器芯片实现,DECL7S采用7448(共阳)设计。实验内容(1)设计工程......