推荐阅读:

      为什么要学习设计模式呢?我以前也思考过很多次这个问题,现在也还困惑。为什么我最后还是选择了学设计模式呢?因为在游戏中,用到的次数太多了,真的有必要学习了,所以我打算来好好研究一下。

      今天开始,我将与大家一起来学习设计模式,博主是做游戏的,所以我采用C#语言来与大家分享设计模式,话不多说,我们直接进入正题。

游戏开发中常用的设计模式之一单例模式

单例模式是我遇到的第一个设计模式,也是最常用到的设计模式,几乎没个游戏都会用到单例。

所谓单例,就是一个类只有一个实例。

单例模式:确保一个类只有一个实例,并提供一个全局访问点。

下面这个类图可以帮助大家更形象的理解。



下面举个两个例子来帮助大家理解单例模式的应用

1.游戏中的使用:游戏中玩家的属性,不使用单利模式可能会出现玩家死亡增加血量的情况。为此,我们引入单例,使得同一时间只允许一个实例对其操作。

2.现实中的例子:打印机,一个设备如果同时打印两个文件,会出现两个文件内容交错现象。

现在我们已经大致了解了单例模式的应用场景,那么,在游戏中我们如何实现呢?

从单例模式中我们可以总结两个要点:

(1)确保一个类只有一个实例;

(2)提供一个访问它的全局访问点;

下面通过采用两人对话的方式来帮助大家更快掌握分析思路:

菜鸟:怎样确保一个类只有一个实例了?

老鸟:那就让我帮你分析下,你创建类的实例会想到用什么方式来创建的>呢?

新手:用new关键字啊,只要new下就创建了该类的一个实例了,之后就>>可以使用该类的一些属性和实例方法了



老鸟:那你想过为什么可以使用new关键字来创建类的实例吗?



菜鸟:这个还有条件的吗?........., 哦,我想起来了,如果类定义私有的构造函数就不能在外界通过new创建实例了(注:有些初学者就会问,有时候我并没有在类中定义构造函数为什么也可以使用new来创建对象,那是因为编译器在背后做了手脚了,当编译器看到我们类中没有定义构造函数,此时编译器会帮我们生成一个公有的无参构造函数)



老鸟:不错,回答的很对,这样你的疑惑就得到解答了啊



菜鸟:那我要在哪里创建类的实例了?



老鸟:你傻啊,当然是在类里面创建了(注:这样定义私有构造函数就是上面的一个思考过程的,要创建实例,自然就要有一个变量来保存该实例把,所以就有了私有变量的声明,但是实现中是定义静态私有变量,朋友们有没有想过——这里为什么定义为静态的呢?对于这个疑问的解释为:每个线程都有自己的线程栈,定义为静态主要是为了在多线程确保类有一个实例)



菜鸟:哦,现在完全明白了,但是我还有另一个疑问——现在类实例创建在类内部,那外界如何获得该的一个实例来使用它了?



老鸟:这个,你可以定义一个公有方法或者属性来把该类的实例公开出去了(注:这样就有了公有方法的定义了,该方法就是提供方法问类的全局访问点)

菜鸟:怎样确保一个类只有一个实例了?



老鸟:那就让我帮你分析下,你创建类的实例会想到用什么方式来创建的呢?



新手:用new关键字啊,只要new下就创建了该类的一个实例了,之后就可以使用该类的一些属性和实例方法了



老鸟:那你想过为什么可以使用new关键字来创建类的实例吗?



菜鸟:这个还有条件的吗?........., 哦,我想起来了,如果类定义私有的构造函数就不能在外界通过new创建实例了(注:有些初学者就会问,有时候我并没有在类中定义构造函数为什么也可以使用new来创建对象,那是因为编译器在背后做了手脚了,当编译器看到我们类中没有定义构造函数,此时编译器会帮我们生成一个公有的无参构造函数)



老鸟:不错,回答的很对,这样你的疑惑就得到解答了啊



菜鸟:那我要在哪里创建类的实例了?



老鸟:你傻啊,当然是在类里面创建了(注:这样定义私有构造函数就是上面的一个思考过程的,要创建实例,自然就要有一个变量来保存该实例把,所以就有了私有变量的声明,但是实现中是定义静态私有变量,朋友们有没有想过——这里为什么定义为静态的呢?对于这个疑问的解释为:每个线程都有自己的线程栈,定义为静态主要是为了在多线程确保类有一个实例)



菜鸟:哦,现在完全明白了,但是我还有另一个疑问——现在类实例创建在类内部,那外界如何获得该的一个实例来使用它了?



