如何避免在表示层使用数据访问层?
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
}
另请参阅依赖倒置原则,因为这有助于解耦此类事情。
我有一个使用 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
}
另请参阅依赖倒置原则,因为这有助于解耦此类事情。