明天十一放假,今天不知什么原因看到一篇unity单例模式的介绍,瞬间来了戾气。

(一)最简单的单利

public class WebRequestUtility : MonoBehaviour
{
public static WebRequestUtility Instance; private void Awake()
{
Instance = this;
}
}

这是最简单有效,最实用最没问题的单利模式。如果说存在什么问题,那就是在别的脚本awake中引用,若果其他脚本中的awake比上述awke先执行,则会报空引用。此问题通过设置脚本执行顺序课解决。题外话,一般初始化启动尽量用start,除非是非常确定要先运行用awake,不要为了先执行而用awake;

(二)构造函数法

public class WebRequestUtility : MonoBehaviour
{
public static WebRequestUtility Instance; WebRequestUtility()
{
Instance = this;
} }

构造函数中初始化instance最先执行,会在awake之前(官方为找到直接说明,但是鉴于脚本先初始化而后运行,会比awake先执行,事实也是这样,如果有人发现有问题再议,来互怼)。但在2017unity版本中遇到过bug(紧遇到一次),即刚开始运行时,构造函数多次运行,原因未找到。

(三)DontDestroyOnLoad情况下的单例模式

有时候单例不能随着场景的加载而消失,需要一直存在,所以需要不能销毁,但是加载时场景中就会存在两个单例(这个说法本身有问题,即本来已经有单例,但是在场景加载时awake又重新运行的情况)

public class WebRequestUtility : MonoBehaviour
{
public bool bDontDestroyOnLoad = false; public static WebRequestUtility Instance; private void InitializeInstance()
{
if (Instance != null & Instance == this)
return; if(bDontDestroyOnLoad)
{
if(Instance==null)
{
Instance = this;
DontDestroyOnLoad(gameObject);
}
else
{
Destroy(gameObject);
}
}
else
{
Instance = this;
}
} private void Awake()
{
InitializeInstance();
}
}

上述代码参考自unity的NetworkManager组件的源代码。逻辑很简单,只允许有一个单例,引用时,除了保证单例已经初始化(不为空),还要保证单例==this。否则就是单例被重新赋值了,即在别的地方又重新初始化,既然使用单例模式这是不允许的。

(四)静态属性或者静态方法法

public class WebRequestUtility : MonoBehaviour
{
private static WebRequestUtility instance; public static WebRequestUtility Instance
{
get
{
if(instance==null)
{
instance = new WebRequestUtility();
} return instance;
}
}
}

这种也是一种比较好的方法,但是没有(一)(二)简洁,一般用于获取单例时需要再初始化等问题时,对于大部分单例模式(一)(二)已经够用。

(四)单例模式乱象(真正互怼开始)

本想再这部分直接开始怼的,但是写到这人戾气已经没了,还是好好学习的好啊,学习让我心平气和。继续说某度出的unity单例的问题。

1)单例变”多例“

从他代码角度确实时要单例,但是经过几行代码后变成多例,然后又经过几行代码,然后可从多个事例中return一个”单例“(???),本想直接上链接怼的,想想还是算了。可以看看此文https://blog.csdn.net/qq_15267341/article/details/54232854,多简介,多明了。

2)使用锁(lock)

使用锁没有任何问题,但是unity中不推荐使用。再.net中单例模式必须加锁,因为再多线程中统一使用单例会出现冲突等等问题,比如死锁,或者逻辑未理清出问题。但是unity是单线程操作,通过协程来实现“异步“操作,所以不存在此问题。当然也存在开线程的问题,但是开线程或者异步操作只针对纯数据层面的操作,因为在非主线程中是无法进行组件操作的(通俗讲就是操作unity自定义的东西)。所以在unity中异步或者开线程时,基本可以避免在其他线程中调用主线程中的单例情况(针对游戏层面,vr ar来说)。在极少数非要进行相关处理的(一般是在回调时出现),也可以通过在update中实时检测来解决。当然了,如果你的单例不涉及到unity相关组件操作,那也就不用继承mono,就可以用纯C#的语法来处理了。

最后竟然一点戾气都没有了。。。。。。。。。。。。。。。。。。

