MATLAB实现Catmull-Clark细分(CC细分)

2023-12-25 22:30

本文主要是介绍MATLAB实现Catmull-Clark细分(CC细分),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

终于调试好了Catmull-Clark细分(CC细分)的全部程序,将之前只适用于封闭四边形网格的程序进行了完善

主要一段代码来自于三维网格细分算法(Catmull-Clark subdivision & Loop subdivision)附源码,这个博主的很多篇博文都写的非常好,但是经常丢三落四的,像在这篇博文中他就用到了函数outline.m用来计算网格的边界,但是博主却没有给出outline函数,我自己重新编写了这个函数,并且能够成功执行^^,现在我贴出完整代码

function [VV, FF, S] = CCSubdivision(V, F, iter)  % Catmull_Clark subdivision  if ~exist('iter','var')  iter = 1;  end  VV = V;  FF = F;  for i = 1:iter   nv = size(VV,1);  nf = size(FF,1);   O = outline(FF);  original = 1:nv;  boundary = O(:,1)';  interior = original(~ismember(original, boundary));  no = length(original);  nb = length(boundary);  ni = length(interior);  %% Sv  Etmp = sort([FF(:,1) FF(:,2);FF(:,2) FF(:,3);FF(:,3) FF(:,4);FF(:,4) FF(:,1)],2);  [E, ~, idx] = unique(Etmp, 'rows');  Aeven = sparse([E(:,1) E(:,2)], [E(:,2) E(:,1)], 1, no, no);  Aodd = sparse([FF(:,1) FF(:,2)], [FF(:,3) FF(:,4)], 1, no, no);  Aodd = Aodd + Aodd';  val_even = sum(Aeven,2);  beta = 3./(2*val_even);  val_odd = sum(Aodd,2);  gamma = 1./(4*val_odd);  alpha = 1 - beta - gamma;  Sv = sparse(no,no);  Sv(interior,:) = ...  sparse(1:ni, interior, alpha(interior), ni, no) + ...  bsxfun(@times, Aeven(interior,:), beta(interior)./val_even(interior)) + ...  bsxfun(@times, Aodd(interior,:), gamma(interior)./val_odd(interior));  Sboundary = ...  sparse([O(:,1);O(:,2)],[O(:,2);O(:,1)],1/8,no,no) + ...  sparse([O(:,1);O(:,2)],[O(:,1);O(:,2)],3/8,no,no);  Sv(boundary,:) = Sboundary(boundary,:);  %% Sf  Sf = 1/4 .* sparse(repmat((1:nf)',1 ,4), FF, 1);  i0 = no + (1:nf)';  %% Se  flaps = sparse([idx;idx], ...  [FF(:,3) FF(:,4);FF(:,4) FF(:,1);FF(:,1) FF(:,2);FF(:,2) FF(:,3)], ...  1);  onboundary = (sum(flaps,2) == 2);  flaps(onboundary,:) = 0;  ne = size(E,1);  Se = sparse( ...  [1:ne 1:ne]', ...  [E(:,1); E(:,2)], ...  [onboundary;onboundary].*1/2 + ~[onboundary;onboundary].*3/8, ...  ne, ...  no) + ...  flaps*1/16;  %% new faces & new vertices  i1 = no +   nf + (1:nf)';  i2 = no + 2*nf + (1:nf)';  i3 = no + 3*nf + (1:nf)';  i4 = no + 4*nf + (1:nf)';  FFtmp = [i0 i4 FF(:,1) i1; ...  i0 i1 FF(:,2) i2; ...  i0 i2 FF(:,3) i3; ...  i0 i3 FF(:,4) i4];  reidx = [(1:no)'; no+(1:nf)'; no+nf+idx];  FF = reidx(FFtmp);  S = [Sv; Sf; Se];  VV = S*VV;  end  end  
其中outline函数如下

function out = outline( FF )
%OUTLINE Summary of this function goes here
%   Detailed explanation goes here
Etmp = sort([FF(:,1) FF(:,2);FF(:,2) FF(:,3);FF(:,3) FF(:,4);FF(:,4) FF(:,1)],2);
[~, ~, idx] = unique(Etmp, 'rows');oriEtmp = [FF(:,1) FF(:,2);FF(:,2) FF(:,3);FF(:,3) FF(:,4);FF(:,4) FF(:,1)];
hh=sortrows([oriEtmp,idx],3);x2=diff(sortrows(idx));
vector = all(x2==0, 2);index1=find(vector);
index2=index1+1;
index=[index1;index2];hh(index,:)=[];
out=hh(:, 1:2);end
对于任意四边形网格是适用的,下面是我们的测试代码

[V,F]=obj__read('six.obj');
V=V';F=F';
iter=4;
[VV, FF] = CCSubdivision(V, F, iter);
%[VV, FF] = CCsub(V, F, iter);
obj_write('six1.obj',VV',FF');[V,F]=obj__read('torus.obj');
V=V';F=F';
iter=4;
[VV, FF] = CCSubdivision(V, F, iter);
%[VV, FF] = CCsub(V, F, iter);
obj_write('torus1.obj',VV',FF');
最后贴上细分效果

开放四边形网格
 六面开口盒子四次细分
封闭四边形网格
 四边形框四次细分
搞定哈哈^^

这篇关于MATLAB实现Catmull-Clark细分(CC细分)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

Java中Map的五种遍历方式实现与对比

《Java中Map的五种遍历方式实现与对比》其实Map遍历藏着多种玩法,有的优雅简洁,有的性能拉满,今天咱们盘一盘这些进阶偏基础的遍历方式,告别重复又臃肿的代码,感兴趣的小伙伴可以了解下... 目录一、先搞懂:Map遍历的核心目标二、几种遍历方式的对比1. 传统EntrySet遍历(最通用)2. Lambd

springboot+redis实现订单过期(超时取消)功能的方法详解

《springboot+redis实现订单过期(超时取消)功能的方法详解》在SpringBoot中使用Redis实现订单过期(超时取消)功能,有多种成熟方案,本文为大家整理了几个详细方法,文中的示例代... 目录一、Redis键过期回调方案(推荐)1. 配置Redis监听器2. 监听键过期事件3. Redi

SpringBoot全局异常拦截与自定义错误页面实现过程解读

《SpringBoot全局异常拦截与自定义错误页面实现过程解读》本文介绍了SpringBoot中全局异常拦截与自定义错误页面的实现方法,包括异常的分类、SpringBoot默认异常处理机制、全局异常拦... 目录一、引言二、Spring Boot异常处理基础2.1 异常的分类2.2 Spring Boot默

基于SpringBoot实现分布式锁的三种方法

《基于SpringBoot实现分布式锁的三种方法》这篇文章主要为大家详细介绍了基于SpringBoot实现分布式锁的三种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、基于Redis原生命令实现分布式锁1. 基础版Redis分布式锁2. 可重入锁实现二、使用Redisso

SpringBoo WebFlux+MongoDB实现非阻塞API过程

《SpringBooWebFlux+MongoDB实现非阻塞API过程》本文介绍了如何使用SpringBootWebFlux和MongoDB实现非阻塞API,通过响应式编程提高系统的吞吐量和响应性能... 目录一、引言二、响应式编程基础2.1 响应式编程概念2.2 响应式编程的优势2.3 响应式编程相关技术

C#实现将XML数据自动化地写入Excel文件

《C#实现将XML数据自动化地写入Excel文件》在现代企业级应用中,数据处理与报表生成是核心环节,本文将深入探讨如何利用C#和一款优秀的库,将XML数据自动化地写入Excel文件,有需要的小伙伴可以... 目录理解XML数据结构与Excel的对应关系引入高效工具:使用Spire.XLS for .NETC

Nginx更新SSL证书的实现步骤

《Nginx更新SSL证书的实现步骤》本文主要介绍了Nginx更新SSL证书的实现步骤,包括下载新证书、备份旧证书、配置新证书、验证配置及遇到问题时的解决方法,感兴趣的了解一下... 目录1 下载最新的SSL证书文件2 备份旧的SSL证书文件3 配置新证书4 验证配置5 遇到的http://www.cppc

Nginx之https证书配置实现

《Nginx之https证书配置实现》本文主要介绍了Nginx之https证书配置的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起... 目录背景介绍为什么不能部署在 IIS 或 NAT 设备上?具体实现证书获取nginx配置扩展结果验证

SpringBoot整合 Quartz实现定时推送实战指南

《SpringBoot整合Quartz实现定时推送实战指南》文章介绍了SpringBoot中使用Quartz动态定时任务和任务持久化实现多条不确定结束时间并提前N分钟推送的方案,本文结合实例代码给大... 目录前言一、Quartz 是什么?1、核心定位:解决什么问题?2、Quartz 核心组件二、使用步骤1