返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >聊聊C#中的Mixin的具体用法
  • 188
分享到

聊聊C#中的Mixin的具体用法

2024-04-02 19:04:59 188人浏览 独家记忆
摘要

目录写在前面从一个简单例子说起在类中实现单例在父类中实现单例轮到Mixin出场定义Mixin在C#中在8.0之前从C#8.0开始写在前面 Mixin本意是指冰淇淋表面加的那些草莓酱,

写在前面

Mixin本意是指冰淇淋表面加的那些草莓酱,葡萄干等点缀物,它们负责给冰淇淋添加风味。在OOP里面也有Mixin这个概念,和它的本意相似,OOP里面的Mixin意在为类提供一些额外功能——在不破坏类本身或者它的继承链的基础上,在某些情况下可能会起到妙用。今天跟着小编一起来看看吧。

从一个简单例子说起

试想我们在写一个游戏引擎,创建如下类:

    class ScriptManager
    {
        public void AddScript(){}

        public void RemoveScript(){}
    }

    class EntityManager
    {
        public void AddEntity() {}

        public void RemoveEntity() {}
    }

    class AnimationManager
    {
        public void AddAnimationToWorld() {}

        public void RemoveAnimationFromWorld() {}
    }

代码非常简单,三个manager类分别控制脚本、实体和动画。但是我们突然发现,这三个类应该都是单例才合适。按照我们之前在C#中的Singleton中介绍的方法,我们这么改写一下这三个类。

在类中实现单例

最简单的,我们可以这么改

    class ScriptManager
    {
        private static ScriptManager _instance = null;
        public static ScriptManager Instance
        {
            get
            {
                if(_instance == null)
                {
                    lock(typeof(ScriptManager))
                    {
                        if(_instance == null)
                        {
                            _instance = new ScriptManager();
                        }
                    }
                }
                return _instance;
            }
        }
        public void AddScript(){}

        public void RemoveScript(){}
        private ScriptManager() {} //车门焊死,不让外部调用
    }

	class EntityManager
	{
		//类似的修改方法
	}
	
	class AnimationManager
	{
		//类似的修改方法
	}

    static void Main(string[] args)
    {
        var instance1 = ScriptManager.Instance;
        var instance2 = ScriptManager.Instance;
        var result = instance1 == instance2; //true
    }

看起来没有什么问题,确实也满足了可用的要求,但是仅仅可用是不够的,我们想要更好的解决方案,而且这种修改方法虽然简单,但如果我们想要修改的类不止这三个,或者,我们想要添加的不仅仅是单例方法,我们需要写的代码会成倍增加,所以我们想要更好的解决方案。

在父类中实现单例

很容易就能想到,既然这块代码逻辑都是一样的,我们为什么不把它提炼到父类?像这样

    class SingletonHolder<T>
        where T : class
    {
        private static T _instance = null;
        public static T Instance
        {
            get
            {
                if (_instance == null)
                {
                    lock (typeof(T))
                    {
                        if (_instance == null)
                        {
                            _instance = (T)Activator.CreateInstance(typeof(T), true); //调用非公有构造器
                        }
                    }
                }
                return _instance;
            }
        }
    }
    
    class ScriptManager : SingletonHolder<ScriptManager>
    {
		//省略
    }

    class EntityManager : SingletonHolder<EntityManager>
    {
		//省略
    }

    class AnimationManager : SingletonHolder<AnimationManager>
    {
		//省略
    }

    static void Main(string[] args)
    {
        var ScriptManager1 = ScriptManager.Instance;
        var ScriptManager2 = ScriptManager.Instance;
        var result = ScriptManager1 == ScriptManager2; //true

        var EntityManager1 = EntityManager.Instance;
        var EntityManager2 = EntityManager.Instance;
        result = EntityManager1 == EntityManager2; //true

        var AnimationManager1 = AnimationManager.Instance;
        var AnimationManager2 = AnimationManager.Instance;
        result = AnimationManager1 == AnimationManager2; //true
    }

确实可以,这样就算有再多的类需要实现单例,只要让它们继承SingletonHolder就可以了,这样的代码方便扩展也方便维护,毕竟功能逻辑都在父类里面。

