CG=C for Graphics  用于计算机图形编程的C语言超集

前提知识点:

1.CG代码必须用

CGPROGRAM

。。。

ENDCG括起来

2.顶点着色器与片段着色器的主函数名称可随意,但需要再#pragma vert 与#pragma fragment中声明并且与主函数名完全匹配,shader才会找到入口

3.float4是一种压缩数组,float4 vert与float vert[4]严格意义上讲不同,虽然都是存放4个float,但float4作为向量类型做点乘、内积等处理更快速

4.语义 :变量除了变量名与数据类型之外,还在:后声明其语义

例子一:RGB立方体

例题阐述:

给一个立方体(cube)创建一个shader,在立方体的直角坐标系中,三个坐标x,y,z的取值范围为{0,0,0}至{1,1,1},而RGBA颜色red,green,blue,alpha的取值范围正好是{0,0,0,0}至{1,1,1,1},将alpha写死为1,那么r,g,b分别与x,y,z建立一一映射,可得到一个6个面展示所有RGB颜色的立方体如图:

具体代码及其含义如下:

  1. Shader "Custom/RGBCube" {
  2. SubShader
  3. {
  4. Pass {
  5. CGPROGRAM
  6. #pragma vertex vert //顶点着色器入口函数声明
  7. #pragma fragment frag // 片段着色器入口函数声明
  8. //顶点输出结构体
  9. struct vertexOutput {
  10. //声明结构体的成员pos,类型为float类型的4元向量,语义为SV_POSITION,col同理;
  11. float4 pos : SV_POSITION;
  12. float4 col : TEXCOORD0;
  13. };
  14. //顶点着色器入口函数vert,与pragma第一条声明匹配,返回类型为刚刚定义的顶点输出结构体
  15. vertexOutput vert(float4 vertexPos : POSITION)
  16. {
  17. vertexOutput output; //这里不需要struct关键字
  18. //顶点着色器将数据写入输出结构体中。
  19. output.pos = mul(UNITY_MATRIX_MVP, vertexPos);
  20. //mul是顶点变换函数,UNITY_MATRIX_MVP是unity的内建矩阵,vertexPos是这个函数的形参
  21. //此行代码的作用为将形参vertexPos(本例即Cube对象的顶点向量)按照unity的内建矩阵进行顶点变换
  22. output.col = vertexPos + float4(0.5, 0.5, 0.5, 0.0);
  23. //这行代码是实现RGB立方体的关键
  24. //vertexPos的值域为题干所提到的x,y,z三元组各自减去0.5构成的值域
  25. //但是这里接受的类型为float4,可见第四元应该是无意义的常数1
  26. //意思是vertexPos的值域为{-0.5,-0.5,-0.5,1}至{0.5,0.5,0.5,1}
  27. //而对这个值域进行+{0.5,0.5,0.5,0}的矢量相加才能得到RGB (A恒定为1)的所有颜色区间
  28. return output;
  29. //将输出结构体返回,进入下一个环节(简单理解为给片段着色器)
  30. //ps:更细致的环节有顶点变换-->顶点着色-->几何元的构建-->光栅化几何元
  31. //-->片段着色-->略
  32. }
  33. //片段着色器入口函数frag,与pragma第二条声明匹配,返回类型为float4语义为COLOR,
  34. //这里除了颜色没有其他的输出,所以没有输出结构体
  35. float4 frag(vertexOutput input) : COLOR
  36. //此函数的形参类型为顶点着色器的输出结构体,没有语义
  37. //原因就在于片段着色器位于顶点着色器的下一个环节,参数按照这个顺序传递
  38. {
  39. //由于col属性已经在顶点着色器中计算,直接返回进入下一环节
  40. //下一环节是什么这里不探讨了
  41. return input.col;
  42. }
  43. ENDCG
  44. }
  45. }
  46. //如果以上SubShader渲染失败则回滚采用Diffuse
  47. FallBack "Diffuse"
  48. }

补充:

刚刚写掉了,为何vertexPos的值域为{-0,5,-0.5,-0.5,1}至{0.5,0.5,0.5,1}而非{0,0,0,1}至{1,1,1,1}呢?

因为我们的直角坐标系原点没有在顶点上而是在cube的几何中心,故其值域为{-0,5,-0.5,-0.5,1}至{0.5,0.5,0.5,1},over

