Почему появляется ошибка “Object reference not set to an instance of an object” при создании проекта в VS 2019?

Нижняя линия

Вы пытаетесь использовать что-то, что есть null(или Nothingв VB.NET). Это означает, что вы либо устанавливаете его null, либо никогда не устанавливаете вообще ни на что.

Как и все остальное, nullпередается. Если null в методе «A», может быть , что метод «В» принят null к методу «A».

null могут иметь разные значения:

  1. Переменные объектов, которыеи , следовательно ,указывают на нет.В этом случае, если вы обращаетесь к членам таких объектов, это вызываетNullReferenceException

В оставшейся части этой статьи более подробно рассматриваются ошибки, которые часто допускают многие программисты, которые могут привести к возникновению файла NullReferenceException.

Более конкретно

runtimeМетании NullReferenceException всегда означает то же самое: вы пытаетесь использовать ссылку, и ссылка не инициализируется (или он был когда – то инициализируется, но не больше не инициализирован).

Это означает, что ссылка есть null, и вы не можете получить доступ к членам (например, методам) через nullссылку. Самый простой случай:

string foo = null;
foo.ToUpper();

Это вызовет появление NullReferenceExceptionво второй строке, потому что вы не можете вызвать метод экземпляра ToUpper()для stringссылки, указывающей на null.

Отладка

Как найти источник NullReferenceException? Помимо рассмотрения самого исключения, которое будет выброшено именно в том месте, где оно возникает, применяются общие правила отладки в Visual Studio: размещайте стратегические точки останова и проверяйте свои переменные , либо наведя указатель мыши на их имена, открыв ( Quick) Наблюдайте за окном или используйте различные панели отладки, такие как Locals и Autos.

Если вы хотите узнать, где установлена ​​или не установлена ​​ссылка, щелкните ее имя правой кнопкой мыши и выберите «Найти все ссылки». Затем вы можете разместить точку останова в каждом найденном месте и запустить свою программу с присоединенным отладчиком. Каждый раз, когда отладчик прерывается в такой точке останова, вам необходимо определить, ожидаете ли вы, что ссылка будет ненулевой, проверить переменную и убедиться, что она указывает на экземпляр, когда вы этого ожидаете.

Следуя таким образом за потоком программы, вы можете найти место, где экземпляр не должен иметь значение NULL и почему он не установлен должным образом.

Примеры

Некоторые распространенные сценарии, в которых может быть создано исключение:

Общий

ref1.ref2.ref3.member

Если ref1, ref2 или ref3 имеет значение null, вы получите файл NullReferenceException. Если вы хотите решить проблему, выясните, какое из них является нулевым, переписав выражение в его более простой эквивалент:

var r1 = ref1;
var r2 = r1.ref2;
var r3 = r2.ref3;
r3.member

В частности, в HttpContext.Current.User.Identity.Name, то HttpContext.Currentможет быть пустым, или Userсвойство может быть пустым, или Identityсвойство может быть пустым.

Косвенный

public class Person 
{
    public int Age { get; set; }
}
public class Book 
{
    public Person Author { get; set; }
}
public class Example 
{
    public void Foo() 
    {
        Book b1 = new Book();
        int authorAge = b1.Author.Age; // You never initialized the Author property.
                                       // there is no Person to get an Age from.
    }
}

Если вы хотите избежать пустой ссылки на дочерний (Person) объект, вы можете инициализировать его в конструкторе родительского (Book) объекта.

Инициализаторы вложенных объектов

То же самое относится к инициализаторам вложенных объектов:

Book b1 = new Book 
{ 
   Author = { Age = 45 } 
};

Это означает:

Book b1 = new Book();
b1.Author.Age = 45;

Пока используется newключевое слово, оно создает только новый экземпляр Book, но не новый экземпляр Person, поэтому Authorсвойство остается null.

Вложенные инициализаторы коллекций

public class Person 
{
    public ICollection<Book> Books { get; set; }
}
public class Book 
{
    public string Title { get; set; }
}

Вложенная коллекция Initializersведет себя так же:

Person p1 = new Person 
{
    Books = {
         new Book { Title = "Title1" },
         new Book { Title = "Title2" },
    }
};

Это означает:

Person p1 = new Person();
p1.Books.Add(new Book { Title = "Title1" });
p1.Books.Add(new Book { Title = "Title2" });

new PersonТолько создает экземпляр Person, но Booksколлекция до сих пор null. InitializerСинтаксис коллекции не создает коллекцию для p1.Books, он только преобразуется в p1.Books.Add(...)операторы.

Множество

int[] numbers = null;
int n = numbers[0]; // numbers is null. There is no array to index.

Элементы массива

Person[] people = new Person[5];
people[0].Age = 20 // people[0] is null. The array was allocated but not
                   // initialized. There is no Person to set the Age for.

Неровные массивы

long[][] array = new long[1][];
array[0][0] = 3; // is null because only the first dimension is yet initialized.
                 // Use array[0] = new long[2]; first.

Сборник / Список / Словарь

Dictionary<string, int> agesForNames = null;
int age = agesForNames["Bob"]; // agesForNames is null.
                               // There is no Dictionary to perform the lookup.

Переменная диапазона (косвенная / отложенная)

public class Person 
{
    public string Name { get; set; }
}
var people = new List<Person>();
people.Add(null);
var names = from p in people select p.Name;
string firstName = names.First(); // Exception is thrown here, but actually occurs
                                  // on the line above.  "p" is null because the
                                  // first element we added to the list is null.

