在PowerShell中使用C#单例时出现问题

本文关键字:单例时 问题 PowerShell | 更新日期: 2023-09-27 18:20:41

当我试图将我的singleton类加载到PowerShell中时,我在让它协同工作方面遇到了一些困难。具体来说,当我尝试访问实例时,我会得到以下错误:

You cannot call a method on a null-valued expression.
At C:'Users'*Omitted*'Documents'visual studio
2012'Projects'ClassLibrary1'ClassLibrary1'bin'Debug'test.ps1:5 char:1
+ $test = [ClassLibrary1.Class1]::Instance.Foo()
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : InvalidOperation: (:) [], RuntimeException
    + FullyQualifiedErrorId : InvokeMethodOnNull

当我运行这个示例脚本时:

$currentScriptDirectory = Get-Location
[System.IO.Directory]::SetCurrentDirectory($currentScriptDirectory.Path)
[Reflection.Assembly]::LoadFrom("ClassLibrary1.dll") | fl
[ClassLibrary1.Class1] | Get-Member -Static
$test = [ClassLibrary1.Class1]::Instance.Foo()

关于这个简单的类:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace ClassLibrary1
{
    public class Class1
    {
        private static Class1 instance = null;
        private Class1()
        {
        }
        public static Class1 Instance
        {
            get
            {
                if (null == instance)
                {
                    instance = new Class1();
                }
                return instance;
            }
        }
        public string Foo()
        {
            return "HI THERE";
        }
    }
}

关于如何在不从单例更改类的情况下消除这个错误,有什么想法吗?我继承了与我一起工作的类,无法更改其架构。

在PowerShell中使用C#单例时出现问题

主要问题是混淆了静态和实例。静态表示它是类级别的,例如实例。Foo方法应该是instance,因为这似乎是您想要的。这是固定代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace ClassLibrary1
{
    public class Class1
    {
        private static Class1 instance = null;
        private Class1()
        {
        }
        public static Class1 Instance
        {
            get
            {
                if (null == instance)
                {
                    instance = new Class1();
                }
                return instance;
            }
        }
        public string Foo()
        {
            return "HI THERE";
        }
    }
}

然而,这并不是一个无懈可击的singleton模式。Jon Skeet做了一些不错的,在这里。

以下是它的正确实现:

public sealed class Class1
{
    private static readonly Lazy<Class1> lazy =
        new Lazy<Class1>(() => new Class1());
    public static Class1 Instance { get { return lazy.Value; } }
    private Class1()
    {
    }
    public string Foo()
    {
        return "HI THERE";
    }
}