如何避免在表示层使用数据访问层?

How to avoid using data access layer in presentation layer?

我有一个使用 Sqlite 进行 CRUD 操作的数据层,所以在我的 Sqlite Class 中我必须传递数据库的路径,为此我在数据层中有接口 IConnectionProvider 作为下面:

public interface IConnectionProvider
 {
        string DbPath { get; }
 }

SqliteDb.cs

public class SQLiteDb : DbContext, IDbContext
    {
        internal const string DefaultDatabaseName = "My.db";

        private string connectionString = null;

        public SQLiteDb(IConnectionProvider connectionProvider)
        {
            this.connectionString = connectionProvider.DbPath;
        }

        protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        {
            optionsBuilder.UseSqlite(connectionString);
        }
    }

我的项目架构如下:

 Data Layer -> View Model -> Presentation Layer

我想避免直接在表示层中使用数据层,但要传递 DBPath,我必须添加它的引用。

下面是IConnectionProvider在我的表示层的实现:

public class SqlConnectionProvider : IConnectionProvider
    {
        public string dbPath = string.Empty;
        public string DbPath
        {
            get
            {
                if (string.IsNullOrEmpty(dbPath))
                {
                    dbPath = Path.Combine(ApplicationData.Current.LocalFolder.Path, "MyDb.sqlite");
                }

                return dbPath;
            }
        }
  }

有什么方法可以避免直接在表示层中使用 DataLayer?我想在 VM 中创建包装器,但我认为这不是一个好主意,因为会创建更多的依赖项。

执行此操作的常用方法是使用存储库模式。在 ViewModel Layer 中定义一个 Interface,并在 DataLayer 中实现它。然后 ViewModel 对 DataLayer 一无所知。然后通过依赖注入在主 class 中使用 ConnectionProvider 实例化存储库。此外,dbContext 必须移动到 DataLayer,您应该在存储库中使用它。我建议将连接字符串放在配置文件中并在 DL 中使用它。 classes 应该看起来像这样

namespace ViewModel;
interface IDatabaseRepository {
   DataObject LoadData()
}

namespace DataLayer;
class DataRepository {
   public DataRepository(DbContext context) {
      this.context = context;
   }

   public DataObject LoadData() {
      //load data from DB using dbContext
   }
}

namespace ViewModel;
class ViewModel {
   public ViewModel(IDataRepository repository) {
      this.repository = repository;
   }

   // use the repository inside this class to acces data
}

另请参阅依赖倒置原则,因为这有助于解耦此类事情。