• C#基础:单例模式与多线程


    一、单例模式

    我们先来看看两种创建单例模式的示例代码。

    1、饿汉式

     饿汉式创建单例模式是在程序里面直接初始化了一个对象实例:

    class Good
    {
        /// <summary>
        /// 私有的静态变量,直接初始化
        /// </summary>
        private static Good Instance = new Good();
    
        /// <summary>
        /// 私有的构造函数
        /// </summary>
        private Good()
        {
    
        }
    
        /// <summary>
        /// 获取静态实例的静态方法
        /// </summary>
        /// <returns></returns>
        public static Good GetInstance()
        {
            return Instance;
        }
    }

    2、懒汉式

    上面使用饿汉式创建单例模式有一个缺点:如果程序不使用也会创建一个实例,这样也会占用一部分内存。有时候需要真正第一次用到的时候才去创建实例,这时候就需要使用懒汉式创建单例模式。

    class Good
    {
        /// <summary>
        /// 私有的静态变量
        /// </summary>
        private static Good Instance = null;
    
        /// <summary>
        /// 私有的构造函数
        /// </summary>
        private Good()
        {
    
        }
    
        /// <summary>
        /// 获取静态实例的静态方法
        /// </summary>
        /// <returns></returns>
        public static Good GetInstance()
        {
            if(Instance==null)
            {
                Instance = new Good();
            }
            return Instance;
        }
    }

    二、单例模式和多线程

    上面两种创建单例模式的方法,在单线程使用的时候都没有问题,饿汉式创建的单例模式在多线程使用时也没有问题,懒汉式方式创建的单例模式在多线程下就有问题了。那么该如何解决呢?

    可以在GetInstance方法上面添加[MethodImpl(MethodImplOptions.Synchronized)]标注,标注为同步方法。也可以使用lock关键字,我们看看一下如何使用lock关键字:

    class Good
        {
            /// <summary>
            /// 私有的静态变量
            /// </summary>
            private static Good Instance = null;
            private static object locker = new object();
            /// <summary>
            /// 私有的构造函数
            /// </summary>
            private Good()
            {
    
            }
    
            /// <summary>
            /// 获取静态实例的静态方法
            /// </summary>
            /// <returns></returns>
            public static Good GetInstance()
            {
                // 使用lock
                lock(locker)
                {
                    if (Instance == null)
                    {
                        Instance = new Good();
                    }
                    return Instance;
                }
            }
        }

    使用了lock关键字在多线程环境下就可以保证单例了。但是这样修改代码还是有问题,其实只有Instance为null的时候的那次加锁才是有意义的,以后的调用,每个线程都要锁定locker,就会造成性能下降。可以使用双重检查(double-check)解决性能问题。我们对上面的代码进行如下的改造;

    class Good
    {
        /// <summary>
        /// 私有的静态变量
        /// </summary>
        private static Good Instance = null;
        private static object locker = new object();
        /// <summary>
        /// 私有的构造函数
        /// </summary>
        private Good()
        {
    
        }
    
        /// <summary>
        /// 获取静态实例的静态方法
        /// </summary>
        /// <returns></returns>
        public static Good GetInstance()
        {
            // 先检查Instance变量是否为null
            if(Instance == null)
            {
                // 使用lock
                lock (locker)
                {
                    if (Instance == null)
                    {
                        Instance = new Good();
                    }                   
                }
            }
            return Instance;
        }
    }

    这样只有第一次初始化的时候才会加锁,以后在访问的时候,Instance变量已经不为null了,就直接返回Instance变量了。

  • 相关阅读:
    力扣(LeetCode) 14. 最长公共前缀
    力扣(LeetCode)965. 单值二叉树
    力扣(LeetCode)258. 各位相加
    力扣(LeetCode)389. 找不同
    PTA 阶乘之和取模
    A. Sea Battle
    PTA 二叉树路径
    PTA 重构二叉树
    PTA 笛卡尔树
    绿豆蛙的归宿
  • 原文地址:https://www.cnblogs.com/dotnet261010/p/12335702.html
Copyright © 2020-2023  润新知