Unity3D之ScriptableObject学习笔记
不同与C#提供的Serializable序列化功能,ScriptableObject是Unity3D提供的一个数据存储类,我们接下来学习一下这个类的功能。
官方文档
http://docs.unity3d.com/Manual/class-ScriptableObject.html
http://docs.unity3d.com/ScriptReference/ScriptableObject.html
使用情景
在Unity3D中,我们记录游戏的配置数据可以使用文件文件(XML、JSON等格式),也可以使用二进制文件(自定义格式),两种方式都需要自己进行解析,而Unity比较贴心,为我们提供了另外一种格式的数据记录方式,就是ScriptableObject。
简单的示例
下面我们来看一个简单的示例。
首先我们需要创建记录配置的类,如下:
ShopConfig,这个类是会被作为配置数据打包到AssetBundle中的类,所以必须要继承自ScriptableObject,同时要注意文件名必须和类名一致:
using System.Collections.Generic;
using UnityEngine; /// <summary>
/// 商品配置表.
/// </summary>
public class ShopConfig : ScriptableObject
{
/// <summary>
/// 商品页签枚举.
/// </summary>
public enum ShopTag
{
hot,
item,
weapon
} /// <summary>
/// 商品列表.
/// </summary>
public List<ShopListInfo> ShopList;
}
ShopListInfo,这个类被ShopConfig引用也会被打包到AssetBundle中,但是其不会作为打包的数据类型所以不用继承ScriptableObject,但是必须添加[System.Serializable]的Attribute:
using System.Collections.Generic; /// <summary>
/// 指定页签的商品列表.
/// </summary>
[System.Serializable]
public class ShopListInfo
{
/// <summary>
/// 页签.
/// </summary>
public ShopConfig.ShopTag tag; /// <summary>
/// 商品列表.
/// </summary>
public List<ShopItemInfo> list;
}
ShopItemInfo,同上:
/// <summary>
/// 商品.
/// </summary>
[System.Serializable]
public class ShopItemInfo
{
/// <summary>
/// 名称.
/// </summary>
public string name; /// <summary>
/// 价格.
/// </summary>
public int price;
}
下面我们要创建用于打包的脚本:
using System.Collections.Generic;
using UnityEditor;
using UnityEngine; public class CreateConfig
{
[MenuItem("Tools/CreateConfig")]
private static void Create()
{
CreateShopConfig();
} private static void CreateShopConfig()
{
ShopConfig shopConfig = ScriptableObject.CreateInstance<ShopConfig>(); //填充数据, 可以从外部有策划配置好的配置表(如CSV、XML、JSON甚至是二进制文件)中通过通用代码读取所有数据来进行填充
//这里只是测试就直接手写了(⊙﹏⊙)b shopConfig.ShopList = new List<ShopListInfo>(); ShopListInfo list = new ShopListInfo();
list.tag = ShopConfig.ShopTag.hot;
list.list = new List<ShopItemInfo>();
list.list.Add(new ShopItemInfo { name = "优你弟内裤", price = });
list.list.Add(new ShopItemInfo { name = "扣扣死内裤", price = });
list.list.Add(new ShopItemInfo { name = "内裤", price = });
shopConfig.ShopList.Add(list); list = new ShopListInfo();
list.tag = ShopConfig.ShopTag.item;
list.list = new List<ShopItemInfo>();
list.list.Add(new ShopItemInfo { name = "金疮药", price = });
list.list.Add(new ShopItemInfo { name = "和合散", price = });
shopConfig.ShopList.Add(list); list = new ShopListInfo();
list.tag = ShopConfig.ShopTag.weapon;
list.list = new List<ShopItemInfo>();
list.list.Add(new ShopItemInfo { name = "轩辕剑", price = });
list.list.Add(new ShopItemInfo { name = "桃木剑", price = });
list.list.Add(new ShopItemInfo { name = "小李飞刀", price = });
list.list.Add(new ShopItemInfo { name = "大李飞刀", price = });
shopConfig.ShopList.Add(list); //填充好数据后就可以打包到 AssetBundle 中了
//第一步必须先创建一个保存了配置数据的 Asset 文件, 后缀必须为 asset
AssetDatabase.CreateAsset(shopConfig, "Assets/ShopConfig.asset"); //第二步就可以使用 BuildPipeline 打包了
BuildPipeline.BuildAssetBundle(null, new[]
{
AssetDatabase.LoadAssetAtPath("Assets/ShopConfig.asset", typeof(ShopConfig))
},
Application.streamingAssetsPath + "/Config.assetbundle",
BuildAssetBundleOptions.CollectDependencies | BuildAssetBundleOptions.CompleteAssets | BuildAssetBundleOptions.UncompressedAssetBundle,
BuildTarget.StandaloneWindows64);
}
}
我们运行一下,就可以打包出AssetBundle了,这里要注意两点:
- 继承自ScriptableObject的类不能使用new来创建,要使用ScriptableObject.CreateInstance<T>()方法来创建;
- 必须先创建对应的Asset文件才能打包,同时Asset文件的后缀必须是asset,否则Unity不能识别;
打包好了,我们弄个脚本加载看看,如下:
using UnityEngine;
using System.Collections; public class TestScript : MonoBehaviour
{
void Start()
{
AssetBundle assetBundle = AssetBundle.CreateFromFile(Application.streamingAssetsPath + "/Config.assetbundle"); ShopConfig shopConfig = assetBundle.Load("ShopConfig", typeof(ShopConfig)) as ShopConfig;
Debug.Log(shopConfig.ShopList.Count);
}
}
挂到摄像机就行了,我们看看结果:

