Unity 适用于单机游戏的红点系统(前缀树 | 数据结构 | 设计模式 | 算法 | 含源码)

文章目录

    • 功能包括
    • 如何使用

功能包括

  • 红点数据本地持久化

  • 如果子节点有红点,父节点也要显示红点,父节点红点数为子节点红点数的和;

  • 当子节点红点更新时,对应的父节点也要更新;

  • 当所有子节点都没有红点时,父节点才不显示红点、

  • 红点的显示方式分三种:

1.带数字的,每次经过要减1
2.不带数字只显示红点的
3.不带数字但是红点上显示感叹号的

如何使用

把这三个脚本复制到项目中
你没有这个类CryptoPrefs用PlayerPrefs代替即可

RedPointTree

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System.IO;
using Newtonsoft.Json; // 引入Json.NET库进行序列化和反序列化
/// <summary>
/// 节点名
/// </summary>
public enum ENodeNames
{Shop,Map,User,SongBtn,SongBtn_Event,VipBtn,
}
public class RedPointDataDTO
{public string Name { get; set; }public int PassCnt { get; set; }public int EnCnt { get; set; }public int RedPointCnt { get; set; }// 如果有必要,也可以添加子节点数据,但需确保不包含Unity特有的类型public List<RedPointDataDTO> Children { get; set; } = new List<RedPointDataDTO>();
}
public class RedPointTree : MonoSingleton<RedPointTree>
{/// <summary>/// 字显示为!/// </summary>public  int MaxNum = 999;/// <summary>/// 字不显示/// </summary>public  int NullNum = 998;private RedPointNode root;private const string RED_POINT_PREFS_KEY = "RedPointData";/// <summary>/// 保存红点数据到CryptoPrefs/// </summary>public void SaveRedPoints(){// 将红点树转换为可序列化的字符串var dto = ConvertToDto(this.root);string jsonData = JsonConvert.SerializeObject(dto, Formatting.None);CryptoPrefs.SetString(RED_POINT_PREFS_KEY, jsonData);CryptoPrefs.Save();}private RedPointDataDTO ConvertToDto(RedPointNode node){var dto = new RedPointDataDTO{Name = node.name,PassCnt = node.passCnt,EnCnt = node.enCnt,RedPointCnt = node.redpoinCnt};foreach (var child in node.children.Values){dto.Children.Add(ConvertToDto(child));}return dto;}/// <summary>/// 从CryptoPrefs加载红点数据/// </summary>public void LoadRedPoints(){if (CryptoPrefs.HasKey(RED_POINT_PREFS_KEY)){// 从CryptoPrefs加载并反序列化红点数据string jsonData = CryptoPrefs.GetString(RED_POINT_PREFS_KEY);var dto = JsonConvert.DeserializeObject<RedPointDataDTO>(jsonData);// 清空当前树结构,避免数据叠加this.root.children.Clear();this.root = ConvertFromDto(dto);}}private RedPointNode ConvertFromDto(RedPointDataDTO dto){var node = new RedPointNode(dto.Name);node.passCnt = dto.PassCnt;node.enCnt = dto.EnCnt;node.redpoinCnt = dto.RedPointCnt;foreach (var childDto in dto.Children){var childNode = ConvertFromDto(childDto);node.children[childDto.Name] = childNode;}return node;}public RedPointTree() {root=new RedPointNode("Root");}/// <summary>/// 初始化/// </summary>public new void Init(){LoadRedPoints(); // 先尝试加载已有的红点数据if (this.root == null){//创建根节点this.root = new RedPointNode("Root");}// 构建前缀树foreach (var name in Enum.GetNames(typeof(ENodeNames))){this.InsterNode(name);}//测试塞入红点数据// ChangeRedPointCnt(ENodeNames.SongBtn.ToString(), 20);// ChangeRedPointCnt(ENodeNames.SongBtn_Event.ToString(), 1);// ChangeRedPointCnt(ENodeNames.User.ToString(), 999);// ChangeRedPointCnt(ENodeNames.Card.ToString(), 1);// ChangeRedPointCnt(ENodeNames.Shop.ToString(), 1);}/// <summary>/// 插入节点/// </summary>/// <param name="name"></param>public void InsterNode(string name){if (string.IsNullOrEmpty(name)){return;}if (SearchNode(name) != null){//如果已经存在 则不重复插入GameLog.Log("你已经插入了该节点" + name);return;}//node从根节点出发RedPointNode node = root;node.passCnt += 1;//将名字按|符合分割string[] pathList = name.Split('_');foreach (var path in pathList){if(!node.children.ContainsKey(path)){node.children.Add(path, RedPointNode.New(path));}node = node.children[path];node.passCnt = node.passCnt+1;}node.enCnt = node.enCnt + 1;}/// <summary>/// 查询节点是否在树中并返回节点/// </summary>/// <param name="name"></param>/// <returns></returns>public RedPointNode SearchNode(string name){if (string.IsNullOrEmpty(name)){return null;}RedPointNode node=this.root;string[] pathList=name.Split('_');foreach (var path in pathList){if(!node.children.ContainsKey(path)){return null;}node = node.children[path];}if (node.enCnt > 0){return node;}return null;}/// <summary>/// 删除节点/// </summary>/// <param name="name"></param>public void DeleteNode(string name){if (SearchNode(name) == null){return;}RedPointNode node= this.root;node.passCnt = node.passCnt - 1;string[] pathList = name.Split('_');foreach (var path in pathList){RedPointNode childNode = node.children[path];childNode.passCnt = childNode.passCnt - 1;if (childNode.passCnt == 0){//如果该节点没有任何孩子,则直接删除node.children.Remove(path);return;}node = childNode;}node.enCnt=node.enCnt - 1;}/// <summary>/// 修改节点的和点数/// </summary>/// <param name="name">红点名字</param>/// <param name="delta">增量</param>public void ChangeRedPointCnt(string name, int delta){RedPointNode targetNode = SearchNode(name);if (targetNode == null){return;}//如果是减红点 并且和点数不够减了 则调整delta 使其不减为0if (delta < 0 && targetNode.redpoinCnt + delta < 0){delta = -targetNode.redpoinCnt;}RedPointNode node=this.root;string[] pathList= name.Split('_');foreach (var path in pathList){RedPointNode childNode = node.children[path];childNode.redpoinCnt = childNode.redpoinCnt + delta;node = childNode;//调用回调函数foreach (var cb in node.updateCb.Values){cb?.Invoke(node.redpoinCnt);}}// 在更新红点计数后保存数据SaveRedPoints();}/// <summary>/// 直接设置当前红点的数值/// </summary>/// <param name="name"></param>/// <param name="delta"></param>public void SetRedPointCnt(string name, int delta){RedPointNode targetNode = SearchNode(name);if (targetNode == null){return;}RedPointNode node=this.root;string[] pathList= name.Split('_');foreach (var path in pathList){RedPointNode childNode = node.children[path];childNode.redpoinCnt =  delta;node = childNode;//调用回调函数foreach (var cb in node.updateCb.Values){cb?.Invoke(node.redpoinCnt);}}// 在更新红点计数后保存数据SaveRedPoints();}/// <summary>/// 设置红点更新回调函数/// </summary>/// <param name="name">节点名</param>/// <param name="key">回调key 自定义字符串</param>/// <param name="cb">回调函数</param>public void SetCallBack(string name, string key, Action<int> cb){RedPointNode node=SearchNode(name);if (node == null){return;}if (!node.updateCb.ContainsKey(key)){node.updateCb.Add(key, cb); }else{node.updateCb[key] = cb;}}/// <summary>/// 查询节点红点数/// </summary>/// <param name="name"></param>/// <returns></returns>public int GetRedPointCnt(string name){RedPointNode node=SearchNode(name);if (node == null){return 0;}return node.redpoinCnt;}
}