[转]解读Unity中的CG编写Shader系列1——初识CG的更多相关文章

  1. 解读Unity中的CG编写Shader系列1——初识CG

    CG=C for Graphics  用于计算机图形编程的C语言超集 前提知识点: 1.CG代码必须用 CGPROGRAM ... ENDCG括起来 2.顶点着色器与片段着色器的主函数名称可任意,但须 ...

  2. 解读Unity中的CG编写Shader系列八(镜面反射)

    转自http://www.itnose.net/detail/6117378.html 讨论完漫反射之后,接下来肯定就是镜面反射了 在开始镜面反射shader的coding之前,要扩充一下前面提到的知 ...

  3. [转]解读Unity中的CG编写Shader系列9——镜面反射

    讨论完漫反射之后,接下来肯定就是镜面反射了在开始镜面反射shader的coding之前,要扩充一下前面提到的知识,加深理解镜面反射与漫反射的区别.注:这篇文章实现的镜面反射是逐顶点着色(per-ver ...

  4. [转]解读Unity中的CG编写Shader系列7——漫反射

    如果前面几个系列文章的内容过于冗长缺乏趣味着实见谅,由于时间原因前面的混合部分还没有写完,等以后再补充,现在开始关于反射的内容了.折射与反射在物理世界中,光的反射与折射往往是同时存在的,光源由真空或者 ...

  5. 解读Unity中的CG编写Shader系列七(不透明度与混合)

    转自http://www.itnose.net/detail/6098539.html 1.不透明度 当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段 ...

  6. 解读Unity中的CG编写Shader系列三

    转自http://www.itnose.net/detail/6096068.html 在上一个例子中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上.这篇文章 ...

  7. [转]解读Unity中的CG编写Shader系列6——不透明度与混合

    1.不透明度当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到 ...

  8. [转]解读Unity中的CG编写Shader系列3——表面剔除与剪裁模式

    在上一个例子中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上.这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后面 ...

  9. 解读Unity中的CG编写Shader系列3——表面剔除与剪裁模式

    在上一个样例中,我们得到了由mesh组件传递的信息经过数学转换至合适的颜色区间以颜色的形式着色到物体上. 这篇文章将要在此基础上研究片段的擦除(discarding fragments)和前面剪裁.后 ...

随机推荐

  1. Py修行路 python基础 (八)函数(随时更改)

    为何要用函数: 1.解决代码重用的问题 2.提高代码的可维护性,统一维护 3.程序的组织结构清晰,可读性强 定义函数 先定义后使用!!! def funcname(arg1,arg2,.....)  ...

  2. springboot成神之——Basic Auth应用

    本文介绍Basic Auth在spring中的应用 目录结构 依赖 入口DemoApplication 验证Authenication 配置WebSecurityConfig 控制器TestContr ...

  3. c语言相关知识点解析

    程序基本结构 常量变量标识符 数据类型 整型类型 浮点类型(实型) 基本类型转换 字符串 函数类型 枚举类型 enum 数组类型 结构体类型 共用体类型 字符串函数 运算符 流程控制语句 输入输出语句 ...

  4. C#使用protobuf

    C# protobuf的使用方法 通过.proto文件导出C#支持的.cs类文件 protocolbuffer(以下简称PB)是google 的一种数据交换的格式,它独立于语言,独立于平台.googl ...

  5. 使用Handler类来更新UI

    在android里面,我们经常要上网获取一些数据,然后更新UI,但获取数据是要时间的,如果在主线程里面直接使用获取数据的代码的话.整个activity就会卡在那,直至你获取到数据更新完UI才会加载完成 ...

  6. oracle——存储过程参数

    oracle 存储过程类型: 1.in:输入类型,即由应用程序将数据传入oracle存储过程中:这种参数在存储过程中是只读参数,在存储过程中无法对该类型的参数进行修改: 2.out:输出参数,是在存储 ...

  7. Java调用Webservice(asmx)的几个例子

    Java调用Webservice(asmx)的几个例子 2009-06-28 17:07 写了几个调用例子: 1. import org.apache.axis.client.*;import org ...

  8. 如何用navicat premium 链接Oracel数据库

    1.连接->Oracle 2.填写配置信息 3.填写完成后连接测试,发现报错 : TNS:listener does not currently know of service requeste ...

  9. Pacemaker实现双机热备

    在互联网高速发展的今天,尤其在电子商务的发展,要求服务器能够提供不间断服务.在电子商务中,如果服务器宕机,造成的损失是不可估量的.要保证服务器不间断服务,就需要对服务器实现冗余.在众多的实现服务器冗余 ...

  10. Vue基础以及指令, Vue组件

    Vue基础篇一 Vue指令 Vue的指令directive很像我们所说的自定义属性,指令时Vue模板中最常用的功能,它带有v-前缀,功能是当表达式改变的时候,相应的行为作用在DOM上. <div ...