数据正确没问题。
Asset文件
等等,我们好像忘了啥,创建出的Asset文件有啥用呢,我们点击该文件可以直接在Inspector窗口直接编辑!

这个绝了,策划直接连Excel啥别的配置工具都不需要,用Unity就可以直接编辑和配置了,不过唯一的缺点就是打出来的数据也就Unity能用了,其它语言比如后台要使用得先弄清楚Asset文件的数据结构才行(后台总不能用Unity写吧)。
总结
优点
- 除了支持float、int、string等常见的类型外,还支持List等复杂数据类型,最重要的支持Unity的Vector3等数据;
- 创建出来的文件可以直接在Unity中编辑;
缺点
- 其它语言要解析数据需要了解详细的格式,而且要花时间编写解析代码;
- 对于大量的数据还是Excel用起来舒服一点;
- 我要是配置表的结构改变了,asset文件中填好的数据是不是就要报废了?
使用方式
我认为有两种使用方式:
- 使用Excel啥的进行配置,保存为csv,打包数据时读取csv表的数据填充asset文件;
- 创建空的asset文件直接在Unity中编辑;
Unity3D之ScriptableObject学习笔记的更多相关文章
- Unity3D之UGUI学习笔记(一):UGUI介绍以及Canvas
UGUI是Unity3D4.6官方提供的UI系统,支持2D和3D UI的开发. Unity3D UI史 OnGUI 在Unity4.6之前,官方提供的是OnGUI函数来开发UI界面,当然问题也比较多, ...
- Unity3D之UGUI学习笔记(三):EventSystem
在UGUI中,EventSystem实现了所有关于交互方面的功能,和NGUI不一样的地方是,我们终于可以摆脱添加Box Collider了! 下面我们来学习一下. 对于按钮来说,直接有onClick的 ...
- Unity3D音乐音效学习笔记
对于Unity3D的音乐音效这块一直没有好好的看过,现在准备好好的研究一下,并作为一个笔记记录下. 支持格式 在游戏中,一般存在两种音乐,一种是时间较长的背景音乐,一种是时间较短的音效(比如按钮点击, ...
- Unity3d - RPG项目学习笔记(一)
通过NGUI和工程素材,学习泰课项目——黑暗之光. 现阶段心得整理: 一.开始界面 开始界面显示顺序为:①白幕渐隐:②镜头拉近:③标题渐显:④按键响应. 1.1 白幕渐隐 NGUI是一个非常强大的插件 ...
- Unity3D之移植学习笔记:移植到Android平台更好的方法
接上文,之前我们采用了直接将Unity项目导出为Eclipse项目来修改的方式,这种做法存在的一个最大的问题就是:每当Unity被修改之后,都需要重新导出,而每次导出的项目在Android平台方面的J ...
- Unity3D之移植学习笔记:移植到Android平台
首先,我们需要一台已经配置好可以开发Android应用的计算机,这里我使用的是Windows系统+Eclipse+ADT的开发环境,当然也可以使用Android Studio或者使用Mac系统都可以. ...
- Unity3D之UGUI学习笔记(二):Rect Transform与Anchor
Rect Transform 我们都知道,Unity3D中所有的GameObject都必须要携带一个Transform组件,且该组件无法移除,那么作为UI显示的GameObject则不是携带Trans ...
- I3D Next-Gen Game Development with Unity3D Vol I学习笔记(上)
这部教程主题是讲述如何在Unity中安放模型以及对场景进行优化(面对的是次世代平台).第二章看到一半,看不下,索性直接凭借Maya和Torque的经验自己操作. 看教程总结的原则如下(or tips) ...
- Unity3d编辑器扩展学习笔记
编辑器扩展 1.添加菜单栏:把特性应用于静态方法 参数1:菜单名的空格后面是定义快捷键(单符号得用"_"开头,组合键%=Ctrl,#=Shift,&=Alt) 参数2:通过 ...
随机推荐
- ACM - ICPC World Finals 2013 D Factors
原题下载:http://icpc.baylor.edu/download/worldfinals/problems/icpc2013.pdf 题目翻译: 问题描述 一个最基本的算数法则就是大于1的整数 ...
- ASP.NET MVC 学习5、登陆页面改为SSO验证
单点登录(SSO,single sign-on)是一个会话或用户身份验证过程,用户只需要登录一次就可以访问所有相互信任的应用系统,二次登录时无需重新输入用户名和密码.简化账号登录过程并保护账号和密码安 ...
- 8款PHP调试工具
Web 开发并不是一项轻松的任务,有超级多服务端脚本语言提供给开发者,但是当前 PHP 因为具有额外的一些强大的功能而越来越流行.PHP 是最强大的服务端脚本语言之一,同时也是 web 开发者和设计者 ...
- Android开发之WebService介绍
经常有网友问:“在Android平台如何调用WebService”?经过沟通我发现,甚至有些朋友连什么是WebSerivce都不知道就在问怎么使用,更别说和WebService有关的SOAP.WSDL ...
- POJ 1904 King's Quest ★(强连通分量:可行完美匹配边)
题意 有n个女生和n个男生,给定一些关系表示男生喜欢女生(即两个人可以结婚),再给定一个初始匹配,表示这个男生和哪个女生结婚,初始匹配必定是合法的.求每个男生可以和哪几个女生可以结婚且能与所有人不发生 ...
- ios第三方开源库
1.AFNetworking 目前比较推荐的iOS网络请求组件,默认网络请求是异步,通过block回调的方式对返回数据进行处理. 2.FMDB 对sqlite数据库操作进行了封装,demo也比较简单. ...
- java AES加密算法
package com.siro.tools; import javax.crypto.Cipher;import javax.crypto.spec.IvParameterSpec;import j ...
- 抽屉显示控件SlidingDrawer入门
SlidingDrawer是一个抽屉控件,代码具体路径为:android.widget.SlidingDrawer,该控件从API Level3引入,在API 17及之后的版本将不再被支持.具体效果 ...
- iOS已发布应用中对异常信息捕获和处理
iOS已发布应用中对异常信息捕获和处理 iOS开发中我们会遇到程序抛出异常退出的情况,如果是在调试的过程中,异常的信息是一目了然,但是如果是在已经发布的程序中,获取异常的信息有时候是比较困难的. iO ...
- 【剑指offer 面试题17】合并两个排序的链表
思路: 比较两个链表端点值的大小,通过递归的方式排列. #include <iostream> using namespace std; struct ListNode { int val ...