老鸟:这个,你可以定义一个公有方法或者属性来把该类的实例公开出去了(注:这样就有了公有方法的定义了,该方法就是提供方法问类的全局访问点)

      在游戏中,就像在电影里,应该只有一个导演(游戏管理人)。游戏管理人是一个类,这个类在游戏中指挥发生的事情。它控制对象的呈现。它控制位置更新。它将玩家的输入指向正确的游戏角色。

        引擎应该阻止创建一个以上的游戏管理人类的实例,通过单例设计模式来实现。此设计模式确保为给定类实例化有且只有一个对象。

下面我们来看看代码怎么实现吧:

/// <summary>
/// 单例模式的实现
/// </summary>
public class GameManager
{
// 定义一个静态变量来保存类的实例
private static GameManager _instance; // 定义私有构造函数,使外界不能创建该类实例
private GameManager()
{
} /// <summary>
/// 定义公有方法提供一个全局访问点,同时你也可以定义公有属性来提供全局访问点
/// </summary>
/// <returns></returns>
public static GameManager Instance()
{
// 如果类的实例不存在则创建,否则直接返回
if (_instance== null)
{
_instance= new GameManager();
}
return _instance;
}
}

      注意:上述方法在单线程情况下堪称完美,但是在多线程的情况下会得到多个GameManager实例,因为在两个线程同时运行GameManager方法时,此时两个线程判断(_instance==null)这个条件时都返回真,此时两个线程就都会创建GameManager的实例。那么,这时候你可能会想到:使GameManager方法在同一时间只运行一个线程运行。

/// <summary>
/// 单例模式的实现
/// </summary>
public class GameManager
{
// 定义一个静态变量来保存类的实例
private static GameManager _instance; // 定义一个标识确保线程同步
private static readonly object locker = new object(); // 定义私有构造函数,使外界不能创建该类实例
private GameManager()
{
} /// <summary>
/// 定义公有方法提供一个全局访问点,同时你也可以定义公有属性来提供全局访问点
/// </summary>
/// <returns></returns>
public static GameManager GetInstance()
{
// 当第一个线程运行到这里时,此时会对locker对象 "加锁",
// 当第二个线程运行该方法时,首先检测到locker对象为"加锁"状态,该线程就会挂起等待第一个线程解锁
// lock语句运行完之后(即线程运行完之后)会对该对象"解锁"
lock (locker)
{
// 如果类的实例不存在则创建,否则直接返回
if (_instance== null)
{
_instance= new GameManager();
}
} return _instance;
}
}

      上述方法可以解决多线程问题,但是对于每个线程都要加锁后判断,难免有些繁琐,在线程创建了实例后,后面的线程只需要判断是否if(_instance== null)。在lock语句前面加一句(_instance==null)的判断就可以避免锁所增加的额外开销,这种实现方式我们就叫它 “双重锁定”

	/// <summary>
/// 单例模式的实现
/// </summary>
public class GameManager
{
// 定义一个静态变量来保存类的实例
private static GameManager _instance; // 定义一个标识确保线程同步
private static readonly object locker = new object(); // 定义私有构造函数,使外界不能创建该类实例
private GameManager()
{
} /// <summary>
/// 定义公有方法提供一个全局访问点,同时你也可以定义公有属性来提供全局访问点
/// </summary>
/// <returns></returns>
public static GameManager GetInstance()
{
// 当第一个线程运行到这里时,此时会对locker对象 "加锁",
// 当第二个线程运行该方法时,首先检测到locker对象为"加锁"状态,该线程就会挂起等待第一个线程解锁
// lock语句运行完之后(即线程运行完之后)会对该对象"解锁"
// 双重锁定只需要一句判断就可以了
if (_instance== null)
{
lock (locker)
{
// 如果类的实例不存在则创建,否则直接返回
if (_instance== null)
{
_instance= new GameManager();
}
}
}
return _instance;
}
}

      看了上面的代码你或许有疑问:为什么前面判断了_instance== null,后面又判断了,你或许会觉得没必要,但是,这个是很重要的,因为这时候我们考虑的是多线程,前一时刻判断了不代表下一时刻还是该状态。

