4

我正在尝试创建一个负责读取所有用户访问设置的对象。

我这样创建了类:

public class SettingsManager
{
    private static string _connString =
        @"Data Source=MyDB;Initial Catalog=IT;Integrated Security=True;Asynchronous Processing=true;";
    private const string _spName = "SettingTest";
    private IEnumerable<string> _mySettings;
    private static readonly Lazy<SettingsManager> _instance = new Lazy<SettingsManager>(() => new SettingsManager());

    private SettingsManager()
    {
        //Console.WriteLine("Hello from constructor");
        if (_mySettings == null)
            _mySettings = ReadSettings();
    }

    public static SettingsManager Instance
    {
        get { return _instance.Value; }
    }

    public bool CanDo(string setting)
    {
        return _mySettings.Contains(setting);
    }

    private IEnumerable<string> ReadSettings()
    {
        try
        {
            using (var conn = new SqlConnection(_connString))
            {
                using (var cmd = new SqlCommand())
                {
                    cmd.Connection = conn;
                    cmd.CommandText = _spName;
                    cmd.CommandType = CommandType.StoredProcedure;
                    conn.Open();
                    using (var reader = cmd.ExecuteReader())
                    {
                        return reader.Select(SettingParser).ToList();
                    }
                }
            }
        }
        catch
        {
        }
        return null;
    }

    private string SettingParser(SqlDataReader r)
    {
        try
        {
            return r[0] is DBNull ? null : r[0].ToString();
        }
        catch
        {
        }
        return null;
    }
}

和 SqlDataReader 扩展

public static class DBExtensions
{
    public static IEnumerable<T> Select<T>(
        this SqlDataReader reader, Func<SqlDataReader, T> projection)
    {
        while (reader.Read())
        {
            yield return projection(reader);
        }
    }
}

然后在我的应用程序中,我可以这样称呼它:

SettingsManager.Instance.CanDo("canWrite")

这返回真/假值取决于用户访问级别。

我的问题是:

  1. 这个线程安全吗?数据库是否有可能被多次查询?如何防止这种情况?

  2. 我应该使用等待和异步吗?我只查询 db 一次。我该如何改进呢?(await 和 async 对我来说真的很新,因为我刚从 .NET3.5 迁移到 4.5)

4

3 回答 3

2

1)这个线程安全吗?

是的。

数据库是否有可能被多次查询?

不。

2)我应该使用等待和异步吗?

这将取决于您是否需要对数据库进行异步访问。如果需要,可以使用异步 ADO.NET API。

于 2013-02-18T08:53:00.867 回答
1

1)是的,这是线程安全的

线程安全的 Singleton:此实现使用内部类使 .NET 实例化完全惰性。只有 GetInstance() 会触发类型初始化器;所以,这个版本和经典版本一样懒惰。而且,它将像任何其他版本一样快速运行。

public sealed class Singleton
{

   private Singleton() {}

   public static Singleton GetInstance()
   {
      return NestedSingleton.singleton;
   }

   class NestedSingleton
   {
      internal static readonly Singleton singleton = new Singleton();

      static NestedSingleton() {}
   }
}

2)这取决于您是否想要异步访问您的数据库。

于 2013-02-18T08:49:45.693 回答
1
  1. 这个线程安全吗?

是的。您正在Lazy<T>正确使用类型以确保线程安全。

  1. 我应该使用等待和异步吗?

我会推荐它。现在,当您的应用程序代码第一次调用SettingsManager.Instance.CanDo("canWrite")时,它将阻塞直到数据库响应。也可以有一个async你的消费者可以做(await SettingsManager.Instance).CanDo("canWrite")(或await SettingsManager.CanDoAsync("canWrite"))的实现。这意味着您的消费者在等待数据库时不会阻塞。

您可以使用被调用的async就绪版本,该版本最初由 Stephen Toub 开发包含在我的 AsyncEx 库中。Lazy<T>AsyncLazy<T>

于 2013-02-18T13:15:52.500 回答