#c# #dependency-injection #connection-string #nullreferenceexception
#c# #внедрение зависимостей #строка подключения #исключение nullreferenceexception
Вопрос:
Я использую базу данных в своем приложении, и поэтому мне нужен доступ к моей строке подключения, которая хранится в моем appsettings.json
файле. Кажется неуклюжим добавлять сервис несколько раз в мой Startup.cs
файл, где я каждый раз делаю одно и то же. Поскольку у меня есть строка подключения, которую необходимо использовать в классах, которые затем используются на страницах Blazor, я попытался создать строку подключения, предоставляющую сервис:
public class ConnectionStringService
{
public string DataDB { get; set; }
}
//In Startup.cs
services.Configure<ConnectionStringService>(Configuration.GetSection("ConnectionStrings"));
services.AddScoped<ConnectionStringService>();
//I don't want a billion services.AddScoped<>(), one for every time I want to use a connection string, because it would get messy.
И попытайтесь использовать его как:
public class SearchesTimeline
{
[Inject]
private ConnectionStringService connectionStrings { get; }
//private readonly ConnectionStringService connectionStrings;
//public SearchesTimeline(IOptions<ConnectionStringService> options)
//{
// connectionStrings = options.Value;
//}
public async Task<List<List<Timelineinfo>>> GetTimeline(string? current, int? dateChange, string? date)
{
if (current != null amp;amp; dateChange != null)
{
date = Convert.ToDateTime(current).AddDays((double)dateChange).ToString("yyyy-MM-dd");
}
//This line is being executed
else if (date == null | string.IsNullOrWhiteSpace(Extensions.GetQueryParm("d")))
{
date = DateTime.Today.ToString("yyyy-MM-dd");
}
else
{
date = Extensions.GetQueryParm("d");
}
for (int i = 0; i < 7; i )
{
//get data
DataAccess data = new DataAccess();
var query = "SELECT t.TimelineinfoId From timelineinfo t where t.date = " dates[i].ToString("yyyy-MM-dd") ";";
dateData[i] = await data.LoadData<Timelineinfo, dynamic>(query, new { }, connectionStrings.DataDB);
//Here, where I access connectionStrings.DataDB, I get the error
}
//more code stuff
}
}
//On the blazor page:
SearchesTimeline searches = new Searches.SearchesTimeline();
[Parameter]
public string? current { get; set; }
[Parameter]
public string? date { get; set; }
protected override async Task OnInitializedAsync()
{
await searches.GetTimeline(current, 0, date);
}
public async Task NextWeek()
{
await searches.GetTimeline(current, 7, date);
}
public async Task PrevWeek()
{
await searches.GetTimeline(current, 7, date);
}
Но когда я пытаюсь запустить это, я получаю исключение NullReferenceException: «Ссылка на объект не установлена для экземпляра объекта». Итак, этот код не работает.
Есть ли способ настроить одну службу для всех моих потребностей в строке подключения и использовать ее способом, аналогичным описанному выше?
Комментарии:
1. какая строка в коде вызывает исключение? Как присваивается значение свойству
DataDB
ConnectionStringService
класса?2. @ChetanRanpariya Я обновил свой вопрос дополнительной информацией
3. @ChetanRanpariya Присваивается значение с
services.Configure()
, которое извлекает информацию из моего файла appsettings.json
Ответ №1:
services.Configure<ConnectionStringService>(Configuration.GetSection("ConnectionStrings"));
После этой строки вам больше не нужно добавлять тип в свой DI. Вы можете получить значение, используя IOptions<> интерфейс. Вот так :
[Inject] private IOptions<ConnectionStringService> connectionStrings { get; }
и затем вы можете использовать ее, обратившись к свойству Value объекта:
connectionStrings.Value
Комментарии:
1. Это не работает, ConnectionStrings по-прежнему имеет значение null, и в любом случае это было бы не так
connectionStrings.Value.DataDB
?
Ответ №2:
В итоге я создал статическую функцию, которая возвращала строку подключения везде, где мне это было нужно. Я изменил свой Startup.cs
файл, чтобы содержать:
public static string ConnectionString { get; private set; }
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
//Add alongside whatever else you have in Configure()
ConnectionString = Configuration["ConnectionStrings:DataDB"];
}
Функция действительно проста, просто:
public static string GetConnectionString()
{
return Startup.ConnectionString; //returns DataDB connection string
}