设计模式(C#)——01单例模式的更多相关文章

  1. Java设计模式学习01——单例模式(转)

    原地址:http://blog.csdn.net/xu__cg/article/details/70182988 Java单例模式是一种常见且较为简单的设计模式.单例模式,顾名思义一个类仅能有一个实例 ...

  2. 【白话设计模式四】单例模式(Singleton)

    转自:https://my.oschina.net/xianggao/blog/616385 0 系列目录 白话设计模式 工厂模式 单例模式 [白话设计模式一]简单工厂模式(Simple Factor ...

  3. php设计模式笔记:单例模式

    php设计模式笔记:单例模式 意图: 保证一个类仅有一个实例,并且提供一个全局访问点 单例模式有三个特点: 1.一个类只有一个实例2.它必须自行创建这个实例3.必须自行向整个系统提供这个实例 主要实现 ...

  4. Java设计模式之《单例模式》及应用场景

    摘要: 原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6510196.html 所谓单例,指的就是单实例,有且仅有一个类实例,这个单例不应该 ...

  5. Java设计模式之【单例模式】

    Java设计模式之[单例模式] 何为单例 在应用的生存周期中,一个类的实例有且仅有一个 当在一些业务中需要规定某个类的实例有且仅有一个时,就可以用单例模式 比如spring容器默认初始化的实例就是单例 ...

  6. Java设计模式中的单例模式

    有时候在实际项目的开发中,我们会碰到这样一种情况,该类只允许存在一个实例化的对象,不允许存在一个以上的实例化对象,我们将这种情况称为Java设计模式中的单例模式.设计单例模式主要采用了Java的pri ...

  7. C#设计模式学习笔记-单例模式随笔

    最近学习 设计模式,从单例模式入手 啥是单例模式: 要实现一个单例类的话,首先,肯定是不能让用户自行生产的,那就是说明不能让用户new,所以,就必须把构造函数设置成为私有的 因为静态变量的生命周期跟整 ...

  8. IOS设计模式浅析之单例模式(Singleton)

    说在前面 进入正式的设计模式交流之前,扯点闲话.我们在项目开发的过程中,经常会不经意的使用一些常见的设计模式,如单例模式.工厂方法模式.观察者模式等,以前做.NET开发的时候,认真拜读了一下程杰老师的 ...

  9. C#设计模式学习笔记-单例模式(转)

    C#设计模式学习笔记-单例模式 http://www.cnblogs.com/xun126/archive/2011/03/09/1970807.html 最近在学设计模式,学到创建型模式的时候,碰到 ...

随机推荐

  1. windows上node开发注意事项

    windows上进行node.react开发的必要步骤: 1.使用nvm进行node及npm包管理工具,记得使用npm config set ...:2.另外react仅支持python3.0以下的版 ...

  2. Python实现淘宝秒杀聚划算自动提醒源码

    快来加入群[python爬虫交流群](群号570070796),发现精彩内容. 本实例能够监控聚划算的抢购按钮,在聚划算整点聚的时间到达时发出提醒(音频文件自己定义位置)并自动弹开页面(URL自己定义 ...

  3. sentos中bonding(网卡绑定技术)1

    一.GRUB添加kernel参数 1.# vim /etc/sysconfig/grubGRUB_CMDLINE_LINUX="......      net.ifnames=0" ...

  4. wscript.shell 使用

    <%@ Page Language="VB" validateRequest = "false" aspcompat = "true" ...

  5. go 学习之路(三)

    一.strings和strconv使用 1.strings.HasPrefix(s string,prefix string) bool :判断字符串s是否以prefix开头 2.stings.Has ...

  6. CentOS系统故障 | 一桩"血案"引发的容器存储驱动比较

    写在前面: 由于红帽在Linux界的影响力,相信很多朋友在测试和生产系统用的是RedHat或者CentOS系统,这次我在CentOS系统上遇到了一个很有意思的故障,通过这次故障的原因分析及解决,特意写 ...

  7. Java匹马行天下之J2EE框架开发——Spring—>用IDEA开发Spring程序(01)

    一.心动不如行动 一.创建项目 *注:在IDEA中我创建的Maven项目,不了解Maven的朋友可以看我之前的博客“我们一起走进Maven——知己知彼”,了解Maven后可以看我之前的博客“Maven ...

  8. c#小灶——数据类型

    C#中有许多数据类型,存储不同的数据要用不同的数据类型.我们这里面向初学只介绍值类型,引用类型和指针类型在后续的学习中会有接触. 整型 int是最常用的整型,用来存储整数.除了int之外,还有其他不常 ...

  9. spring-boot-plus1.2.0-RELEASE发布-快速打包-极速部署-在线演示

    spring-boot-plus 一套集成spring boot常用开发组件的后台快速开发脚手架 Purpose 每个人都可以独立.快速.高效地开发项目! Everyone can develop p ...

  10. Appium+python自动化(二十八)- 滑呀滑,滑到奈何桥喝碗孟婆汤 - 高级滑动(超详解)

    简介 奈何桥上叹奈何,三生石前憾三生,彼岸花下非彼岸,奈何三生彼岸人. 相传过了鬼门关便上一条路叫黄泉路,路上盛开着只见花,不见叶的彼岸花.花叶生生两不见,相念相惜永相失,路尽头有一条河叫忘川河,河上 ...