不过仔细想想,这样的代码还是有点问题,类继承意味着子类应该是父类的特化,代表着一种is-a的关系,但是我们这几个Manager类和SingletonHolder并不是这种关系,它们和SingletonHolder更多像是一种实现契约的关系;如果一定要说is-a,它们应该是引擎模块(ModuleManager)的一种特化。所以让它们继承自SingletonHolder其实不是最好的方法,虽然语法正确、行为正确但是并不是语义正确,作为程序员,我们应该追求尽善尽美。而且未来真有可能会抽象出一个父类ModuleManager,到时候就发现唯一的类继承名额已经给SingletonHolder给占用了,所以我们需要寻找一种既能注入逻辑代码,又不涉及类继承的方法。

轮到Mixin出场

定义

In object-oriented programming languages, a mixin (or mix-in) is a class that contains methods for use by other classes without having to be the parent class of those other classes. How those other classes gain access to the mixin's methods depends on the language. Mixins are sometimes described as being "included" rather than "inherited".
Mixins encourage code reuse and can be used to avoid the inheritance ambiguity that multiple inheritance can cause (the "diamond problem"), or to work around lack of support for multiple inheritance in a language. A mixin can also be viewed as an interface with implemented methods. This pattern is an example of enforcing the dependency inversion principle.

这是在Wiki上面Mixin的定义,允许程序员以在类继承之外的方式为类添加一些方法,即,既能为类提供方法实现,又可以避免成为类的父类,避免了类继承和多重继承所带来的问题,这种概念正是我们需要的。

Mixin在C#中

在C#中,它们通常以拥有实现的接口出现(default implementation interface from C#8.0),而在C#8.0之前,我们通常以辅助类的方式来实现Mixin,我们下面以这两种方式改写之前的类。

在8.0之前

我们定义出一个接口,然后在外部基于这个接口实现单例逻辑(不用扩展方法是因为扩展方法不支持static method,如果想要注入的是非static method可以使用基于接口的扩展方法)

    class SingletonHolder<T>
        where T : class, ISingleton
    {
        private static T _instance = null;
        public static T Instance
        {
            get
            {
                if (_instance == null)
                {
                    lock (typeof(T))
                    {
                        if (_instance == null)
                        {
                            _instance = (T)Activator.CreateInstance(typeof(T), true);
                        }
                    }
                }
                return _instance;
            }
        }
    }

    interface ISingleton
    {
        //没有任何方法因为只是一个标记
    }

    class ScriptManager : ISingleton
    {
        private ScriptManager() {}
        public void AddScript(){}

        public void RemoveScript(){}
    }

    class EntityManager : ISingleton
    {
        private EntityManager() {}
        public void AddEntity() {}

        public void RemoveEntity() {}
    }

    class AnimationManager : ISingleton
    {
        private AnimationManager() {}
        public void AddAnimationToWorld() {}

        public void RemoveAnimationFromWorld() {}
    }

    static void Main(string[] args)
    {
        var ScriptManager1 = SingletonHolder<ScriptManager>.Instance;
        var ScriptManager2 = SingletonHolder<ScriptManager>.Instance;
        var result = ScriptManager1 == ScriptManager2; //true

        var EntityManager1 = SingletonHolder<EntityManager>.Instance;
        var EntityManager2 = SingletonHolder<EntityManager>.Instance;
        result = EntityManager1 == EntityManager2; //true

        var AnimationManager1 = SingletonHolder<AnimationManager>.Instance;
        var AnimationManager2 = SingletonHolder<AnimationManager>.Instance;
        result = AnimationManager1 == AnimationManager2; //true
    }

这就是Mixin的用处,看起来这种实现方式的好处有:

  • 类只需要声明实现ISingleton即可完成单例相关编码
  • ISingleton是接口,类可以声明实现多个接口而不会有类继承的单一限制,同时也不会有那种is-a的类继承烦恼
  • ISingleton是空接口,任何类实现它不需要额外的对该类自身的修改,就像淋上草莓酱不会对冰淇淋本身造成影响一样,符合开闭原则