Unity 单例模式的更多相关文章

  1. 【转】Unity+单例模式的依赖注入

    http://www.cnblogs.com/floyd/archive/2009/06/17/1505117.html

  2. unity脚本的运行顺序以及单例的实现

    unity引擎把所有脚本先行编译后,在运行的时候一批,一批的函数进行执行. unity脚本自带函数执行顺序如下:将下面脚本挂在任意物体运行即可得到 Awake ->OnEable-> St ...

  3. unity单例封装

    # 1.前言本篇主要针对Unity单例模式,完成一个可以重复继承使用的抽象类,减少重复的工作与代码.同时,对存在的多种单例进行优劣分析.# 2.Unity单例问题分析## 2.1 单例原则单例要满足以 ...

  4. 【Unity3D基础教程】给初学者看的Unity教程(七):在Unity中构建健壮的单例模式(Singleton)

    作者:王选易,出处:http://www.cnblogs.com/neverdie/ 欢迎转载,也请保留这段声明.如果你喜欢这篇文章,请点推荐.谢谢! 该博客中的代码均出自我的开源项目 : 迷你微信 ...

  5. 【Unity技巧】使用单例模式Singleton

    这几天想把在实习里碰到的一些好的技巧写在这里,也算是对实习的一个总结.好啦,今天要讲的是在Unity里应用一种非常有名的设计模式——单例模式. 开场白 单例模式的简单介绍请看前面的链接,当然网上还有很 ...

  6. Unity中C#单例模式使用总结

    一.单例模式优点 单例模式核心在于对于某个单例类,在系统中同时只存在唯一一个实例,并且该实例容易被外界所访问: 意味着在内存中,只存在一个实例,减少了内存开销: 二.单例模式特点 只存在唯一一个实例: ...

  7. MonoSingleton——Unity中的单例模式

    Unity中有很多特别的类需要以单例模式呈现,比如全局的UI管理类,各种缓存池,以及新手导航类等等.而Unity中,因为所有继承自Monobehaviour的脚本在实现的时候都是单线程的,所以像网上流 ...

  8. 单例模式在Unity中的应用

    起因:每个游戏场景中都会有许多的游戏对象,而各个游戏场景之间也是同等的关系.如何去管理它们,是我们要解决的问题. 场景中各脚本间的直接访问,会在各脚本间形成一个巨大而又混乱的网络,这给以后代码的维护带 ...

  9. Unity C# 设计模式(一)单例模式

    动机(Motivation):    在软件系统中,经常有这样一些特殊的类,必须保证它们在系统中只存在一个实例,才能确保它们的逻辑正确性.以及良好的效率 意图:    保证一个类仅有一个实例,并提供一 ...

随机推荐

  1. 【面试题】Java基础部分面试题

    Java基础面试题 Equals与==的区别 使用==比较原生类型如:boolean,,int,char等等,  使用equals()比较对象. 1.  ==是判断两个变量或类型是不是指向同一个内存空 ...

  2. 从 Int 到 Integer 对象,细细品来还是有不少东西

    int 是 Java 八大原始类型之一,是 Java 语言中为数不多不是对象的东西,Integer 是 int 的包装类,里面使用了一个 int 类型的变量来存储数据,提供了一些整数之间的常用操作,常 ...

  3. Mybaits-从零开始-Hello World(暂不考虑命名规范化)

    1.mybatis-config.xml <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE ...

  4. 体验StartOS

    旧电脑,原来使用的是win xp.随着win及其支持打软件打“成长”,电脑运行越来越慢了.一个操作需要很长的时间等待,终于,失去了耐心,换了新的电脑. 旧电脑弃置多年,留之无用,弃之可惜.偶发奇想,换 ...

  5. Dedecms手机站三种不同建设方法和优劣分析

    dedecms简单易用功能强大,是国内使用最多的cms建站系统,百度站长平台专门推出了“织梦移动化指南”,由此可见dedecms的影响力.织梦也是站长使用和学习最早的cms建站系统,解放了我的双手,让 ...

  6. 手把手教你如何在Windows下allure与jenkins的集成生成让你一见钟情的测试报告 - 03(非常详细,非常实用)

    简介 好了,国庆假期结束,开始搬砖.为什么要把allure和jenkins集成了?原因是集成以后,我们就可以直接查看allure的结果,不需要重复输入命令.重复使用浏览器打开文件来查看allure的结 ...

  7. 跨域及JSONP原理

    什么是跨域:a.com 域名下的js无法操作b.com或是c.a.com域名下的对象 为什么浏览器要引入跨域问题? 跨域问题来源于浏览器的同源策略,为啥要有这个策略呢? 为了安全.假设现在有a.com ...

  8. request.getPathInfo()的作用

    项目中使用该方法来获取URL,但不知道获取的URL的起始: 如:http://localhost:8080/web_hrs/Action/这里可以是任何东西  ---->request.getP ...

  9. Kafka 学习笔记之 High Level Consumer相关参数

    High Level Consumer相关参数 自动管理offset auto.commit.enable = true auto.commit.interval.ms = 60*1000 手动管理o ...

  10. 死磕 java线程系列之线程池深入解析——体系结构

    (手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本. 简介 Java的线程池是块硬骨头,对线程池的源码做深入研究不仅能提高对Java整个并发编程的理解,也能提高自己 ...