RedPointNode

 
using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class RedPointNode 
{/// <summary>/// 节点名/// </summary>public string name;/// <summary>/// 节点被经过的次数/// </summary>public int passCnt = 0;/// <summary>/// 节点作为尾结点的次数/// </summary>public int enCnt = 0;/// <summary>/// 红点数/// </summary>public int redpoinCnt = 0;public Dictionary<string, RedPointNode> children ;public Dictionary<string, Action<int>> updateCb ;public RedPointNode(string name){this.name = name;this.passCnt = 0;this.enCnt = 0;this.redpoinCnt = 0;this.children = new Dictionary<string, RedPointNode>();this.updateCb = new Dictionary<string, Action<int>>();}public static RedPointNode New(string name){return new RedPointNode(name);}
}

RedPointMono

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;public class RedPointMono : MonoBehaviour
{public ENodeNames NodeName;public Text RedPointText;private void Awake(){RedPointTree.Instance.SetCallBack(NodeName.ToString(), this.gameObject.name, (redpointCnt) => { UpdateRedPoint(redpointCnt); });}void OnEnable(){//RedPointText = transform.GetChild(0).transform.GetComponent<Text>();UpdateRedPoint(RedPointTree.Instance.GetRedPointCnt(NodeName.ToString()));}private void OnDestroy(){//注销红点回调RedPointTree.Instance.SetCallBack(NodeName.ToString(), this.gameObject.name, null);}//更新红点private void UpdateRedPoint(int redpointCnt){//throw new NotImplementedException();if (redpointCnt>=RedPointTree.Instance.MaxNum){RedPointText.text = "!";}else  if (redpointCnt==RedPointTree.Instance.NullNum){RedPointText.text = "";}else{RedPointText.text = redpointCnt.ToString();}gameObject.SetActive(redpointCnt > 0);}
}

然后红点结构是这样的
在这里插入图片描述
因为是基于前缀树的,父子节点关系在这里体现
SongBtn,//父节点
SongBtn_Event,//子节点
这样当SongBtn_Event有红点的时候SongBtn也会有

参考链接:https://blog.csdn.net/linxinfa/article/details/121899276

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/71645.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用API有效率地管理Dynadot域名,为域名部署DNS安全拓展(DNSSEC)

关于Dynadot Dynadot是通过ICANN认证的域名注册商&#xff0c;自2002年成立以来&#xff0c;服务于全球108个国家和地区的客户&#xff0c;为数以万计的客户提供简洁&#xff0c;优惠&#xff0c;安全的域名注册以及管理服务。 Dynadot平台操作教程索引&#xff08;包括域名邮…

Web - JS基础语法与表达式

概述 这篇文章主要介绍了 JavaScript 的基础语法&#xff0c;包括代码书写位置、ERPL 环境、变量&#xff08;命名规则、默认值、初始化&#xff09;、数据类型&#xff08;基本和复杂&#xff0c;及各类型特点、转换&#xff09;、表达式和运算符&#xff08;算数、特殊算数、…

一台服务器将docker image打包去另一天服务器安装这个镜像

一台服务器将docker image打到去另一天服务器安装这个镜像 1. 打包2.另一台服务器执行 1. 打包 docker save -o nebula-graph-studio.tar harbor1.vm.example.lan/dockerio/vesoft/nebula-graph-studioxxx.tar 是打包好的文件 后面的是 docker image 2.另一台服务器执行 docke…

一周学会Flask3 Python Web开发-response响应格式

锋哥原创的Flask3 Python Web开发 Flask3视频教程&#xff1a; 2025版 Flask3 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili 在HTTP响应中&#xff0c;数据可以通过多种格式传输。大多数情况下&#xff0c;我们会使用HTML格式&#xff0c;这也是Flask中…

TCP和Http协议

TCP 三次握手&#xff1a; 第一次握手 &#xff1a; 初始状态&#xff1a;开始时&#xff0c;客户端处于 CLOSED&#xff08;关闭&#xff09;状态&#xff0c;服务端处于 LISTEN&#xff08;监听&#xff09;状态&#xff0c;等待客户端的连接请求。客户端发送请求&#xff…

图论 之 最小生成树

文章目录 题目1584.连接所有点的最小费用 最小生成树MST&#xff0c;有两种算法进行求解&#xff0c;分别是Kruskal算法和Prim算法Kruskal算法从边出发&#xff0c;适合用于稀疏图Prim算法从顶点出发&#xff0c;适合用于稠密图&#xff1a;基本思想是从一个起始顶点开始&#…

前端面试之Box盒子布局:核心知识与实战解析

目录 引言&#xff1a;布局能力决定前端高度 一、盒模型基础&#xff1a;看得见的像素战争 1. 标准盒模型 vs IE盒模型 2. 核心组成公式 3. 视觉格式化模型 二、传统布局三剑客 1. 浮动布局&#xff08;Float Layout&#xff09; 2. 定位布局&#xff08;Position Layou…

OnlyOffice:前端编辑器与后端API实现高效办公

OnlyOffice&#xff1a;前端编辑器与后端API实现高效办公 一、OnlyOffice概述二、前端编辑器&#xff1a;高效、灵活且易用1. 完善的编辑功能2. 实时协作支持3. 自动保存与版本管理4. 高度自定义的界面 三、后端API&#xff1a;管理文档、用户与权限1. 轻松集成与定制2. 实时协…

Python多线程编程理解面试题解析

一、多线程介绍 Python 的多线程是一种实现并发编程的方式&#xff0c;允许程序同时执行多个任务。然而&#xff0c;由于 Python 的全局解释器锁&#xff08;GIL&#xff09;的存在&#xff0c;多线程在某些场景下可能无法充分利用多核 CPU 的性能。以下是对 Python 多线程的理…

如何通过 Python 实现一个消息队列,为在线客服系统与海外运营的APP对接

对方有两个核心需求: 访客上线的时候,要通知对方的业务系统,业务系统根据访客的身份信息,推送个性化的欢迎词。访客完成下单的时候,要能推送一个下单成功的通知,并且包含订单信息和链接。根据这两个需求,那就需要实现由客服系统到业务系统的消息队列推送,以及通过 Open…

中文Build a Large Language Model (From Scratch) 免费获取全文

中文pdf下载地址&#xff1a;https://pan.baidu.com/s/1aq2aBcWt9vYagT2-HuxdWA?pwdlshj 提取码&#xff1a;lshj 原文、代码、视频项目地址&#xff1a;https://github.com/rasbt/LLMs-from-scratch 翻译工具&#xff1a;沉浸式翻译&#xff08;https://app.immersivetrans…

项目设置内网 IP 访问实现方案

在我们平常的开发工作中&#xff0c;项目开发、测试完成后进行部署上线。比如电商网站、新闻网站、社交网站等&#xff0c;通常对访问不会进行限制。但是像企业内部网站、内部管理系统等&#xff0c;这种系统一般都需要限制访问&#xff0c;比如内网才能访问等。那么一个网站应…

elf_loader:一个使用Rust编写的ELF加载器

本文介绍一个使用Rust实现的ELF加载器。 下面是elf_loader的仓库链接&#xff1a; github&#xff1a; https://github.com/weizhiao/elf_loaderhttps://github.com/weizhiao/elf_loader crates.io&#xff1a; https://crates.io/crates/elf_loaderhttps://crates.io/cra…

数据库驱动免费下载(Oracle、Mysql、达梦、Postgresql)

数据库驱动找起来好麻烦&#xff0c;我整理到了一起&#xff0c;需要的朋友免费下载&#xff1a;驱动下载 目前收录了Oracle、Mysql、达梦、Postgresql的数据库驱动的多个版本&#xff0c;后续可能会分享更多。

对接扣子双向流式 TTS Demo

Web端对接Demo <!DOCTYPE html> <html lang"zh-CN"><head><meta charset"UTF-8"><title>TTS 测试</title> </head><body><h1>TTS 测试页面</h1><textarea id"textInput" rows&…

科普:“git“与“github“

Git与GitHub的关系可以理解为&#xff1a;Git是一种软件工具&#xff0c;而GitHub则是一个在线平台&#xff0c;它们是“一家子”。二者的关联最直接体现在你通过Git在GitHub仓库中clone软件包到你的机器中来。 具体来说&#xff1a; 一、Git 定义&#xff1a;Git是一个开源的…

jsherp importItemExcel接口存在SQL注入

一、漏洞简介 很多人说管伊佳ERP&#xff08;原名&#xff1a;华夏ERP&#xff0c;英文名&#xff1a;jshERP&#xff09;是目前人气领先的国产ERP系统虽然目前只有进销存财务生产的功能&#xff0c;但后面将会推出ERP的全部功能&#xff0c;有兴趣请帮点一下 二、漏洞影响 …

【目标检测】【BiFPN】EfficientDet:Scalable and Efficient Object Detection

EfficientDet&#xff1a;可扩展且高效的目标检测 0.论文摘要 模型效率在计算机视觉中变得越来越重要。在本文中&#xff0c;我们系统地研究了用于目标检测的神经网络架构设计选择&#xff0c;并提出了几项关键优化以提高效率。首先&#xff0c;我们提出了一种加权双向特征金…

拖动线条改变区域大小

浏览网页时&#xff0c;经常看到这样一个功能&#xff0c;可以通过拖拽线条&#xff0c;改变左右区域大小 在管理后台中更为常见&#xff0c;菜单的宽度如果固定死&#xff0c;而后续新增的菜单名称又不固定&#xff0c;所以很可能导致换行&#xff0c;样式不太美观&#xff0c…

输入框元素覆盖冲突

后端响应中的 "trainingKbGroupName": "基础死型" 通过searchForm2.initFormData(rowData[0]);操作会把基础死型四个字填充到<div class"col-sm-5 form-group"> <label class"col-sm-3 control-label">知识点分组名称<…