从C#8.0开始

从C#8.0开始,接口可以有方法的默认实现(包括static method),我们可以更加简单的实现Mixin解决之前的问题

    interface SingletonHolder<T>
        where T:class
    {
        private static T _instance = null;
        static T Instance
        {
            get
            {
                if(_instance == null)
                {
                    lock(typeof(T))
                    {
                        if(_instance == null)
                        {
                            _instance = (T)Activator.CreateInstance(typeof(T), true);
                        }
                    }
                }
                return _instance;
            }
        }
    }
    class ScriptManager : SingletonHolder<ScriptManager>{}
    class EntityManager : SingletonHolder<EntityManager>{}
    class AnimationManager : SingletonHolder<AnimationManager>{}

这就是Mixin以及它在C#中的简单使用方法,希望通过这篇介绍能让大家对这种用法有所了解,在想要给类添加代码逻辑但是又不想改变类内部或者影响类的继承体系的时候,使用Mixin这种基于接口的代码逻辑注入也许能有奇效哦。

到此这篇关于聊聊C#中的Mixin的具体用法的文章就介绍到这了,更多相关C# Mixin用法内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 聊聊C#中的Mixin的具体用法

本文链接: https://lsjlt.com/news/150054.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 聊聊C#中的Mixin的具体用法
    目录写在前面从一个简单例子说起在类中实现单例在父类中实现单例轮到Mixin出场定义Mixin在C#中在8.0之前从C#8.0开始写在前面 Mixin本意是指冰淇淋表面加的那些草莓酱,...
    99+
    2024-04-02
  • 聊聊安装和使用Github源码的具体方法
    Github源码是开源社区中最受欢迎的源码托管平台之一,它为开发者提供了一个分享和合作的社区平台。使用Github源码需要一定的技术知识和操作技巧,接下来我将为您介绍安装和使用Github源码的具体方法。第一步,创建Github账号Gith...
    99+
    2023-10-22
  • 聊聊Gitee中Fork的具体操作和工作原理
    Gitee是一个国内广受欢迎的代码托管平台,由于其稳定性和优秀的社交化特性,越来越多的开发者选择在Gitee上托管自己的代码。而其中,Fork是Gitee上一个重要而实用的功能之一。本文将介绍Gitee中Fork的具体操作和工作原理,希望能...
    99+
    2023-10-22
  • 聊聊DW CSS的一些具体含义和解释
    DW是指Dreamweaver,是一种常用于网页制作的集成开发环境(IDE)。而CSS是指层叠样式表(Cascading Style Sheets),是Web页面设计中必不可少的技术。在Dreamweaver中,CSS通过提供方便的编辑器和...
    99+
    2023-05-14
  • 聊聊pytorch中Optimizer与optimizer.step()的用法
    当我们想指定每一层的学习率时: optim.SGD([ {'params': model.base.parameters()}, ...
    99+
    2024-04-02
  • 聊聊Node.js path模块中的常用工具函数
    本篇文章带大家聊聊Node中的path模块,介绍一下path的常见使用场景、执行机制,以及常用工具函数,希望对大家有所帮助!在开发过程中,会经常用到 Node.js  ,它利用 V8 提供的能力,拓展了 JS 的能力。而在 Nod...
    99+
    2022-06-09
    path Node 执行机制
  • 聊聊JavaScript中.?、??、??=的用法以及含义
    目录前言可选链(.)空值合并运算符()空值赋值运算符(=)趣味问答时间:值得注意的是 : 是忽视 null ,undefined 等错误的值最后前言 在项目中我们往往要做很多很多的空...
    99+
    2024-04-02
  • 详细聊聊golang中函数的用法
    随着计算机技术的不断发展,编程语言也在不断更新换代,其中Golang是近年来非常热门的一种编程语言,它的高效、安全、易用受到了很多开发者的喜爱。在Golang中,函数是一种非常重要的编程元素,本文将详细介绍Golang函数的用法。一、函数的...
    99+
    2023-05-14
  • 一起聊聊C++中的智能指针
    目录一:背景二:关键词解析1. auto_ptr2. auto_ptr 多引用问题一:背景 我们知道 C++ 是手工管理内存的分配和释放,对应的操作符就是 new/dele...
    99+
    2024-04-02
  • 聊聊php中常用的排序方法(算法)
    PHP作为一门重要的编程语言,其实在多个方面都拥有着很好的表现。在数据处理中,排序算法是最为常见和重要的一部分。PHP中提供了多种排序算法,下面详细介绍PHP中常用的排序方法。冒泡排序冒泡排序是PHP中最经典的排序算法之一。该算法通过遍历比...
    99+
    2023-05-14
    php 排序
  • 聊聊Controller中RequestMapping的作用
    目录Controller@RequestMapping作用1.value,method2.consumes,produces3.params,headersController配置总...
    99+
    2024-04-02
  • 聊聊PHP中die()和sleep()函数的用法
    在上一篇《聊聊PHP中删除字符串的逗号和尾部斜杠的方法》给大家介绍了PHP删除字符串中的逗号以及尾部斜杠的方法,感兴趣的朋友可以去学习了解一下~ 本文也将给大家通过示例来讲解标题所述...
    99+
    2024-04-02
  • 聊聊php中箭头符号(->)的用法
    PHP箭头(->)是一种用于对象访问的符号。在PHP中,对象是一组属性和方法的集合。箭头符号允许开发人员访问和操作这些属性和方法。在PHP中,对象可以通过实例化类创建,然后使用箭头符号来访问对象的属性和方法。例如,下面是一个简单的PH...
    99+
    2023-05-14
    php 箭头
  • 聊聊golang cookiejar的使用方法
    在golang中,有许多方便的库可以帮助我们进行http请求、cookie管理等操作。其中,cookie是一个常用的概念,它可以帮助我们在不同的http请求之间保持登录状态,记录用户习惯等信息。在本篇文章中,我们将介绍如何使用golang标...
    99+
    2023-05-14
  • C#实现简单的聊天窗体
    本文实例为大家分享了C#实现简单的聊天窗体的具体代码,供大家参考,具体内容如下 一、要使用(学习)到的知识点 1、textBox控件 (1)功能:允许用户输入文本,并提供多行编辑和密...
    99+
    2024-04-02
  • 聊聊技术写作的个人体会
    有群友问过,是什么原因使我开始写技术公众号,又是什么动力让我坚持写的。 在我看来,写作是一件不能敷衍的事,通过写作来学习,反而要比单纯地学习的效果要好。为了写成一篇“拿得出手”的文章,我要反复查找资料,阅读与思考,拆解与整合,最终写成的...
    99+
    2023-01-30
    技术
  • 聊聊golang中的方法和接口
    Go语言中的方法和接口是非常重要的概念,掌握好这两个知识点的使用可以提高Go语言程序的可复用性和灵活性。方法在Go语言中,方法是一种与特定类型相关联的函数。它可以被认为是一个带有特殊的接收器参数类型的函数。在声明方法时,接收器参数在方法名称...
    99+
    2023-05-14
  • 聊聊Golang中常用的编程工具及其优缺点
    Golang,也称作Go语言,是近年来Web开发领域越来越受欢迎的编程语言。它在高性能、并发编程和代码可读性等方面都表现出色,因此吸引了众多开发者的关注。而在该语言的发展过程中,不同的编程工具对于Golang语言的发展和使用发挥了重要作用。...
    99+
    2023-05-14
  • 一起详细聊聊C#中的Visitor模式
    目录写在前面模式演进举个例子使用了Tpye-Switch的版本尝试使用重载的版本单分派与双分派Visitor模式总结写在前面 Visitor模式在日常工作中出场比较少,如果统计大家不...
    99+
    2024-04-02
  • 深入聊聊Node进程管理工具-pm2的使用方法
    如何使用Node进程管理工具-pm2,下面本篇文章带大家聊聊Node进程管理工具-pm2的使用方法,希望对大家有所帮助!pm2 是什么pm2 是一个守护进程管理工具,它能帮你守护和管理你的应用程序。通常一般会在服务上线的时候使用 pm2 进...
    99+
    2023-05-14
    node Node.js
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作