События (C #)

public class Demo
{
    public event EventHandler StateChanged;
    
    protected virtual void OnStateChanged(EventArgs e)
    {        
        StateChanged(this, e); // Exception is thrown here 
                               // if no event handlers have been attached
                               // to StateChanged event
    }
}

(Примечание: компилятор VB.NET вставляет нулевые проверки для использования событий, поэтому нет необходимости проверять события Nothingв VB.NET.)

Неправильные соглашения об именах:

Если вы назвали поля иначе, чем локальные, вы могли понять, что никогда не инициализировали поле.

public class Form1
{
    private Customer customer;
    
    private void Form1_Load(object sender, EventArgs e) 
    {
        Customer customer = new Customer();
        customer.Name = "John";
    }
    
    private void Button_Click(object sender, EventArgs e)
    {
        MessageBox.Show(customer.Name);
    }
}

Это можно решить, следуя соглашению о добавлении к полям префикса подчеркивания:

    private Customer _customer;

Жизненный цикл страницы ASP. NET:

public partial class Issues_Edit : System.Web.UI.Page
{
    protected TestIssue myIssue;

    protected void Page_Load(object sender, EventArgs e)
    {
        if (!IsPostBack)
        {
             // Only called on first load, not when button clicked
             myIssue = new TestIssue(); 
        }
    }
        
    protected void SaveButton_Click(object sender, EventArgs e)
    {
        myIssue.Entry = "NullReferenceException here!";
    }
}

Значения сеанса ASP. NET

// if the "FirstName" session value has not yet been set,
// then this line will throw a NullReferenceException
string firstName = Session["FirstName"].ToString();

Модели пустого представления ASP. NET MVC

Если исключение возникает при обращении к свойству @Modelв ASP.NET MVC View, вы должны понимать , что Modelполучает набор в методе действия, когда вы returnвид. Когда вы возвращаете пустую модель (или свойство модели) из вашего контроллера, исключение возникает, когда представления обращаются к нему:

// Controller
public class Restaurant:Controller
{
    public ActionResult Search()
    {
        return View();  // Forgot the provide a Model here.
    }
}

// Razor view 
@foreach (var restaurantSearch in Model.RestaurantSearch)  // Throws.
{
}
    
<p>@Model.somePropertyName</p> <!-- Also throws -->

Порядок и события создания элемента управления WPF

WPFэлементы управления создаются во время вызова InitializeComponentв том порядке, в котором они появляются в визуальном дереве. A NullReferenceExceptionбудет возникать в случае ранее созданных элементов управления с обработчиками событий и т. Д., Которые срабатывают, во время InitializeComponentкоторых ссылаются на поздно созданные элементы управления.

Например:

<Grid>
    <!-- Combobox declared first -->
    <ComboBox Name="comboBox1" 
              Margin="10"
              SelectedIndex="0" 
              SelectionChanged="comboBox1_SelectionChanged">
       <ComboBoxItem Content="Item 1" />
       <ComboBoxItem Content="Item 2" />
       <ComboBoxItem Content="Item 3" />
    </ComboBox>
        
    <!-- Label declared later -->
    <Label Name="label1" 
           Content="Label"
           Margin="10" />
</Grid>

Вот comboBox1создан раньше label1. Если comboBox1_SelectionChangedпопытается сослаться на `label1, он еще не будет создан.

private void comboBox1_SelectionChanged(object sender, SelectionChangedEventArgs e)
{
    label1.Content = comboBox1.SelectedIndex.ToString(); // NullReferenceException here!!
}

Изменение порядка объявлений в XAML(т. Е. Перечисление label1ранее comboBox1, игнорирование вопросов философии дизайна), по крайней мере, решило бы проблему NullReferenceExceptionздесь.

В ролях с as

var myThing = someObject as Thing;

Это не генерирует, InvalidCastExceptionа возвращает, nullкогда приведение не выполняется (и когда someObjectсамо значение равно нулю). Так что имейте это в виду.

LINQ FirstOrDefault()иSingleOrDefault()

Простые версии First()и Single()выдают исключения, когда ничего нет. В nullэтом случае возвращаются версии “OrDefault” . Так что имейте это в виду.

для каждого

foreachвыбрасывает, когда вы пытаетесь выполнить итерацию nullколлекции. Обычно вызывается неожиданным nullрезультатом методов, возвращающих коллекции.

List<int> list = null;    
foreach(var v in list) { } // NullReferenceException here

Более реалистичный пример – выбор узлов из XML-документа. Будет выброшено, если узлы не найдены, но первоначальная отладка показывает, что все свойства действительны:

foreach (var node in myData.MyXml.DocumentNode.SelectNodes("//Data"))

Способы избежать

Явно проверяйте nullи игнорируйте nullзначения.

Если вы ожидаете, что ссылка иногда будет null, вы можете проверить ее nullперед доступом к членам экземпляра:

void PrintName(Person p)
{
    if (p != null) 
    {
        Console.WriteLine(p.Name);
    }
}

Явно проверьте nullи укажите значение по умолчанию.

Методы, которые вы вызываете в ожидании возврата экземпляра, могут вернуться null, например, когда искомый объект не может быть найден. В этом случае вы можете выбрать возврат значения по умолчанию:

string GetCategory(Book b) 
{
    if (b == null)
        return "Unknown";
    return b.Category;
}

Явно проверяйте nullвызовы метода from и генерируйте настраиваемое исключение.

Вы также можете создать собственное исключение, только чтобы поймать его в вызывающем коде:

string GetCategory(string bookTitle) 
{
    var book = library.FindBook(bookTitle);  // This may return null
    if (book == null)
        throw new BookNotFoundException(bookTitle);  // Your custom exception
    return book.Category;
}

Используйте, Debug. Assertесли значение никогда не должно быть null, чтобы выявить проблему раньше, чем возникнет исключение.

Когда вы знаете во время разработки, что метод может, но никогда не должен возвращаться null, вы можете использовать Debug.Assert()команду break как можно скорее, когда это произойдет:

string GetTitle(int knownBookID) 
{
    // You know this should never return null.
    var book = library.GetBook(knownBookID);  

    // Exception will occur on the next line instead of at the end of this method.
    Debug.Assert(book != null, "Library didn't return a book for known book ID.");

    // Some other code

    return book.Title; // Will never throw NullReferenceException in Debug mode.
}

Хотя эта проверка не завершится в вашей сборке выпуска , из-за чего она NullReferenceExceptionснова вызовет проверку book == nullво время выполнения в режиме выпуска.

Читайте также:  Количество попыток ввода правильного Pin-кода достигнуто из-за ошибки 0x8010006c, которая недоступна для карты

Используйте GetValueOrDefault()для nullableтипов значений, чтобы предоставить значение по умолчанию, когда они есть null.

DateTime? appointment = null;
Console.WriteLine(appointment.GetValueOrDefault(DateTime.Now));
// Will display the default value provided (DateTime.Now), because appointment is null.

appointment = new DateTime(2022, 10, 20);
Console.WriteLine(appointment.GetValueOrDefault(DateTime.Now));
// Will display the appointment date, not the default

Используйте нулевой оператор объединения: ??[C #] или If()[VB].

Сокращение для предоставления значения по умолчанию, когда nullвстречается:

IService CreateService(ILogger log, Int32? frobPowerLevel)
{
   var serviceImpl = new MyService(log ?? NullLog.Instance);
 
   // Note that the above "GetValueOrDefault()" can also be rewritten to use
   // the coalesce operator:
   serviceImpl.FrobPowerLevel = frobPowerLevel ?? 5;
}

Используйте оператор нулевого условия: ?. или ?[x]для массивов (доступно в C # 6 и VB. NET 14):

Это также иногда называют оператором безопасной навигации или оператором Элвиса (по форме). Если выражение в левой части оператора имеет значение NULL, тогда правая часть не будет оцениваться, и вместо этого возвращается значение NULL. Это означает такие случаи:

var title = person.Title.ToUpper();

Если у человека нет заголовка, это вызовет исключение, потому что он пытается вызвать ToUpperсвойство с нулевым значением.

В C# 5и ниже, это может быть настораживают:

var title = person.Title == null ? null : person.Title.ToUpper();

Теперь переменная заголовка будет иметь значение null, а не вызывать исключение. В C # 6 для этого используется более короткий синтаксис:

var title = person.Title?.ToUpper();

Это приведет к тому, что переменная заголовка будет существовать null, и вызов ToUpperне будет выполнен, если она person.Titleесть null.

Конечно, вы все равно должны проверить titleна nullили использовать оператор нуля условия вместе с оператором нулевой коалесценции ( ??) указать значение по умолчанию:

// regular null check
int titleLength = 0;
if (title != null)
    titleLength = title.Length; // If title is null, this would throw NullReferenceException
    
// combining the `?` and the `??` operator
int titleLength = title?.Length ?? 0;

Точно так же для массивов вы можете использовать ?[i]следующее:

int[] myIntArray = null;
var i = 5;
int? elem = myIntArray?[i];
if (!elem.HasValue) Console.WriteLine("No value");

Это сделает следующее: Если myIntArrayесть null, выражение вернется, nullи вы можете безопасно его проверить. Если он содержит массив, он будет делать то же самое, что и:
elem = myIntArray[i];и возвращает i- й элемент.

Использовать нулевой контекст (доступно в C # 8):

Введенные в C# 8контексты NULL и ссылочные типы, допускающие значение NULL, выполняют статический анализ переменных и выдают предупреждение компилятора, если значение потенциально может быть nullили было установлено на null. Ссылочные типы, допускающие значение NULL, позволяют явно указать типы null.

Контекст аннотации, допускающей значение NULL, и контекст предупреждения, допускающего значение NULL, можно установить для проекта с помощью Nullableэлемента в csprojфайле. Этот элемент настраивает, как компилятор интерпретирует допустимость значений NULL для типов и какие предупреждения генерируются. Допустимые настройки:

  • enable: Контекст аннотации, допускающей значение NULL, включен. Контекст предупреждения, допускающего значение NULL, включен. Переменные ссылочного типа, например строка, не допускают значения NULL. Все предупреждения о допустимости пустых значений включены.
  • disable: Контекст аннотации, допускающей значение NULL, отключен. Контекст предупреждения, допускающего значение NULL, отключен. Переменные ссылочного типа не обращают внимания, как и более ранние версии C #. Все предупреждения о допустимости пустых значений отключены.
  • safeonly: Контекст аннотации, допускающей значение NULL, включен. Контекст предупреждения, допускающего значение NULL, является безопасным. Переменные ссылочного типа не допускают значения NULL. Включены все предупреждения о допустимости обнуления.
  • warnings: Контекст аннотации, допускающей значение NULL, отключен. Контекст предупреждения, допускающего значение NULL, включен. Переменные ссылочного типа не обращают внимания. Все предупреждения о допустимости пустых значений включены.
  • safeonlywarnings: Контекст аннотации, допускающей значение NULL, отключен. Контекст предупреждения, допускающего значение NULL, является безопасным. Переменные ссылочного типа не обращают внимания. Включены все предупреждения о допустимости обнуления.

Ссылочный тип, допускающий значение NULL, отмечается с использованием того же синтаксиса, что и типы значений, допускающие значение NULL: a ?добавляется к типу переменной.

Специальные методы отладки и исправления нулевых ссылок в итераторах

C#поддерживает «блоки итераторов» (называемые «генераторами» в некоторых других популярных языках). NullReferenceExceptionможет быть особенно сложно отладить в блоках итератора из-за отложенного выполнения:

public IEnumerable<Frob> GetFrobs(FrobFactory f, int count)
{
    for (int i = 0; i < count; ++i)
    yield return f.MakeFrob();
}
...
FrobFactory factory = whatever;
IEnumerable<Frobs> frobs = GetFrobs();
...
foreach(Frob frob in frobs) { ... }

Если whateverрезультаты в nullто MakeFrobвыбросит. Теперь вы можете подумать, что правильно сделать следующее:

// DON'T DO THIS
public IEnumerable<Frob> GetFrobs(FrobFactory f, int count)
{
   if (f == null) 
      throw new ArgumentNullException("f", "factory must not be null");
   for (int i = 0; i < count; ++i)
      yield return f.MakeFrob();
}

Почему это не так? Поскольку блок итератора фактически не запускается до тех пор, пока не будет foreach! Вызов GetFrobsпросто возвращает объект, который при итерации запускает блок итератора.

Написав такую nullпроверку, вы предотвратите NullReferenceException, но вы переместите ее NullArgumentExceptionв точку итерации , а не в точку вызова , и это очень сбивает с толку при отладке .

Правильное исправление:

// DO THIS
public IEnumerable<Frob> GetFrobs(FrobFactory f, int count)
{
   // No yields in a public method that throws!
   if (f == null) 
       throw new ArgumentNullException("f", "factory must not be null");
   return GetFrobsForReal(f, count);
}
private IEnumerable<Frob> GetFrobsForReal(FrobFactory f, int count)
{
   // Yields in a private method
   Debug.Assert(f != null);
   for (int i = 0; i < count; ++i)
        yield return f.MakeFrob();
}

То есть создайте частный вспомогательный метод, который имеет логику блока итератора и общедоступный поверхностный метод, который выполняет nullпроверку и возвращает итератор. Теперь, когда GetFrobsвызывается, nullпроверка выполняется немедленно, а затем GetFrobsForRealвыполняется при повторении последовательности.

Если вы изучите справочный источник для LINQобъектов, вы увидите, что этот метод используется повсюду. Его писать немного сложнее, но он значительно упрощает отладку ошибок, связанных с нулевым значением. Оптимизируйте свой код для удобства вызывающего абонента, а не для удобства автора .

Замечание о разыменовании null в небезопасном коде

C#имеет «небезопасный» режим, который, как следует из названия, чрезвычайно опасен, потому что обычные механизмы безопасности, которые обеспечивают безопасность памяти и безопасность типов, не применяются. Вы не должны писать небезопасный код, если у вас нет полного и глубокого понимания того, как работает память .

В небезопасном режиме вы должны знать два важных факта:

  • вызывает то же исключение, что и разыменование нулевой
  • разыменование недопустимого ненулевого указателяв некоторых случаях вызвать это исключение.

Чтобы понять, почему это так, полезно NullReferenceExceptionсначала понять, как .NET производит . (Эти сведения относятся к .NET, работающему в Windows; другие операционные системы используют аналогичные механизмы.)

Память виртуализирована в Windows; каждый процесс получает пространство виртуальной памяти из множества «страниц» памяти, которые отслеживаются операционной системой. На каждой странице памяти установлены флаги, которые определяют, как ее можно использовать: чтение, запись, выполнение и т. Д. Самая нижняя страница помечена как «выдает ошибку, если когда-либо использовалась каким-либо образом».

И нулевой указатель, и нулевая ссылка в C#внутри представляются как число ноль, и поэтому любая попытка разыменовать его в соответствующее хранилище памяти приводит к тому, что операционная система выдает ошибку. Затем среда выполнения .NET обнаруживает эту ошибку и превращает ее в файл NullReferenceException.

Вот почему разыменование как нулевого указателя, так и нулевой ссылки вызывает одно и то же исключение.

А как насчет второго пункта? Разыменование любого недопустимого указателя, который попадает на самую нижнюю страницу виртуальной памяти, вызывает ту же ошибку операционной системы и, следовательно, такое же исключение.

Почему это имеет смысл? Что ж, предположим, что у нас есть структура, содержащая два целых числа и неуправляемый указатель, равный нулю. Если мы попытаемся разыменовать второй int в структуре, CLRне будет пытаться получить доступ к хранилищу в нулевом местоположении; он получит доступ к хранилищу в четвертом месте. Но логически это разыменование null, потому что мы попадаем на этот адрес через null.

Читайте также:  Построить код хэмминга для комбинации 101010 показать процесс исправления и обнаружения ошибки

Если вы работаете с небезопасным кодом и получаете NullReferenceException, просто имейте в виду, что указатель-нарушитель не обязательно должен быть нулевым. Это может быть любое место на самой нижней странице, и будет создано это исключение.

55 оборотов, 39 пользователей 26% LopDev

7

Объекты класса / Создание экземпляра

Dim reg As CashRegister
...
TextBox1.Text = reg.Amount         ' NRE

Проблема в том, что Dimне создается объект CashRegister ; он только объявляет переменную с именем regэтого типа. Объявление объектной переменной и создание экземпляра – это разные вещи.

Средство

NewОператор часто может быть использован для создания экземпляра при объявлении его:

Dim reg As New CashRegister        ' [New] creates instance, invokes the constructor

' Longer, more explicit form:
Dim reg As CashRegister = New CashRegister

Когда уместно создать экземпляр позже:

Private reg As CashRegister         ' Declare
  ...
reg = New CashRegister()            ' Create instance

Примечание: не используйте Dimснова в процедуре, включая конструктор ( Sub New):

Private reg As CashRegister
'...

Public Sub New()
   '...
   Dim reg As New CashRegister
End Sub

Это создаст локальную переменную, regкоторая существует только в этом контексте (под). regПеременной с уровнем модуля , Scopeкоторый вы будете использовать в любом другом месте остается Nothing.

Missing the New operator is the #1 cause of NullReference Exceptions seen in the Stack  Overflow questions reviewed.

Visual Basic tries to make the process clear repeatedly using New: Using the New Operator creates a new object and calls Sub New — the constructor — where your object can perform any other initialization.

Для ясности: Dim(или Private) объявляет только переменную и ее Type. Объем переменной – существует ли она для всего модуля / класса или является локальным для процедуры – определяется , где она объявлена. Private | Friend | Publicопределяет уровень доступа, а не Scope .

Для получения дополнительной информации см .:

  • Новый оператор
  • Область в Visual Basic
  • Уровни доступа в Visual Basic
  • Типы значений и ссылочные типы

Массивы

Также необходимо создать экземпляры массивов:

Private arr as String()

Этот массив только был объявлен, но не создан. Есть несколько способов инициализировать массив:

Private arr as String() = New String(10){}
' or
Private arr() As String = New String(10){}

' For a local array (in a procedure) and using 'Option Infer':
Dim arr = New String(10) {}

Примечание: Начиная с VS 2010, при инициализации локальный массив с использованием буквальным и Option Infer, то As и Newэлементы не являются обязательными:

Dim myDbl As Double() = {1.5, 2, 9.9, 18, 3.14}
Dim myDbl = New Double() {1.5, 2, 9.9, 18, 3.14}
Dim myDbl() = {1.5, 2, 9.9, 18, 3.14}

Тип данных и размер массива выводятся из присваиваемых данных. Заявления на уровне класса / модуля по- прежнему требуют As с Option Strict:

Private myDoubles As Double() = {1.5, 2, 9.9, 18, 3.14}

Пример: массив объектов класса

Dim arrFoo(5) As Foo

For i As Integer = 0 To arrFoo.Count - 1
   arrFoo(i).Bar = i * 10       ' Exception
Next

Массив создан, а Fooобъекты в нем – нет.

Средство

For i As Integer = 0 To arrFoo.Count - 1
    arrFoo(i) = New Foo()         ' Create Foo instance
    arrFoo(i).Bar = i * 10
Next

Использование List(Of T)will сделает довольно трудным наличие элемента без действительного объекта:

Dim FooList As New List(Of Foo)     ' List created, but it is empty
Dim f As Foo                        ' Temporary variable for the loop

For i As Integer = 0 To 5
    f = New Foo()                    ' Foo instance created
    f.Bar =  i * 10
    FooList.Add(f)                   ' Foo object added to list
Next

Для получения дополнительной информации см .:

  • Заявление о выводе варианта
  • Область в Visual Basic
  • Массивы в Visual Basic

Списки и коллекции

Коллекции .NET (которых существует множество разновидностей – списки, словарь и т. Д.) Также должны быть созданы или созданы.

Private myList As List(Of String)
..
myList.Add("ziggy")           ' NullReference

Вы получаете такое же исключение по той же причине – myListбыло только объявлено, но экземпляр не создан. Лекарство то же самое:

myList = New List(Of String)

' Or create an instance when declared:
Private myList As New List(Of String)

Распространенный надзор – это класс, который использует коллекцию Type:

Public Class Foo
    Private barList As List(Of Bar)

    Friend Function BarCount As Integer
        Return barList.Count
    End Function

    Friend Sub AddItem(newBar As Bar)
        If barList.Contains(newBar) = False Then
            barList.Add(newBar)
        End If
    End Function

Любая процедура приведет к NRE, потому что barListона только объявляется, а не создается. Создание экземпляра Fooтакже не приведет к созданию экземпляра внутреннего barList. Возможно, это было намерением сделать это в конструкторе:

Public Sub New         ' Constructor
    ' Stuff to do when a new Foo is created...
    barList = New List(Of Bar)
End Sub

Как и раньше, это неверно:

Public Sub New()
    ' Creates another barList local to this procedure
     Dim barList As New List(Of Bar)
End Sub

Для получения дополнительной информации см. List(Of T)Класс .


Объекты поставщика данных

Работа с базами данных представляет много возможностей для NullReference , потому что может быть много объектов ( Command, Connection, Transaction, Dataset, DataTable, DataRows….) в использовании сразу. Примечание. Не имеет значения, какой поставщик данных вы используете – MySQL, SQL Server, OleDB и т. Д. – концепции одинаковы.

Пример 1

Dim da As OleDbDataAdapter
Dim ds As DataSet
Dim MaxRows As Integer

con.Open()
Dim sql = "SELECT * FROM tblfoobar_List"
da = New OleDbDataAdapter(sql, con)
da.Fill(ds, "foobar")
con.Close()

MaxRows = ds.Tables("foobar").Rows.Count      ' Error

Как и раньше, dsобъект набора данных был объявлен, но экземпляр так и не был создан. Он DataAdapterзаполнит существующий DataSet, а не создаст его. В этом случае, поскольку dsэто локальная переменная, среда IDE предупреждает вас, что это может произойти:

img

Когда он объявлен как переменная уровня модуля / класса, как кажется con, компилятор не может знать, был ли объект создан процедурой восходящего потока. Не игнорируйте предупреждения.

Средство

Dim ds As New DataSet

Пример 2

ds = New DataSet
da = New OleDBDataAdapter(sql, con)
da.Fill(ds, "Employees")

txtID.Text = ds.Tables("Employee").Rows(0).Item(1)
txtID.Name = ds.Tables("Employee").Rows(0).Item(2)

Опечатка проблема здесь: Employeesпротив Employee. Имя DataTable«Сотрудник» не создано, поэтому NullReferenceExceptionрезультаты пытаются получить к нему доступ. Другая потенциальная проблема заключается в предположении, что это Itemsможет быть не так, если SQL включает предложение WHERE.

Средство

Поскольку здесь используется одна таблица, использование Tables(0)позволит избежать орфографических ошибок. Осмотр Rows.Countтакже может помочь:

If ds.Tables(0).Rows.Count > 0 Then
    txtID.Text = ds.Tables(0).Rows(0).Item(1)
    txtID.Name = ds.Tables(0).Rows(0).Item(2)
End If

Fill- это функция, возвращающая количество Rowsпострадавших, которое также можно протестировать:

If da.Fill(ds, "Employees") > 0 Then...

Пример 3

Dim da As New OleDb.OleDbDataAdapter("SELECT TICKET.TICKET_NO,
        TICKET.CUSTOMER_ID, ... FROM TICKET_RESERVATION AS TICKET INNER JOIN
        FLIGHT_DETAILS AS FLIGHT ... WHERE [TICKET.TICKET_NO]= ...", con)
Dim ds As New DataSet
da.Fill(ds)

If ds.Tables("TICKET_RESERVATION").Rows.Count > 0 Then

Он DataAdapterпредоставит, TableNamesкак показано в предыдущем примере, но не анализирует имена из таблицы SQL или базы данных. В результате ds.Tables(“TICKET_RESERVATION”)ссылается на несуществующую таблицу.

Устранение такой же, ссылки на таблицу с помощью индекса:

If ds.Tables(0).Rows.Count > 0 Then

См. Также Класс DataTable .


Пути к объектам / Вложенные

If myFoo.Bar.Items IsNot Nothing Then
   ...

Код только тестирует, в Itemsто время как оба myFooи Barтакже могут быть Nothing. Средство , чтобы проверить всю цепочку или путь объектов по одному за раз:

If (myFoo IsNot Nothing) AndAlso
    (myFoo.Bar IsNot Nothing) AndAlso
    (myFoo.Bar.Items IsNot Nothing) Then
    ....

AndAlsoявляется важным. Последующие тесты не будут выполняться при обнаружении первого Falseусловия. Это позволяет коду безопасно «углубляться» в объект (ы) по одному «уровню» за раз, оценивая его myFoo.Barтолько после того, как (и если) myFooбудет определено, что он действительный. Цепочки или пути объектов могут быть довольно длинными при кодировании сложных объектов:

myBase.myNodes(3).Layer.SubLayer.Foo.Files.Add("somefilename")

Невозможно сослаться на что-либо «ниже по течению» от nullобъекта. Это также относится к элементам управления:

myWebBrowser.Document.GetElementById("formfld1").InnerText = "some value"

Здесь myWebBrowserили Documentможет быть Ничто, или formfld1элемент может не существовать.


Элементы управления пользовательского интерфейса

Dim cmd5 As New SqlCommand("select Cartons, Pieces, Foobar " _
     & "FROM Invoice where invoice_no = '" & _
     Me.ComboBox5.SelectedItem.ToString.Trim & "' And category = '" & _
     Me.ListBox1.SelectedItem.ToString.Trim & "' And item_name = '" & _
     Me.ComboBox2.SelectedValue.ToString.Trim & "' And expiry_date = '" & _
     Me.expiry.Text & "'", con)

Помимо прочего, этот код не предполагает, что пользователь не выбрал что-то в одном или нескольких элементах управления пользовательского интерфейса. ListBox1.SelectedItemвполне может быть Nothing, поэтому ListBox1.SelectedItem.ToStringприведет к NRE.

Средство

Проверьте данные перед их использованием (также используйте Option Strictпараметры и SQL):

Dim expiry As DateTime         ' for text date validation
If (ComboBox5.SelectedItems.Count > 0) AndAlso
    (ListBox1.SelectedItems.Count > 0) AndAlso
    (ComboBox2.SelectedItems.Count > 0) AndAlso
    (DateTime.TryParse(expiry.Text, expiry) Then

    '... do stuff
Else
    MessageBox.Show(...error message...)
End If

В качестве альтернативы вы можете использовать (ComboBox5.SelectedItem IsNot Nothing) AndAlso...


Формы Visual Basic

Public Class Form1

    Private NameBoxes = New TextBox(5) {Controls("TextBox1"), _
                   Controls("TextBox2"), Controls("TextBox3"), _
                   Controls("TextBox4"), Controls("TextBox5"), _
                   Controls("TextBox6")}

    ' same thing in a different format:
    Private boxList As New List(Of TextBox) From {TextBox1, TextBox2, TextBox3 ...}

    ' Immediate NRE:
    Private somevar As String = Me.Controls("TextBox1").Text

Это довольно распространенный способ получить NRE. В C #, в зависимости от того, как он закодирован, IDE сообщит, что Controlsне существует в текущем контексте, или «не может ссылаться на нестатический член». Так что, в некоторой степени, это ситуация только для VB. Это также сложно, потому что может привести к каскаду отказов.

Читайте также:  Значение, причины и решения

Таким образом нельзя инициализировать массивы и коллекции. Этот код инициализации будет запущен до того, как конструктор создаст Formили Controls. Как результат:

  • Списки и Коллекция просто будут пустыми
  • Массив будет содержать пять элементов Nothing
  • somevarНазначение приведет к немедленному ЯРДУ , потому что ничего не имеет.Text

Ссылка на элементы массива позже приведет к NRE. Если вы сделаете это в Form_Load, из-за нечетной ошибки, IDE может не сообщить об исключении, когда это произойдет. Исключение появится позже, когда ваш код попытается использовать массив. Это «тихое исключение» подробно описано в этом посте . Для наших целей ключевым моментом является то, что когда что-то катастрофическое происходит при создании формы ( Sub Newили Form Loadсобытия), исключения могут не сообщаться, код выходит из процедуры и просто отображает форму.

Поскольку никакой другой код в вашем событии Sub Newили не Form Loadбудет запускаться после NRE, многие другие вещи можно оставить неинициализированными.

Sub Form_Load(..._
   '...
   Dim name As String = NameBoxes(2).Text        ' NRE
   ' ...
   ' More code (which will likely not be executed)
   ' ...
End Sub

Обратите внимание, что это относится ко всем без исключения ссылкам на элементы управления и компоненты, что делает их незаконными, если они есть:

Public Class Form1

    Private myFiles() As String = Me.OpenFileDialog1.FileName & ...
    Private dbcon As String = OpenFileDialog1.FileName & ";Jet Oledb..."
    Private studentName As String = TextBox13.Text

Частичное средство правовой защиты

Любопытно, что VB не выдает предупреждений, но решение состоит в том, чтобы объявить контейнеры на уровне формы, но инициализировать их в обработчике событий загрузки формы, когда элементы управления действительно существуют. Это можно сделать, Sub Newесли ваш код находится после InitializeComponentвызова:

' Module level declaration
Private NameBoxes as TextBox()
Private studentName As String

' Form Load, Form Shown or Sub New:
'
' Using the OP's approach (illegal using OPTION STRICT)
NameBoxes = New TextBox() {Me.Controls("TextBox1"), Me.Controls("TestBox2"), ...)
studentName = TextBox32.Text           ' For simple control references

Код массива может еще не выйти из леса. Любые элементы управления, которые находятся в элементе управления контейнера (например, GroupBoxили Panel), не будут найдены в Me.Controls; они будут в коллекции Controls этой Panel или GroupBox. Также не будет возвращен элемент управления, если имя элемента управления написано с ошибкой ( “TeStBox2”). В таких случаях Nothingон снова будет сохранен в этих элементах массива, и при попытке ссылки на него будет получен NRE.

Теперь, когда вы знаете, что ищете, их должно быть легко найти:

«Button2» находится на Panel

Средство

Вместо косвенных ссылок по имени с использованием Controlsколлекции формы используйте ссылку на элемент управления:

' Declaration
Private NameBoxes As TextBox()

' Initialization -  simple and easy to read, hard to botch:
NameBoxes = New TextBox() {TextBox1, TextBox2, ...)

' Initialize a List
NamesList = New List(Of TextBox)({TextBox1, TextBox2, TextBox3...})
' or
NamesList = New List(Of TextBox)
NamesList.AddRange({TextBox1, TextBox2, TextBox3...})

Функция ничего не возвращает

Private bars As New List(Of Bars)        ' Declared and created

Public Function BarList() As List(Of Bars)
    bars.Clear
    If someCondition Then
        For n As Integer = 0 to someValue
            bars.Add(GetBar(n))
        Next n
    Else
        Exit Function
    End If

    Return bars
End Function

Это тот случай, когда IDE предупредит вас, что « не все пути возвращают значение, и в NullReferenceExceptionрезультате может появиться a ». Вы можете подавить предупреждение, заменив его Exit Functionна Return Nothing, но это не решит проблему. Все, что пытается использовать return when, someCondition = Falseприведет к NRE:

bList = myFoo.BarList()
For Each b As Bar in bList      ' EXCEPTION
      ...

Средство

Заменить Exit Functionв функции на Return bList. Возвращение пустого List – это не то же самое, что возвращение Nothing. Если есть вероятность, что возвращенный объект может быть Nothing, проверьте перед его использованием:

 bList = myFoo.BarList()
 If bList IsNot Nothing Then...

Плохо реализованная попытка / поймать

Плохо реализованный метод Try / Catch может скрыть причину проблемы и привести к появлению новых:

Dim dr As SqlDataReader
Try
    Dim lnk As LinkButton = TryCast(sender, LinkButton)
    Dim gr As GridViewRow = DirectCast(lnk.NamingContainer, GridViewRow)
    Dim eid As String = GridView1.DataKeys(gr.RowIndex).Value.ToString()
    ViewState("username") = eid
    sqlQry = "select FirstName, Surname, DepartmentName, ExtensionName, jobTitle,
             Pager, mailaddress, from employees1 where username='" & eid & "'"
    If connection.State <> ConnectionState.Open Then
        connection.Open()
    End If
    command = New SqlCommand(sqlQry, connection)

    'More code fooing and barring

    dr = command.ExecuteReader()
    If dr.Read() Then
        lblFirstName.Text = Convert.ToString(dr("FirstName"))
        ...
    End If
    mpe.Show()
Catch

Finally
    command.Dispose()
    dr.Close()             ' <-- NRE
    connection.Close()
End Try

Это случай, когда объект создается не так, как ожидалось, но также демонстрирует счетчик полезности пустого Catch.

В SQL есть лишняя запятая (после mailaddress), что приводит к исключению в .ExecuteReader. После того, Catchкак ничего не делает, Finallyпытается выполнить очистку, но, поскольку вы не можете Closeполучить нулевой DataReaderобъект, новые NullReferenceExceptionрезультаты.

Пустой Catchквартал – игровая площадка дьявола. Этот OP был сбит с толку, почему он получал NRE в Finallyблоке. В других ситуациях пустое место Catchможет привести к тому, что что-то еще дальше по течению выйдет из строя, и вы будете тратить время на поиск неправильных вещей в неправильном месте, где возникла проблема. (Описанное выше «тихое исключение» обеспечивает такую ​​же развлекательную ценность.)

Средство

Не используйте пустые блоки Try / Catch – позвольте коду дать сбой, чтобы вы могли: а) определить причину, б) определить местоположение и в) применить правильное средство. Блоки Try / Catch не предназначены для сокрытия исключений от лица, обладающего уникальной квалификацией для их исправления – разработчика.


DBNull – это не то же самое, что Nothing

For Each row As DataGridViewRow In dgvPlanning.Rows
    If Not IsDBNull(row.Cells(0).Value) Then
        ...

IsDBNullФункция используется для проверки , если значение равно System.DBNull: Из MSDN:

The System.DBNull value indicates that the Object represents missing or non-existent data. DBNull is not the same as Nothing, which indicates that a variable has not yet been initialized.

Средство

If row.Cells(0) IsNot Nothing Then ...

Как и раньше, вы можете проверить значение Nothing, а затем определенное значение:

If (row.Cells(0) IsNot Nothing) AndAlso (IsDBNull(row.Cells(0).Value) = False) Then

Пример 2

Dim getFoo = (From f In dbContext.FooBars
               Where f.something = something
               Select f).FirstOrDefault

If Not IsDBNull(getFoo) Then
    If IsDBNull(getFoo.user_id) Then
        txtFirst.Text = getFoo.first_name
    Else
       ...

FirstOrDefaultвозвращает первый элемент или значение по умолчанию, которое предназначено Nothingдля ссылочных типов и никогда DBNull:

If getFoo IsNot Nothing Then...

Органы управления

Dim chk As CheckBox

chk = CType(Me.Controls(chkName), CheckBox)
If chk.Checked Then
    Return chk
End If

Если CheckBoxwith chkNameне может быть найден (или существует в a GroupBox), тогда chkбудет Nothing, и попытка ссылки на какое-либо свойство приведет к исключению.

Средство

If (chk IsNot Nothing) AndAlso (chk.Checked) Then ...

DataGridView

У DGV периодически наблюдается несколько причуд:

dgvBooks.DataSource = loan.Books
dgvBooks.Columns("ISBN").Visible = True       ' NullReferenceException
dgvBooks.Columns("Title").DefaultCellStyle.Format = "C"
dgvBooks.Columns("Author").DefaultCellStyle.Format = "C"
dgvBooks.Columns("Price").DefaultCellStyle.Format = "C"

Если dgvBooksесть AutoGenerateColumns = True, он создаст столбцы, но не называет их, поэтому приведенный выше код не работает, когда он ссылается на них по имени.

Средство

Назовите столбцы вручную или укажите индекс:

dgvBooks.Columns(0).Visible = True

Пример 2 – Остерегайтесь NewRow

xlWorkSheet = xlWorkBook.Sheets("sheet1")

For i = 0 To myDGV.RowCount - 1
    For j = 0 To myDGV.ColumnCount - 1
        For k As Integer = 1 To myDGV.Columns.Count
            xlWorkSheet.Cells(1, k) = myDGV.Columns(k - 1).HeaderText
            xlWorkSheet.Cells(i + 2, j + 1) = myDGV(j, i).Value.ToString()
        Next
    Next
Next

Когда у вас DataGridViewесть AllowUserToAddRowsas True(по умолчанию), все Cellsв пустой / новой строке внизу будут содержать Nothing. Большинство попыток использовать содержимое (например, ToString) приведет к NRE.

Средство

Используйте For/Eachцикл и проверьте IsNewRowсвойство, чтобы определить, является ли это последней строкой. Это работает независимо от того AllowUserToAddRows, правда это или нет:

For Each r As DataGridViewRow in myDGV.Rows
    If r.IsNewRow = False Then
         ' ok to use this row

Если вы все же используете For nцикл, измените количество строк или используйте, Exit Forкогда IsNewRowоно истинно.


Мои настройки (StringCollection)

При определенных обстоятельствах попытка использовать элемент, из My.Settingsкоторого является a, StringCollectionможет привести к NullReference при первом использовании. Решение то же самое, но не столь очевидное. Рассмотреть возможность:

My.Settings.FooBars.Add("ziggy")         ' foobars is a string collection

Поскольку VB управляет настройками за вас, разумно ожидать, что он инициализирует коллекцию. Будет, но только если вы ранее добавили начальную запись в коллекцию (в редакторе настроек). Поскольку коллекция (очевидно) инициализируется при добавлении элемента, она остается, Nothingкогда в редакторе настроек нет элементов для добавления.

Средство

Инициализируйте коллекцию настроек в Loadобработчике событий формы, если / когда необходимо:

If My.Settings.FooBars Is Nothing Then
    My.Settings.FooBars = New System.Collections.Specialized.StringCollection
End If

Обычно Settingsколлекцию нужно инициализировать только при первом запуске приложения. Альтернативное решение – добавить начальное значение в вашу коллекцию в Project -> Settings | FooBars , сохраните проект, затем удалите поддельное значение.


Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *