多叉树到单链表结构的转换

这篇博客详细介绍了如何将多叉树骨骼数据转换为单链表结构,采用深度优先搜索策略,类似于二叉树的前序遍历。文章提供了FindFirstLeaf、LastSibling、FirstChildHaveSibling和LastLeaf等关键函数的实现,并给出了递归创建骨骼动画的AddAnimationRecursive函数,帮助读者理解转换逻辑。

       终于写完

      由于Animation结构体的定义为单链表形式,需要把多叉树骨骼数据转换为单链表中,十分头疼。

      写之前要把转换的逻辑顺序想清楚,我在这采取深度优先(先深度后宽度)搜索,类似二叉树的前序遍历。把相关的几个函数帖出,区区几十行,但要把逻辑顺序想明白了,或许还有不对的地方。

/********************************************************
函数名称:FindFirstLeaf
功能描述:寻找本节点的叶子节点
参数说明:无

返 回 值:叶子节点
*********************************************************/
Frame_EX* Frame_EX::FindFirstLeaf()
{
 Frame_EX* pFrame = FirstChildHaveSibling();
 if(pFrame == this || pFrame->IsLeaf())
  return pFrame;
 else
  return pFrame->m_pFrameSibling->LastLeaf();
}

//宽度搜索本节点的最后一个兄弟节点,如果没有则返回自身
Frame_EX* Frame_EX::LastSibling()
{
 Frame_EX* pFrame = m_pFrameSibling;
 if(!pFrame)
  return this;

 Frame_EX* pFramePrev;
 while(pFrame)
 {
  pFramePrev = pFrame;
  pFrame = pFrame->m_pFrameSibling;
 }
 return pFramePrev;
}

//深度搜索本节点下第一个有兄弟节点的孩子节点
//如果无孩子节点则返回自身,如果所有孩子节点均无兄弟节点则返回最后一个孩子节点
Frame_EX* Frame_EX::FirstChildHaveSibling()
{
 Frame_EX* pFrame = m_pFrameFirstChild;
 if(!pFrame)
  return this;

 Frame_EX* pFrameAfter = pFrame->m_pFrameFirstChild;

 while(!pFrame->m_pFrameSibling && pFrameAfter)
 {
  pFrame = pFrameAfter;
  pFrameAfter = pFrameAfter->m_pFrameFirstChild;
 }
 return pFrame;
}

//宽度优先(先宽度后深度)搜索本节点下最后一个叶子节点
Frame_EX* Frame_EX::LastLeaf()
{
 if(IsLeaf())
  return this;

 Frame_EX* pFrame;
 pFrame = LastSibling();  //先宽度
 pFrame = pFrame->FirstChildHaveSibling(); //后深度

 return pFrame->LastLeaf();
}

 

/********************************************************
函数名称:AddAnimationRecursive
功能描述:递归创建骨骼动画
参数说明:pFramePrev  动画链表中上一级动画相关联的骨骼
    pFrame   待加入到链表中的骨骼动画

返 回 值:无
*********************************************************/
void SkeletonAnimation::AddAnimationRecursive(Frame_EX* pFramePrev, Frame_EX* pFrame)
{
 if(!pFrame)
  return;

 LPAnimation pAnimation = new Animation();
 pAnimation->pBoneName = pFrame->m_name;
 pAnimation->pBone = pFrame;

 //加入当前动画集合,pFramePrev为NULL加入到m_pAnimationSet中
 if(!pFramePrev)
 {
  pAnimation->pNext = m_pAnimationSet->pAnimations;
  m_pAnimationSet->pAnimations = pAnimation;
  m_pAnimationSet->nAnimation++;
 }
 else
 {
  pAnimation->pNext = NULL;
  m_pAnimationSet->Find(pFramePrev)->pNext = pAnimation;
  m_pAnimationSet->nAnimation++;
 }

 if(pFrame->m_pFrameFirstChild)
  AddAnimationRecursive(pFrame, pFrame->m_pFrameFirstChild);

 if(pFrame->m_pFrameSibling)
  AddAnimationRecursive(pFrame->FindFirstLeaf(), pFrame->m_pFrameSibling);
}

乐播投屏是一款简单好用、功能强大的专业投屏软件,支持手机投屏电视、手机投电脑、电脑投电视等种投屏方式。 端兼容与跨网投屏:支持手机、平板、电脑等种设备之间的自由组合投屏,且无需连接 WiFi,通过跨屏技术打破网络限制,扫一扫即可投屏。 广泛的应用支持:支持 10000+APP 投屏,包括综合视频、网盘与浏览器、美韩剧、斗鱼、虎牙等直播平台,还能将央视、湖南卫视等各大卫视的直播内容一键投屏。 高清流畅投屏体验:腾讯独家智能音画调校技术,支持 4K 高清画质、240Hz 超高帧率,低延迟不卡顿,能为用户提供更高清、流畅的视觉享受。 会议办公功能强大:拥有全球唯一的 “超级投屏空间”,扫码即投,无需安装。支持人共享投屏、远程协作批注,PPT、Excel、视频等文件都能流畅展示,还具备企业级安全加密,保障会议资料不泄露。 人互动功能:支持人投屏,邀请好友加入投屏互动,远程也可加入。同时具备一屏显、语音互动功能,支持人连麦,实时语音交流。 文件支持全面:支持 PPT、PDF、Word、Excel 等办公文件,以及视频、图片等种类型文件的投屏,还支持网盘直投,无需下载和转格式。 特色功能丰富:投屏时可同步录制投屏画面,部分版本还支持通过触控屏或电视端外接鼠标反控电脑,以及在投屏过程中用画笔实时标注等功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值