Unity中的JsonManager

1.具体代码

先贴代码

using LitJson;
using System.IO;
using UnityEngine;/// <summary>
/// 序列化和反序列化Json时  使用的是哪种方案    有两种  JsonUtility 不能直接序列化字典  ligJson可以序列化字典 
/// </summary>
public enum JsonType
{JsonUtility,LitJson,Newtonsoft,
}/// <summary>
/// Json数据管理类 主要用于进行 Json的序列化存储到硬盘 和 反序列化从硬盘中读取到内存中
/// </summary>
public class JsonMgr : SingleTon<JsonMgr>
{public JsonMgr(){}//存储Json数据 序列化public void SaveData(object data, string fileName, string directPath = "", JsonType type = JsonType.Newtonsoft){//确定存储路径string directoryPath = Application.persistentDataPath + "/" + directPath;string filePath = directoryPath + fileName + ".json";//序列化 得到Json字符串string jsonStr = "";switch (type){case JsonType.JsonUtility:jsonStr = JsonUtility.ToJson(data);break;case JsonType.LitJson:jsonStr = JsonMapper.ToJson(data);break;case JsonType.Newtonsoft:jsonStr = Newtonsoft.Json.JsonConvert.SerializeObject(data);break;}if (!Directory.Exists(directoryPath)){Directory.CreateDirectory(directoryPath);}//把序列化的Json字符串 存储到指定路径的文件中File.WriteAllText(filePath, jsonStr);}//读取指定文件中的 Json数据 反序列化public T LoadData<T>(string filePath, JsonType type = JsonType.Newtonsoft) where T : new(){//数据对象T data = new T();//确定从哪个路径读取//首先先判断 默认数据文件夹中是否有我们想要的数据 如果有 就从中获取string path = Application.streamingAssetsPath + "/" + filePath + ".json";//先判断 是否存在这个文件//如果不存在默认文件 就从 读写文件夹中去寻找if (!File.Exists(path))path = Application.persistentDataPath + "/" + filePath + ".json";//如果读写文件夹中都还没有 那就返回一个默认对象if (!File.Exists(path))return data;//进行反序列化string jsonStr = File.ReadAllText(path);switch (type){case JsonType.JsonUtility:data = JsonUtility.FromJson<T>(jsonStr);break;case JsonType.LitJson:data = JsonMapper.ToObject<T>(jsonStr);break;case JsonType.Newtonsoft:data = Newtonsoft.Json.JsonConvert.DeserializeObject<T>(jsonStr);break;}//把对象返回出去return data;}
}

2.使用示例

保存玩家数据

// 定义数据类
[Serializable]
public class PlayerData {public string name;public int level;
}// 保存数据
PlayerData data = new PlayerData { name = "Alice", level = 10 };
JsonMgr.Instance.SaveData(data, "player", "Save/", JsonType.Newtonsoft);

加载玩家数据

// 加载数据
PlayerData loadedData = JsonMgr.Instance.LoadData<PlayerData>("Save/player", JsonType.Newtonsoft);
Debug.Log($"Name: {loadedData.name}, Level: {loadedData.level}");

再补充一下一般我实际的使用方法

