Нужно ли добавлять службу при запуске для всех DIS строки подключения?

#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
}