[System.Serializable] // 添加可序列化标签
public class GameData
{public bool openLevel2 = false;public bool openLevel3 = false;public bool openLevel4 = false;public bool openLevel5 = false;public bool openCustom = false;
}public class LevelManager : MonoBehaviour
{// 添加当前游戏数据引用public GameData currentGameData;private const string SAVE_FILE_NAME = "GameSaveData";private void Start(){GameDataInit();}/// <summary>/// 读取玩家信息,没有则创建/// </summary>private void GameDataInit(){// 从JSON加载数据currentGameData = JsonMgr.Instance.LoadData<GameData>(SAVE_FILE_NAME);Debug.Log("持久化数据路径: " + Application.persistentDataPath);// 如果没有存档数据,创建新数据if (currentGameData == null){currentGameData = new GameData();Debug.Log("创建新的游戏存档");}}/// <summary>/// 保存游戏数据到JSON/// </summary>public void SaveGameData(){JsonMgr.Instance.SaveData(currentGameData, SAVE_FILE_NAME);Debug.Log("游戏数据已保存");}}

 把这个脚本设置成单例或者在合适的地方调用SaveGameData,存储你想保存的类,这样可以自动创建json自动读取,只需要在合适的地方存档就好了

3.核心功能

  • 多方案支持

    JsonType 枚举:支持三种 JSON 序列化/反序列化方案:
    • Newtonsoft(Json.NET):功能强大的第三方库,支持复杂类型和更多特性。
    • LitJson:第三方库,支持字典等复杂类型。
    • JsonUtility:Unity 内置方案,不支持字典等复杂类型。
  • 数据存储与读取

    • SaveData 方法:将对象序列化为 JSON 字符串并保存到指定路径。
    • LoadData<T> 方法:从指定路径读取 JSON 文件并反序列化为对象。

4.代码结构解析

1. 构造函数

public JsonMgr() { }

  • 单例类的默认构造函数,无特殊逻辑。

2. 保存数据方法 SaveData

public void SaveData(object data, string fileName, string directPath = "", JsonType type = JsonType.Newtonsoft)
  • 参数说明

    • data:要保存的对象。
    • fileName:文件名(不含 .json 后缀)。
    • directPath:可选的子目录路径。
    • type:序列化方案(默认使用 Newtonsoft)。
  • 逻辑流程

    1. 确定存储路径
      • 使用 Application.persistentDataPath(可读写路径)构建完整路径。
      • 示例:/Users/xxx/Library/Application Support/Unity/.../directPath/fileName.json
    2. 序列化
      • 根据 type 使用对应方案将对象转换为 JSON 字符串。
      • JsonUtilityJsonUtility.ToJson(data)
      • LitJsonJsonMapper.ToJson(data)
      • NewtonsoftJsonConvert.SerializeObject(data)
    3. 创建目录:若目录不存在则创建。
    4. 写入文件:使用 File.WriteAllText(filePath, jsonStr) 存储数据。

3. 读取数据方法 LoadData<T>

public T LoadData<T>(string filePath, JsonType type = JsonType.Newtonsoft) where T : new()
  • 参数说明

    • filePath:文件路径(不含 .json 后缀)。
    • type:反序列化方案(默认使用 Newtonsoft)。
    • 泛型约束 where T : new():确保类型 T 有无参构造函数。
  • 逻辑流程

    1. 优先从只读路径加载
      • 首先检查 Application.streamingAssetsPath(只读路径,通常用于默认配置文件)。
      • 如果文件不存在,尝试从 Application.persistentDataPath(用户数据路径)加载。
    2. 若文件不存在:返回默认的 T 实例(通过 new T() 创建)。
    3. 反序列化
      • 根据 type 使用对应方案将 JSON 字符串转换为对象。
      • JsonUtilityJsonUtility.FromJson<T>(jsonStr)
      • LitJsonJsonMapper.ToObject<T>(jsonStr)
      • NewtonsoftJsonConvert.DeserializeObject<T>(jsonStr)

关键点分析

1. 路径选择与平台兼容性

  • Application.persistentDataPath
    • 可读写路径,适合保存用户数据(如玩家进度、配置等)。
  • Application.streamingAssetsPath
    • 只读路径,通常用于打包后的默认资源(如初始配置文件)。
    • 注意:在某些平台(如 Android)上,File.ReadAllText 无法直接访问此路径的文件,需通过 UnityWebRequest 或 WWW 加载。当前代码可能在此类平台上抛出异常。

2. 默认值处理

  • 若文件不存在,返回 new T() 创建的默认对象。适用于需要保证最小功能的场景(如游戏配置加载失败时使用默认值)。

3. 异常处理缺失

  • 当前代码未处理文件读写或序列化/反序列化过程中可能出现的异常(如 IO 错误、格式错误)。建议添加 try-catch 块并记录日志。

4. 性能优化建议

  • 异步操作:大量数据读写可能阻塞主线程,可改用异步方法(如 File.ReadAllTextAsync / File.WriteAllTextAsync)。
  • 缓存机制:对频繁读取的数据(如配置文件),可考虑缓存反序列化后的对象。

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

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

相关文章

50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Split Landing Page(拆分展示页)

&#x1f4c5; 我们继续 50 个小项目挑战&#xff01;—— SplitLandingPage 组件 仓库地址&#xff1a;https://github.com/SunACong/50-vue-projects 项目预览地址&#xff1a;https://50-vue-projects.vercel.app/ 在这篇文章中&#xff0c;我们将实现一个交互式的左右面板…

机器学习-ROC曲线​​ 和 ​​AUC指标

1. 什么是ROC曲线&#xff1f;​​ ROC&#xff08;Receiver Operating Characteristic&#xff0c;受试者工作特征曲线&#xff09;是用来评估​​分类模型性能​​的一种方法&#xff0c;特别是针对​​二分类问题​​&#xff08;比如“患病”或“健康”&#xff09;。 ​…

Docker容器创建Redis主从集群

利用虚拟机中的三个Docker容器创建主从集群&#xff0c;容器信息&#xff1a; 容器名角色IP映射端口r1master192.168.150.1017001r2slave192.168.150.1017002r3slave192.168.150.1017003 启动多个redis实例 新建一个docker-compose文件来构建主从集群&#xff1a; 文件内容&…

手写ArrayList和LinkedList

项目仓库&#xff1a;https://gitee.com/bossDuy/hand-tear-collection-series 基于b站up生生大佬&#xff1a;https://www.bilibili.com/video/BV1Kp5tzGEc5/?spm_id_from333.788.videopod.sections&vd_source4cda4baec795c32b16ddd661bb9ce865 LinkedList package com…

每日c/c++题 备战蓝桥杯(Cantor 表)

Cantor 表的探究与实现 在数学中&#xff0c;有理数的可枚举性是一个令人惊叹的结论。今天&#xff0c;就让我们一起深入探讨这个经典问题&#xff0c;并分享一段精心编写的代码&#xff0c;揭开这一数学奥秘的神秘面纱。 问题背景 在 19 世纪末&#xff0c;伟大的数学家康托…

解决idea与springboot版本问题

遇到以下问题&#xff1a; 1、springboot3.2.0与jdk1.8 提示这个包org.springframework.web.bind.annotation不存在&#xff0c;但是pom已经引入了spring-boot-starter-web 2、Error:Cannot determine path to tools.jar library for 17 (D:/jdk17) 3、Error:(3, 28) java: …

Notepad++找回自动暂存的文件

场景&#xff1a; 当你没有保存就退出Notepad&#xff0c;下次进来Notepad会自动把你上次编辑的内容显示出来&#xff0c;以便你继续编辑。除非你手动关掉当前页面&#xff0c;这样Notepad就会删除掉自动保存的内容。 问题&#xff1a; Notepad会将自动保存的文件地址,打开Note…

yolov12毕设前置知识准备 1

1 什么是目标检测呢&#xff1f; 目标检测&#xff08;Object Detection&#xff09;主要用于识别图像或视频中特定类型物体的位置&#xff0c;并标注其类别。 简单来说&#xff0c;就是让计算机像人类一样 “看懂” 图像内容&#xff0c;不仅能识别出物体&#xff08;如人、…

unix/linux source 命令,其内部结构机制

要理解 source (或 .) 命令的内部结构机制,我们需要戴上“操作系统”和“解释器设计”的眼镜,深入到 Shell 如何管理其状态以及如何执行命令的层面。 虽然我们无法直接看到 Shell 内部的 C 代码(除非我们去阅读 Bash 或 Zsh 的源码),但我们可以基于其行为和操作系统的原理…

计算机网络学习20250528

地址解析协议ARP 实现IP地址和Mac地址的转换 ARP工作原理&#xff1a; 每台主机或路由器都有一个ARP表&#xff0c;表项&#xff1a;<IP地址&#xff0c;Mac地址&#xff0c;TTL>&#xff08;TTL一般为20分钟&#xff09; 主机产生ARP查询分组&#xff0c;包含源目的IP地…

【Rust】Rust获取命令行参数以及IO操作

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

微服务中引入公共拦截器

本文使用的微服务版本为springcloudAlbaba :2021.0.4.0 微服务工程&#xff0c;一般公共的东西都放入一个工程&#xff0c;别的微服务都会引入这个工程&#xff0c;比如common-service,那么就可以在这个工程编写一个拦截器&#xff1a;&#xff0c;比如&#xff1a; public cla…

Linux SLES 系统的/var/log/下的常见文件及其作用

在 SUSE Linux Enterprise Server&#xff08;SLES&#xff09; 系统中&#xff0c;/var/log/ 目录是系统日志的集中地&#xff0c;存储了各种服务、内核、系统消息的日志。以下是一些在 /var/log/ 下常见的日志文件及其功能&#xff1a; &#x1f4c2; 常见日志文件及功能 文…

oracle goldengate同步SQL server到SQL server的实时数据同步

参考文档 https://docs.oracle.com/en/middleware/goldengate/core/19.1/oggmp/oracle-goldengate-classic-sql-server.html#GUID-948C5BEE-E7A0-4CE2-BE09-F83145677D18 https://docs.oracle.com/en/middleware/goldengate/core/21.3/ggcab/other-programs-and-settings-sql-…

语音转文字工具

平时工作和学习比较忙&#xff0c;可能没时间听讲座&#xff0c;只能看回放&#xff0c;回访也很长&#xff0c;这时&#xff0c;我们可以借助语言转文字&#xff0c;通过阅读文字快速了解讲座的重点&#xff0c;今天给大家分享一个本人经常用的语言转文字工具&#xff0c;改工…

硬件实时时钟(RTC)

硬件实时时钟&#xff08;RTC&#xff09;详解 硬件实时时钟&#xff08;Real-Time Clock&#xff0c;RTC&#xff09;是计算机主板上的一个独立计时芯片&#xff0c;用于在系统关机后持续记录时间。它不依赖操作系统&#xff0c;由纽扣电池&#xff08;如CR2032&#xff09;供…

pycharm debug的时候无法debug到指定的位置就停住不动了

报错大致是这样的&#xff0c;但是直接run没有问题&#xff0c;debug就停住不动了 Traceback (most recent call last): File "/home/mapengsen/.pycharm_helpers/pydev/_pydevd_bundle/pydevd_comm.py", line 467, in start_client s.connect((host, port)) Timeou…

Python6.1打卡(day33)

DAY 33 MLP神经网络的训练 知识点回顾&#xff1a; 1.PyTorch和cuda的安装 2.查看显卡信息的命令行命令&#xff08;cmd中使用&#xff09; 3.cuda的检查 4.简单神经网络的流程 1.数据预处理&#xff08;归一化、转换成张量&#xff09; 2.模型的定义 …

NodeJS全栈开发面试题讲解——P11消息队列(MQ)

✅ 11.1 为什么要用消息队列&#xff1f;在哪些场景下最适合&#xff1f; ✅ 作用&#xff1a; 削峰填谷&#xff1a;缓解高并发压力&#xff0c;异步处理任务&#xff08;如秒杀下单 → MQ → 异步扣库存&#xff09; 解耦服务&#xff1a;上下游解耦&#xff08;如下单服务…

mysql执行sql语句报错事务锁住

报错情况 1205 - Lock wait timeout exceeded; try restarting transaction先找出长时间运行的事务 SELECT * FROM information_schema.INNODB_TRX ORDER BY trx_started ASC;终止长时间运行的事务 KILL [PROCESS_ID];