Error list is null

Suppose a simple example where a method retrieves a collection (such as a list containing some configuration strings) and tries to examine it in some way: void Init() { XmlDocument config = new

Suppose a simple example where a method retrieves a collection (such as a list containing some configuration strings) and tries to examine it in some way:

void Init()
{
    XmlDocument config = new XmlDocument();
    config.Load(someXml);
    var list = config.SelectNodes("/root/strings/key"); // Normally, list should not be null or empty

    if (list == null || list.Count == 0)
        throw new SomeExceptionType(message);   // What kind of exception to throw?

    // Iterate list and process/examine its elements
    foreach (var e in list) ...
}

In this specific instance, the method cannot continue normally if nothing was retrieved. I’m unsure what exception type to throw in such situations. My options are, as far as I know:

  • throw nothing manually and let NullReferenceException be thrown
    automatically (which doesn’t handle empty list situation),

  • throw custom exception type (probably not a good idea, as I don’t anticipate the caller will try to do anything about the exception,
    i.e. he won’t be looking for a speecific exception type to handle),

  • do something else?

asked Sep 5, 2013 at 13:40

w128's user avatar

w128w128

4,4886 gold badges41 silver badges62 bronze badges

3

You can create your own exception type for appropriate logic:

public class InitializationException : Exception
{
}

and then:

throw new InitializationException {Message = "Collection is empty"};

answered Sep 5, 2013 at 13:45

Dzmitry Martavoi's user avatar

Dzmitry MartavoiDzmitry Martavoi

6,7636 gold badges38 silver badges56 bronze badges

4

I’m not sure there is a single built-in exception you can elegantly throw in this case…a NullReferenceException is inappropriate since an empty list is not a null reference

I would suggest going with Dmintry’s proposed solution since the caller can still just use try...catch(Exception) without having to know or care that the exception is really a SuperDooperListNullOrEmptyFunTimeException

Since this is either an unrecoverable error from the caller’s point of view (i.e they have no control over the selected Xml path, and no control over what the XML is that’s being loaded) then the exception is only going to be either dumped to a log or on-screen for human consumption, at which point it’s moot — as the actual message is more important than the type.

On the other hand, if it is recoverable (caller can re-try the method after having made sure that the xml to load now contains the correctly formatted xml, or caller can notify the user and ask them to go and fix the XML and «would you like to retry now?» kind of thing) then you need to give them a typed exception so they know it’s safe to retry as opposed to a plain old Exception which could mean something else went horribly wrong and retrying will only make things worse…

answered Sep 5, 2013 at 15:01

Stephen Byrne's user avatar

Stephen ByrneStephen Byrne

7,3201 gold badge30 silver badges50 bronze badges

2

That not much of programming problem as it’s a design issue, the reason .NET list object don’t throw exceptions when they are empty is because there is a lot of cases where a empty list is an expectable and acceptable situation.

If in the context the list you work with is never supposed to be empty then throw an exception (a custom one)

If however it’s possible and logical that list may be empty, why interrupt the whole thing, it’s excepted not exceptional, so need for an exception ? A foreach loop and an empty list won’t throw an exception, the loop simply won’t loop.

As for the null possibility (quite rare for SelectNodes if understand well) it’s same issue, in some libraries or functions returning a null is a normal behavior not an exception.

answered Sep 5, 2013 at 13:55

Simon Rapilly's user avatar

3

Cover image for No more NullReferenceException for Lists in C#

Ali Alp

Ali Alp

Posted on Jun 7, 2019

• Updated on Oct 14, 2019

While coding sometimes little tiny things are so frustrating , like forgetting to instantiate a List object before adding a new item, an easy solution will be to write an extension method like this

public static class ListExtension
{
    public static void Add<T>(this List<T> list, T value, out List<T> newInstance)
    {
        if (list == null) list = new List<T>();
        list?.Add(value);
        newInstance = list;
    }
}

Enter fullscreen mode

Exit fullscreen mode

and then use it like this

[Test]
public void SafeAddTest()
{
    List<string> list = null;
    list.Add("test",out list);

    Assert.AreEqual(1,list.Count);
}

Enter fullscreen mode

Exit fullscreen mode

therefore whenever you want to make sure that the list that you are going to add to it is already instantiated you can simply pass the list instance as the second variable and that’s it.

Happy coding :)

Hey 😍

Want to help the DEV Community feel more like a community?

Head over to the Welcome Thread and greet some new community members!

It only takes a minute of your time, and goes a long way!

  • Remove From My Forums
  • Question

  • This is the code which I’m using

    var dta = _jobRepository.Jobs.Where(x => filter.CategoryIds == null || filter.CategoryIds.Contains(x.CategoryId));

    Jobs is returning IQueryable<Job>. I want to check if CategoryId is in list or return all if CategoryIds is null. I’m getting follong error on running this code

    Cannot compare elements of type ‘System.Collections.Generic.List`1’. Only primitive types, enumeration types and entity types are supported.

    Please help.

    • Moved by

      Monday, November 11, 2013 6:15 AM
      EF question.

Answers

  • Try it like this.  You can’t pass a List to a SQL query like that. 

    var dta = _jobRepository.Jobs;
    if ( filter.CategoryIds != null )
    {  
      dta = dta.Where(filter.CategoryIds.Contains(x.CategoryId));
    }

    David


    David http://blogs.msdn.com/b/dbrowne/

    • Proposed as answer by
      Fred Bao
      Monday, November 11, 2013 6:44 AM
    • Marked as answer by
      Moncy K
      Monday, November 11, 2013 6:46 AM

В этом посте я покажу наглядный пример того, как исправить ошибку исключения Null Pointer (java.lang.nullpointerexception). В Java особое значение null может быть назначено для ссылки на объект и означает, что объект в данный момент указывает неизвестную область данных.

NullPointerException появляется, если программа обращается или получает доступ к объекту, а ссылка на него равна нулю (null).

Это исключение возникает следующих случаях:

  • Вызов метода из объекта значения null.
  • Доступ или изменение объекта поля null.
  • Принимает длину null(если бы это был массив Java).
  • Доступ или изменение ячеек объекта null.
  • Показывает «0», значение Throwable.
  • При попытке синхронизации по нулевому объекту.

NullPointerException является RuntimeException, и, таким образом, компилятор Javac не заставляет вас использовать блок try-catch для соответствующей обработки.

ошибка error java lang nullpointerexception

Зачем нам нужно значение null?

Как уже упоминалось, null – это специальное значение, используемое в Java. Это чрезвычайно полезно при кодировании некоторых шаблонов проектирования, таких как Null Object pattern и шаблон Singleton pattern.

Шаблон Singleton обеспечивает создание только одного экземпляра класса, а также направлен на предоставление глобального доступа к объекту.

Например, простой способ создания не более одного экземпляра класса – объявить все его конструкторы как частные, а затем создать открытый метод, который возвращает уникальный экземпляр класса:

import java.util.UUID;

class Singleton {

     private static Singleton single = null;
     private String ID = null;

     private Singleton() {
          /* Make it private, in order to prevent the creation of new instances of
           * the Singleton class. */

          ID = UUID.randomUUID().toString(); // Create a random ID.
     }

     public static Singleton getInstance() {
          if (single == null)
               single = new Singleton();

          return single;
     }

     public String getID() {
          return this.ID;
     }
}

public class TestSingleton {
     public static void main(String[] args) {
          Singleton s = Singleton.getInstance();
          System.out.println(s.getID());
     }
}

В этом примере мы объявляем статический экземпляр класса Singleton. Этот экземпляр инициализируется не более одного раза внутри метода getInstance.

Обратите внимание на использование нулевого значения, которое разрешает создание уникального экземпляра.

Как избежать исключения Null Pointer

Чтобы решить и избежать исключения NullPointerException, убедитесь, что все ваши объекты инициализированы должным образом, прежде чем использовать их.

Когда вы объявляете ссылочную переменную, вы должны создать указатель на объект и убедиться, что указатель не является нулевым, прежде чем запрашивать метод или поле у ​​объекта.

Кроме того, если выдается исключение, используйте информацию, находящуюся в трассировке стека исключения. Трассировка стека выполнения обеспечивается JVM, чтобы включить отладку. Найдите метод и строку, в которой было обнаружено исключение, а затем выясните, какая ссылка равна нулю в этой конкретной строке.

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

  1. Сравнение строк с литералами

Очень распространенный случай, выполнения программы включает сравнение между строковой переменной и литералом. Литерал может быть строкой или элементом Enum.

Вместо того, чтобы вызывать метод из нулевого объекта, рассмотрите возможность вызова его из литерала. Например:

String str = null;
if(str.equals("Test")) {
     /* The code here will not be reached, as an exception will be thrown. */
}

Приведенный выше фрагмент кода вызовет исключение NullPointerException. Однако, если мы вызываем метод из литерала, поток выполнения продолжается нормально:

String str = null;
if("Test".equals(str)) {
     /* Correct use case. No exception will be thrown. */
}
  1. Проверка аргументов метода

Перед выполнением вашего собственного метода обязательно проверьте его аргументы на наличие нулевых значений.

В противном случае вы можете вызвать исключение IllegalArgumentException.

Например:

public static int getLength(String s) {
     if (s == null)
          throw new IllegalArgumentException("The argument cannot be null");

     return s.length();
}
  1. Предпочтение метода String.valueOf() вместо of toString()

Когда код вашей программы требует строковое представление объекта, избегайте использования метода toString объекта. Если ссылка вашего объекта равна нулю, генерируется исключение NullPointerException.

Вместо этого рассмотрите возможность использования статического метода String.valueOf, который не выдает никаких исключений и «ноль», если аргумент функции равен нулю.

  1. Используйте Ternary Operator

Ternary Operator – может быть очень полезным. Оператор имеет вид:

boolean expression ? value1 : value2;

Сначала вычисляется логическое выражение. Если выражение true, то возвращается значение1, в противном случае возвращается значение2. Мы можем использовать Ternary Operator для обработки нулевых указателей следующим образом:

String message = (str == null) ? "" : str.substring(0, 10);

Переменная message будет пустой, если ссылка str равна нулю. В противном случае, если str указывает на фактические данные, в сообщении будут первые 10 символов.

  1. создайте методы, которые возвращают пустые коллекции вместо нуля.

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

public class Example {
     private static List<Integer> numbers = null;

     public static List<Integer> getList() {
          if (numbers == null)
               return Collections.emptyList();
          else
               return numbers;
     }
}
  1. Воспользуйтесь классом Apache’s StringUtils.

Apache’s Commons Lang – это библиотека, которая предоставляет вспомогательные утилиты для API java.lang, такие как методы манипулирования строками.

Примером класса, который обеспечивает манипулирование String, является StringUtils.java, который спокойно обрабатывает входные строки с нулевым значением.

Вы можете воспользоваться методами: StringUtils.isNotEmpty, StringUtils.IsEmpty и StringUtils.equalsчтобы избежать NullPointerException. Например:

if (StringUtils.isNotEmpty(str)) {
     System.out.println(str.toString());
}
  1. Используйте методы: contains(), containsKey(), containsValue()

Если в коде вашего приложения используется Maps, рассмотрите возможность использования методов contains, containsKey и containsValue. Например, получить значение определенного ключа после того, как вы проверили его существование на карте:

Map<String, String> map = …
…
String key = …
String value = map.get(key);
System.out.println(value.toString()); // An exception will be thrown, if the value is null.

System.out.println(value.toString()); // В приведенном выше фрагменте мы не проверяем, существует ли на самом деле ключ внутри карты, и поэтому возвращаемое значение может быть нулевым. Самый безопасный способ следующий:

Map<String, String> map = …
…
String key = …
if(map.containsKey(key)) {
     String value = map.get(key);
     System.out.println(value.toString()); // No exception will be thrown.
}
  1. Проверьте возвращаемое значение внешних методов

На практике очень часто используются внешние библиотеки. Эти библиотеки содержат методы, которые возвращают ссылку. Убедитесь, что возвращаемая ссылка не пуста.

  1. Используйте Утверждения

Утверждения очень полезны при тестировании вашего кода и могут использоваться, чтобы избежать выполнения фрагментов кода. Утверждения Java реализуются с помощью ключевого слова assert и выдают AssertionError.

Обратите внимание, что вы должны включить флажок подтверждения JVM, выполнив его с аргументом -ea. В противном случае утверждения будут полностью проигнорированы.

Примером использования утверждений Java является такая версия кода:

public static int getLength(String s) {
     /* Ensure that the String is not null. */
     assert (s != null);

     return s.length();
}

Если вы выполните приведенный выше фрагмент кода и передадите пустой аргумент getLength, появится следующее сообщение об ошибке:
Exception in thread "main" java.lang.AssertionError
Также вы можете использовать класс Assert предоставленный средой тестирования jUnit.

  1. Модульные тесты

Модульные тесты могут быть чрезвычайно полезны при тестировании функциональности и правильности вашего кода. Уделите некоторое время написанию пары тестовых примеров, которые подтверждают, что исключение NullPointerException не возникает.

Существующие безопасные методы NullPointerException

Доступ к статическим членам или методам класса

Когда ваш вы пытаетесь получить доступ к статической переменной или методу класса, даже если ссылка на объект равна нулю, JVM не выдает исключение.

Это связано с тем, что компилятор Java хранит статические методы и поля в специальном месте во время процедуры компиляции. Статические поля и методы связаны не с объектами, а с именем класса.

class SampleClass {

     public static void printMessage() {
          System.out.println("Hello from Java Code Geeks!");
     }
}

public class TestStatic {
     public static void main(String[] args) {
          SampleClass sc = null;
          sc.printMessage();
     }
}

Несмотря на тот факт, что экземпляр SampleClass равен нулю, метод будет выполнен правильно. Однако, когда речь идет о статических методах или полях, лучше обращаться к ним статическим способом, например, SampleClass.printMessage ().

Оператор instanceof

Оператор instanceof может использоваться, даже если ссылка на объект равна нулю.

Оператор instanceof возвращает false, когда ссылка равна нулю.

String str = null;
if(str instanceof String)
     System.out.println("It's an instance of the String class!");
else
     System.out.println("Not an instance of the String class!");

В результате, как и ожидалось:

Not an instance of the String class!

Смотрите видео, чтобы стало понятнее.



New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and
privacy statement. We’ll occasionally send you account related emails.

Already on GitHub?
Sign in
to your account


Closed

mikeholler opened this issue

Oct 15, 2019

· 20 comments

Comments

@mikeholler

Hey there! I’m using jackson-module-kotlin and I like my null safety, but Jackson doesn’t appear to respect these desires. I’ve been fighting with this issue for nearly 4 hours, and the guidance I’ve found is very confusing. I’m creating the issue here because I believe the solution lies with jackson-databind and not jackson-module-kotlin.

data class ObjectWithStringList(val strings: List<String>)

@Test
fun `list contains null`() {
    expectThrows<JsonProcessingException> {
        val result = ObjectMapper().registerKotlinModule().readValue<ObjectWithStringList>("""{ "strings": ["a", null, "b"] }""")
    }
}

ObjectWithStringList defines strings as List, which means that users should expect that the strings inside the list are never null (List<String?> would mean it could contain nulls). When I run this test, however, it fails (the exception is not thrown). What happens instead is result[1] evaluates to null, which should never happen according to Kotlin.

What do I want? Ideally I would want jackson-module-kotlin to respect collection element types’ nullity, however I realize this is idealistic and may be difficult. Instead, I would like the default behavior of deserializing any list type with nulls (whether String or String?) to throw an exception. And then I would like some way to override this behavior if I really want to allow nulls in my list.

I’ve gone down the rabbit hole and tried a large number of things (which I can summarize upon request, but for brevity’s sake, decided not to here), and have been unable to find anything acceptable. The best solution I’ve found so far is implementing my own collection type (see below) but the drawback to this solution is that I still want it to appear as List, not ArrayListOfNotNull.

data class ObjectWithStringList(val strings: ArrayListOfNotNull<String>)
class ArrayListOfNotNull<E: Any> : ArrayList<E> {
    constructor(): super()
    constructor(initialSize: Int): super(initialSize)
    constructor(elements: Collection<E>): super(elements) { require(elements.none { it == null }) }
    override fun addAll(elements: Collection<E>): Boolean { require(elements.none { it == null }); return super.addAll(elements) }
    override fun addAll(index: Int, elements: Collection<E>): Boolean { require(elements.none { it == null }); return super.addAll(index, elements) }
    override fun add(element: E): Boolean { requireNotNull(element); return super.add(element) }
    override fun add(index: Int, element: E) { requireNotNull(element); super.add(index, element) }
}

Also note that this did not work (I was really hoping it would):

data class ObjectWithStringList(@JsonDeserializer(`as` = ArrayListOfNotNull::class) val strings: List<String>)

Even if it did work, however, it’s not the default behavior (which was my original preference).

With the plethora of ways to integrate into Jackson Databind, what is the recommended approach here? Additionally, I have to believe this affects lots of Kotlin users, whether they realize it or not (and I think the latter is more likely). As a result, I’d like to see some documentation added for handling this type of situation, because I could not find much of substance.

@cowtowncoder

This is probably more related Kotlin (although certainly would need help from databind), since core Jackson has absolutely no idea of ? decoration that Kotlin includes somehow. Kotlin module would need to determine this (I don’t know what is used at JVM level, maybe annotation of some kind in bytecode).

But one big challenge is that although databind does have many mechanisms for skipping or ignoring null values for properties, same is not true for Lists. Big part of this is that in Lists, one really should not just automatically ignore anything: being index-based it is extremely confusing to just not see something for which input had value, or placeholder.

Having said that, Jackson 2.9 did add significantly to configurability of null handling, with annotations like:

@JsonSetter(nulls=Nulls.SKIP, contentNulls = Nulls.FAIL)

(see https://medium.com/@cowtowncoder/jackson-2-9-features-b2a19029e9ff under «Null replacement/error/skipping (on deserialization)»)

which would seem promising, but. This annotation allows specifying handling for both values of properties («main» nulls, aka «value nulls») and contents of values.
So in case of Lists, you would want to specify behavior for contentNulls to either SKIP (will just remove them, effectively) or FAIL (to throw exception indicating invalid input).

So: I think this might work for you.

I think Kotlin module does not change Collection handling a lot, so it might «just work». Unit tests for pure Java can be found from:

src/test/java/com/fasterxml/jackson/databind/deser/filter/NullConversionsForContentTest.java

@mikeholler

Hey @cowtowncoder thank you for the response! I don’t expect jackson-databind to understand ?, but I also know that jackson-module-kotlin has also discussed the understanding of nullity annotations in lists and come to the conclusion here (TLDR; won’t happen): FasterXML/jackson-module-kotlin#27

It’s for that reason that I’m writing you here to understand what we can do in Kotlin using just the jackson-databind libs, because it seems that’s where our best chance at a solution lies. We’re on 2.9.9 right now I think, and the @JsonSetter annotation was a good suggestion, but it also doesn’t seem to work in Kotlin: FasterXML/jackson-module-kotlin#250

I know there are a lot of bindings available in Jackson for modifying functionality; are there any you see that would work somewhat similarly (via annotation, ideally, but open to other solutions) to what you described that Kotlin might support? It’s these kinds of sharp edges I kept stumbling on while I was looking for a solution and I hoped that you or someone else here who knows Jackson deeply could help me find a good compromise.

@cowtowncoder

@mikeholler I don’t think #250 is quite related here: TBH I am not sure I understand what user is trying to do there (and whether it should work or not), but it is not about List handling of nulls.
So I would not assume that this means @JsonSetter in general does not work.

But aside the question of making @JsonSetter work, possible next step would be to make Kotlin module indicate that by default nulls within Collections should be SKIPped. This would be possible via AnnotationIntrospector since databind itself is determining existing of @JsonSetter using its default JacksonAnnotationIntrospector; so whatever configured AI tells it is the truth.

Now, back to ?: what I still do not quite know is exactly how Kotlin uses that information. I guess it is possible it might only be used similar to Java generics, i.e. compile-time only info. If so, we couldn’t do much about it.
But if, on the other hand, it is accessible via Class definition then THAT could indicate settings to use for:

  1. @JsonProperty(required = true) which has effect on Creator properties, and possibly (for 3.0), other properties
    2 @JsonSetter (as per above)

So I think we are on same page regarding finding mechanisms that can be used to support more natural handling from Kotlin POV.

@mikeholler

@cowtowncoder I have to pause on this for a bit today, but I want to partially reply to your comment so I can address the question about #250, which I think does apply:

    data class RecommendedSolution(
        @JsonSetter(contentNulls = Nulls.FAIL)
        val strings: List<String>
    )

    @Test
    fun `test recommended solution`() {
        expectThrows<JsonProcessingException> {
            mapper.readValue<RecommendedSolution>("""{"strings": ["a", null, "c"]}""")
        }
    }

The above test fails. Also, the following different annotation sites fail as well:

    data class RecommendedSolution(
        @field:JsonSetter(contentNulls = Nulls.FAIL)
        val strings: List<String>
    )
    data class RecommendedSolution(
        @param:JsonSetter(contentNulls = Nulls.FAIL)
        val strings: List<String>
    )
    data class RecommendedSolution(
        @property:JsonSetter(contentNulls = Nulls.FAIL)
        val strings: List<String>
    )
    data class RecommendedSolution(
        @get:JsonSetter(contentNulls = Nulls.FAIL)
        val strings: List<String>
    )

Note that @set: and @setparam:, which would make the most sense as annotation types for @JsonSetter don’t exist (by design, since I’m using val not var). Any way to get the @JsonSetter behavior on a creator parameter instead of an actual setter?

@GedMarc

This may be my biggest shot in the dark ever

@NotNull is a validation api (), so jackson wouldn’t catch it. Perhaps someone is feeling generous to build a data type for it, but for now, using the validation api/processor should suffice?
As long as your not using fong kong not null like … i dunno.. org.jetbrains.annotations.NotNull, etc…
Something other than javax.annotation.NotNull


<dependency>   
 <groupId>org.hibernate.validator</groupId>   
 <artifactId>hibernate-validator-annotation-processor</artifactId>   
 <version>6.0.2.Final</version></dependency>

I think this may be a/the solution? As a processor is better than runtime execution but you can simply include it as well…

Everyones post is long so —

A quick note here is that hibernate-validator is entirely separate from the persistence aspects of Hibernate and by adding it as a dependency, we’re not adding these persistence aspects into the project.

If you’re in jdk 12 or up remember to provide your annotation to the annotation processor to your module-info.java

	provides javax.annotation.processing.Processor with bla bla bla;

The exception you should be catching is ValidationException as well, all of these will throw it

@Test
fun `list contains null`() {
    expectThrows<ValidationException> {
        val result = ObjectMapper().registerKotlinModule().readValue<ObjectWithStringList>("""{ "strings": ["a", null, "b"] }""")
    }
}
All of the annotations used in the example are standard JSR annotations:

    @NotNull – validates that the annotated property value is not null
    @AssertTrue – validates that the annotated property value is true
    @Size – validates that the annotated property value has a size between the attributes min and max; can be applied to String, Collection, Map, and array properties
    @Min – vValidates that the annotated property has a value no smaller than the value attribute
    @Max – validates that the annotated property has a value no larger than the value attribute
    @Email – validates that the annotated property is a valid email address

Some annotations accept additional attributes, but the message attribute is common to all of them. This is the message that will usually be rendered when the value of the respective property fails validation.

Some additional annotations that can be found in the JSR are:

    @NotEmpty – validates that the property is not null or empty; can be applied to String, Collection, Map or Array values
    @NotBlank – can be applied only to text values and validated that the property is not null or whitespace
    @Positive and @PositiveOrZero – apply to numeric values and validate that they are strictly positive, or positive including 0
    @Negative and @NegativeOrZero – apply to numeric values and validate that they are strictly negative, or negative including 0
    @Past and @PastOrPresent – validate that a date value is in the past or the past including the present; can be applied to date types including those added in Java 8
    @Future and @FutureOrPresent – validates that a date value is in the future, or in the future including the present

Quick read up : https://www.baeldung.com/javax-validation

@mikeholler

@GedMarc you’re absolutely right, and I applaud your creativity; javax validation is exactly where I reached after having so many issues with Jackson. But sadly, we meet another impasse: meet KT-13228.

Kotlin does not yet support type annotations (which always surprises the heck out of me when I wait just long enough between uses to forget that that’s the case). Looks like it’s being added in 1.4, but it’s not out yet 😢

https://youtrack.jetbrains.com/issue/KT-13228

@GedMarc

@mikeholler not sure if that is sarcastic or not xD either way…

Are you including the validation api?, or like hibernate or just referencing the java.activation library?
javax.validation allows a full run locally so this may just be a datatype subtype which requires the jaxb module

You could possibly get around this by creating a reader that simply throws out the validation?
Not sure if kotlin does jaxb?

This is a copy paste from my stuff, you can just throw it out in a reader.

List<String> errors = new ArrayList<>();
		ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
		Validator validator = factory.getValidator();
		Set constraintViolations = validator.validate(entity);
		if (!constraintViolations.isEmpty())
		{
			for (Object constraintViolation : constraintViolations)
			{
				ConstraintViolation contraints = (ConstraintViolation) constraintViolation;
				String error = contraints.getRootBeanClass()
				                         .getSimpleName() + STRING_DOT + contraints.getPropertyPath() + STRING_EMPTY + contraints.getMessage();
				errors.add(error);
			}
		}
		return errors;

@GedMarc

  • Deserializer not reader..

wrong brain mode

@mikeholler

@GedMarc it’s not sarcastic, it’s a great idea but unfortunately not one I can use. We’re using javax validation through Dropwizard’s built-in model validation. I’d like it to work with that, but the lack of Kotlin type annotations got in the way.

@GedMarc

You don’t need annotations this way :)
I’m not convinced it won’t work in kotlin.
At the very least it does build a base for a datatype extension for you to include as a module

	@JsonDeserialize(using = ValidationAPIDeserializer.class)
	private String name;

It works for me, hmmm.


import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;

import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.ValidationException;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
import javax.validation.constraints.NotNull;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;

public class ValidationAPIDeserializer extends StdDeserializer<String> {

	protected ValidationAPIDeserializer() {
		super(String.class);
	}

	@Override
	public String deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException, ValidationException {
		String value = p.getValueAsString();

		if (value == null || value.trim()
								  .isEmpty()) {
			return null;
		}
		return value;
	}

	@NotNull
	public List<String> validateEntity(Object entity) {
		List<String> errors = new ArrayList<>();
		ValidatorFactory factory = Validation.buildDefaultValidatorFactory();
		Validator validator = factory.getValidator();
		Set constraintViolations = validator.validate(entity);
		if (!constraintViolations.isEmpty()) {
			for (Object constraintViolation : constraintViolations) {
				ConstraintViolation contraints = (ConstraintViolation) constraintViolation;
				String error = contraints.getRootBeanClass()
										 .getSimpleName() + "." + contraints.getPropertyPath() + " " + contraints.getMessage();
				errors.add(error);
			}
		}
		throw new ValidationException(errors.toString());
	}
}

@cowtowncoder

@mikeholler Ok my bad on #250 then, I read it too quickly and somehow determined it was related to non-container properties of a data class.

But as to @JsonSetter, that should be applicable to Fields and Constructor parameters too. This was changed in 2.9, so perhaps you have dependency to older jackson-annotations?

Now, back to null-handling not working… I assume this class

src/main/kotlin/com/fasterxml/jackson/module/kotlin/KotlinValueInstantiator.kt

essentially replaces what BeanDeserializer would do. But that does not yet quite explain why handling of null-values would get overridden for individual JsonDeserializers. What matters there is that collection deserializers get configured with a NullValueProvider, and that is used to deal with values.
Unfortunately code in KotlinValueInstantiator is… quite complicated for me to understand.
But on plus side, codebase of module is quite small (compared to f.ex Scala module).

Another angle on null handling that might help is that instead of annotations, one can also use «config overrids» to specify defaults. Databind test src/test/java/com/fasterxml/jackson/databind/deser/filter/NullConversionsForContentTest.java has, for example:

    mapper.configOverride(List.class)
            .setSetterInfo(JsonSetter.Value.forContentNulls(Nulls.FAIL));

which should also achieve what you want for all properties with nominal (declared) type of List.

Or, for even more global applicability:

mapper.setDefaultSetterInfo(JsonSetter.Value.forContentNulls(Nulls.FAIL));

where one caveat is that «content» refers to Collection elements, Map entry values but NOT to Bean property values (which are «value» nulls, not «content»).

I guess I am mostly just interested in figuring out how to make Kotlin module not block use of this setting: I do not think there is anything fundamental preventing it but maybe some missing wiring.
For example: for most property annotations to take effect, (de)serializers MUST be contextualized (call to createContextual()). It could be missing delegation somewhere.
Or, data class handler using set of BeanProperty objects pre-contextualization, not getting updated.

If Kotlin module had custom Collection deserializers, I would bet they were just missing handling of NullValueHandler, but I did not see any (if there are inner classes, maybe I just missed them).

@mikeholler

But as to @JsonSetter, that should be applicable to Fields and Constructor parameters too. This was changed in 2.9, so perhaps you have dependency to older jackson-annotations?

@cowtowncoder I can apply it to fields and constructors (that’s what I did in my examples). They compile, Jackson just doesn’t pick up how I’m trying to do it, even with the annotation site qualifiers like @field: and @param: shown above.

I really like the configOverride stuff, that’s something I can probably implement. It’ll become a problem when we implement the one or two JSON models where we actually want to allow nulls in lists (since we haven’t been able to fine an annotation-based solution for Kotlin yet that allows behavior to be changed per model, rather than per mapper). It feels like every solution we find is so very close to being what we want, but just far enough to avoid feeling like a truly good solution.

@cowtowncoder

Ah. So I misunderstood that part wrt applicability.

One other thing: it’d be good to debug as to whether problem comes down to annotation not being merge to appropriate property (and so not visible to deserializer factory etc), or just not being used.
At this point I think I should be able to run a Kotlin test, if provided, and debug things from databind side.
So all I’d need is minimal case (smallest data class, trivial read), and I think I could be bit more productive.
I could also verify my understanding that Kotlin module still uses default CollectionDeserializer (… or not).

@mikeholler

Thanks @cowtowncoder, give me until EOB tomorrow to get you a small example project with tests that verifies that. I’ll make it nice and easy for you to get going. Just to be clear, you want an example of the @JsonSetter problem and verification that the annotation is applied to the correct JVM site when compiled, right?

@cowtowncoder

For me test only needs to try to trigger expected behavior: it does not need to check bytecode or anything like that. I can observe (debugger or adding good old print statements :) ) what happens in places like creation and contextualization of CollectionDeserializer (and in general, kinds of deserializers even created) and see what that code sees.
I can also check out what JacksonAnnotationIntrospector finds, or, at even lower level, how AnnotatedClass (actual representation of collected annotations) get as their input.

Does this make more sense? The reason I want minimal case is just to reduce noise wrt resolution of things.

@mikeholler

Yeah for sure. I can absolutely put that together for you.

@mikeholler

@cowtowncoder

That is an excellent outcome. I am not sure what might have fixed it… looking at release notes, I would guess that maybe #2280 did it? Or perhaps #2458. But difficult to know for sure.

One follow-up question: is use of this feature that you could explain on, say, README? I know how it works in general but often am not the best person to explain it. And in case of Kotlin there’s more context too. But having a brief usage sample for commonly expected usage seems useful.

@mikeholler

@cowtowncoder you’re probably right that it was one of those two. Thanks for those links.

One follow-up question: is use of this feature that you could explain on, say, README? I know how it works in general but often am not the best person to explain it. And in case of Kotlin there’s more context too. But having a brief usage sample for commonly expected usage seems useful.

So, kinda? The tough part here is that I have this feeling that this should be the default behavior (failing fast and not allowing any nulls in a list by default, then overriding if nulls are needed is clearly the best outcome). It seems to me a large flaw in jackson-module-kotlin that it allows the behavior it does for nulls, which is completely unexpected for a Kotlin user. We could document a workaround to the flaw, or a plan could be developed to work to correct it. Unfortunately, correcting it would most definitely cause a breaking change to existing users, meaning a major version bump would be necessary.

At its simplest, I could document the @JsonSetter behavior in the README but I’m concerned how that will reflect on the library.

@cowtowncoder

It could be a separate Wiki page for repo too — Workarounds (or something) for Kotlin?
I don’t think it is unreasonable to explain how behavior currently (as of 2.10) differs from what might be expected; especially in that users may well bump into these problems without ways to alleviate the problem which is probably even worse impression.

Hello everyone,

I’m working on MVC 5, there is a viewmodel class and contain a static list as the same class.

  1. public class PartnerProjectVM  
  2.     {  
  3.         public int idPartnerProject { getset; }  
  4.         public string ProjectName { getset; }  
  5.         public string ProjectDescription { getset; }  
  6.   public static List<PartnerProjectVM> listPartnerProjectVM = new List<PartnerProjectVM>();  

 In view,

How can I check

  1. @model Reboxwebapp.Models.ViewModel.PartnerProjectVM  
  2.   @if (listPartnerProjectVM.Count() != null)  
  3.                             {  
  4.                                 <div class=«col-sm-3»>  
  5.                                     <div>  
  6.                                         <a class=«list-group-item active left-menu-head»>Projects List</a>  
  7.                                         @foreach (var item in listPartnerProjectVM)  
  8.                                         {  
  9.                                             <a href=«#» id=«idProject» class=«list-group-item» name=«@item.idPartnerProject»>@item.ProjectName</a>  
  10.                                         }  
  11.   
  12.                                     </div>  
  13.                                 </div>  
  14.                             } 

Here error occurred. please help me… 

У меня есть код и когда он выполняется, он выдает NullReferenceException, говоря:

Ссылка на объект не установлена ​​в экземпляр объекта.

Что это значит и что я могу сделать, чтобы исправить эту ошибку?

4b9b3361

Ответ 1

В чем причина?

Нижняя линия

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

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

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

  1. Объектные переменные, которые неинициализируются и, следовательно, не указывают на ничего. В этом случае, если вы NullReferenceException к свойствам или методам таких объектов, это вызывает NullReferenceException.
  2. Разработчик использует null намеренно, чтобы указать, что нет значимого значения. Обратите внимание, что С# имеет понятие обнуляемых типов данных для переменных (например, таблицы базы данных могут иметь поля с null значением) — вы можете присвоить им null чтобы указать, что в нем нет значения, например int? a = null; int? a = null; где знак вопроса указывает, что ему разрешено хранить значение null в переменной a. Вы можете проверить это с помощью if (a.HasValue) {...} или с if (a==null) {...}. Необязательные переменные, как и a этом примере, позволяют получить доступ к значению через a.Value явно или как обычно, через a.
    Обратите внимание, что доступ к нему через a.Value вызывает InvalidOperationException вместо NullReferenceException если a имеет значение null — вы должны сделать проверку заранее, то есть, если у вас есть другая переменная с возможностью NULL int b; то вам следует выполнять задания типа if (a.HasValue) { b = a.Value; } if (a.HasValue) { b = a.Value; } или короче, if (a != null) { b = a; } if (a != null) { b = a; }.

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

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

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

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

string foo = null;
foo.ToUpper();

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

отладка

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

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

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

Примеры

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

общий

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 может быть null или свойство User может быть null или свойство Identity может быть null.

непрямой

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

Если вы хотите избежать ссылки на null (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; }
}

Инициализаторы вложенных коллекций ведут себя одинаково:

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. Синтаксис инициализатора коллекции не создает коллекцию для 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.

Мероприятия

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

Плохие соглашения об именах:

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

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, вам нужно понять, что 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 в том порядке, в котором они отображаются в визуальном дереве. NullReferenceException будет поднято в случае ранних созданных элементов управления с обработчиками событий и т.д., 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(); // NullReference here!!
}

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

В ролях с, as

var myThing = someObject as Thing;

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

LINQ FirstOrDefault() и SingleOrDefault()

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

для каждого

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

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

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

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

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

Явно проверяю значение 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 на null метода и выставляю собственное исключение.

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

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() чтобы как можно скорее сломаться, когда это произойдет:

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 во время выполнения в режиме деблокирования.

Используйте GetValueOrDefault() для типов значений с null значением, чтобы предоставить значение по умолчанию, когда оно равно 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

Использовать нулевой коалесцирующий оператор: ?? [С#] или 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] для массивов (доступно на С# 6 и VB.NET 14):

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

var title = person.Title.ToUpper();

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

В С# 5 и ниже это можно защитить:

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

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

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

Это приведет к названию переменной бытия null, и призыв к ToUpper не выполняется, если person.Title является null.

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

// 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- й элемент.

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

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

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 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 просто возвращает объект, который при итерации будет запускать блок итератора.

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

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

// 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();
}

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

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

Замечание о нулевых различиях в небезопасном коде

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

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

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

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

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

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

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

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

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

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

Ответ 2

Исключение NullReference — Visual Basic

NullReference Exception для Visual Basic ничем не отличается от NullReference Exception в С#. В конце концов, они оба сообщают об одном и том же исключении, определенном в.NET Framework, которое они оба используют. Причины, уникальные для Visual Basic, встречаются редко (возможно, только один).

В этом ответе будут использоваться термины, синтаксис и контекст Visual Basic. Используемые примеры исходят из большого количества прошлых вопросов о переполнении стека. Это делается для того, чтобы максимизировать актуальность, используя ситуации, часто встречающиеся в сообщениях. Немного больше объяснений также предоставляется тем, кому это может понадобиться. Пример, похожий на ваш, очень вероятно указан здесь.

Заметка:

  1. Это основано на концепции: в вашем проекте нет кода для вставки. Он призван помочь вам понять, что вызывает NullReferenceException (NRE), как его найти, как его исправить и как его избежать. NRE может быть вызвано многими способами, поэтому это вряд ли будет вашей единственной встречей.
  2. Примеры (из Qaru posts) не всегда показывают лучший способ сделать что-то в первую очередь.
  3. Как правило, используется самое простое средство.

Основное значение

Сообщение «Объект, не установленный для экземпляра объекта» означает, что вы пытаетесь использовать объект, который не был инициализирован. Это сводится к одному из следующих:

  • Ваш код объявлял объектную переменную, но не инициализировал ее (создайте экземпляр или «создайте экземпляр»)
  • Что-то, что предполагал ваш код, инициализирует объект, не
  • Возможно, другой код преждевременно аннулировал объект, который все еще используется

Поиск причины

Поскольку проблема — это ссылка на объект, которая является » Nothing, ответ заключается в том, чтобы изучить их, чтобы узнать, какой из них. Затем определите, почему он не инициализирован. Держите мышь над различными переменными, и Visual Studio (VS) покажет их значения — виновником будет Nothing.

IDE debug display

Вы также должны удалить блоки Try/Catch из соответствующего кода, особенно те, где в блоке Catch ничего нет. Это приведет к сбою вашего кода при попытке использовать объект Nothing. Это то, что вы хотите, потому что оно идентифицирует точное местоположение проблемы и позволяет идентифицировать вызывающий объект.

MsgBox в Catch, который отображает Error while..., мало поможет. Этот метод также приводит к очень плохим вопросам, потому что вы не можете описать фактическое исключение, объект или даже строку кода, где это происходит.

Вы также можете использовать Locals Window (Debug → Windows → Locals) для проверки ваших объектов.

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

Смотрите также:

  • Контрольные точки
  • MSDN: Как использовать блок Try/Catch для выхвата исключений
  • MSDN: рекомендации по исключениям

Примеры и средства защиты

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

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, которая существует только в этом контексте (sub). Переменная reg с уровнем модуля Scope который вы будете использовать везде, останется Nothing.

Отсутствие оператора New является причиной № 1 NullReference Exceptions рассмотренной в рассмотренных вопросах.

Visual Basic пытается многократно очистить процесс, используя New: Использование New Operator создает новый объект и вызывает Sub New — конструктор — где ваш объект может выполнять любую другую инициализацию.

Чтобы быть ясным, Dim (или Private) объявляет только переменную и ее Type. Область действия переменной — независимо от того, существует ли она для всего модуля/класса или является локальной процедурой — определяется тем, где она объявлена. Private | Friend | Public 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, Option Infer As <Type> и 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 <Type> с 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) затруднит наличие элемента без действительного объекта:

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

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

  • Вывод опциона Infer
  • Область применения 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 Dataset был объявлен, но экземпляр никогда не создавался. 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 именем «Employee», поэтому результаты 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 может быть Nothing или элемент 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...


Визуальные базовые формы

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. В С#, в зависимости от того, как он кодируется, среда IDE сообщит, что Controls не существуют в текущем контексте, или «не может ссылаться на нестатический член». Итак, в какой-то степени это ситуация только с VB. Это также сложно, потому что это может привести к отказу каскада.

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

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

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

Поскольку никакой другой код в вашем событии 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 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.

Теперь вам будет легко найти, что вы знаете, что ищете: VS shows you the error of your ways

«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 «. Вы можете подавить предупреждение, заменив Exit Function на Return Nothing, но это не решит проблему. Все, что пытается использовать возврат, когда 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

Плохо реализованный 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 — площадка для дьявола. Это О.П. был озадачен, почему он был получать NRE в Finally блоке. В других ситуациях, пустой Catch может привести к чему-то еще намного дальше вниз по течению, идущему в сторону haywire, и заставить вас тратить время на то, чтобы не ошибиться в неполадке проблемы. («Тихая исключение», описанное выше, дает такое же развлекательное значение.)

средство

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


DBNull — это не то же самое, что и ничего

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

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

Значение System.DBNull указывает, что объект представляет отсутствующие или несуществующие данные. DBNull — это не то же самое, что и Nothing, что указывает на то, что переменная еще не была инициализирована.

средство

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

Если CheckBox с chkName не может быть найден (или существует в 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, он будет создавать столбцы, но он их не 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 имеет AllowUserToAddRows как 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 истинно.


My.Settings(StringCollection)

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

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

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

средство

Инициализировать коллекцию настроек в форме Load event handler, если/когда необходимо:

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

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


Ключевые моменты

Вероятно, вы забыли оператора » New.

или

То, что вы предположили, безупречно выполнило бы, чтобы вернуть инициализированный объект в ваш код, не сделал этого.

Не игнорируйте предупреждения компилятора (когда-либо) и используйте параметр Option Strict On (всегда).


Исключение NullReference MSDN

Ответ 3

Другой сценарий — когда вы приводите нулевой объект в тип значения. Например, код ниже:

object o = null;
DateTime d = (DateTime)o;

Это бросит NullReferenceException на приведение. Это кажется вполне очевидным в приведенном выше примере, но это может произойти в более сложных запоздалых сценариях, когда нулевой объект был возвращен из некоторого кода, который вам не принадлежит, и приведение, например, генерируется некоторой автоматической системой.

Одним из примеров этого является этот простой фрагмент привязки ASP.NET с элементом управления Calendar:

<asp:Calendar runat="server" SelectedDate="<%#Bind("Something")%>" />

Здесь SelectedDate на самом деле является свойством — типа DateTime — типа веб-элемента управления Calendar, и привязка может совершенно вернуть что-то нулевое. Неявный ASP.NET Generator создаст кусок кода, который будет эквивалентен приведенному выше коду. И это вызовет NullReferenceException, которое довольно трудно обнаружить, поскольку оно находится в сгенерированном ASP.NET коде, который прекрасно компилируется…

Ответ 4

Это означает, что рассматриваемая переменная ничем не указана. Я мог бы создать это так:

SqlConnection connection = null;
connection.Open();

Это вызовет ошибку, потому что, пока я объявил переменную «connection«, она ничего не указала. Когда я пытаюсь вызвать член «Open«, там нет ссылки для его разрешения, и он выдает ошибку.

Чтобы избежать этой ошибки:

  • Всегда инициализируйте свои объекты, прежде чем пытаться что-либо сделать с ними.
  • Если вы не уверены, является ли объект нулевым, проверьте его с помощью object == null.

Инструмент Resharper JetBrains определит каждое место в вашем коде, которое имеет возможность ошибки нулевой ссылки, позволяя вам установить нулевую проверку. Эта ошибка является источником ошибок номер один, IMHO.

Ответ 5

Это означает, что ваш код использовал ссылочную переменную объекта, которая была установлена ​​в значение null (т.е. она не ссылалась на экземпляр фактического объекта).

Чтобы предотвратить ошибку, объекты, которые могут быть пустыми, должны быть проверены на значение null перед использованием.

if (myvar != null)
{
    // Go ahead and use myvar
    myvar.property = ...
}
else
{
    // Whoops! myvar is null and cannot be used without first
    // assigning it to an instance reference
    // Attempting to use myvar here will result in NullReferenceException
}

Ответ 6

Помните, что независимо от сценария причина всегда одинакова в .NET:

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

Вы либо никогда не присваивали какую-либо переменную, никогда не создавали экземпляр значения, присвоенного переменной, или вы устанавливали переменную равную Nothing/null вручную, или вы вызывали функцию, которая устанавливает переменную в Nothing/null для вас.

Ответ 7

Примером этого исключения является: Когда вы пытаетесь проверить что-то, это null.

Например:

string testString = null; //Because it doesn't have a value (i.e. it null; "Length" cannot do what it needs to do)

if (testString.Length == 0) // Throws a nullreferenceexception
{
    //Do something
} 

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

По сравнению с ArgumentNullException, которое обычно выбрасывается как защитная мера, если метод ожидает, что переданное ему значение не является нулевым.

Дополнительная информация содержится в С# NullReferenceException и Null Parameter.

Ответ 8

Обновление С# 8.0, 2019: Обнуляемые ссылочные типы

С# 8.0 вводит пустые ссылочные типы и ненулевые ссылочные типы. Поэтому необходимо проверять только обнуляемые ссылочные типы, чтобы избежать исключения NullReferenceException.


Если вы не инициализировали ссылочный тип и хотите установить или прочитать одно из его свойств, оно выдаст исключение NullReferenceException.

Пример:

Person p = null;
p.Name = "Harry"; // NullReferenceException occurs here.

Вы можете просто избежать этого, проверив, не является ли переменная нулевой:

Person p = null;
if (p!=null)
{
    p.Name = "Harry"; // Not going to run to this point
}

Чтобы полностью понять, почему выбрасывается исключение NullReferenceException, важно знать разницу между типами значений и [ссылочными типами] [3].

Таким образом, если вы имеете дело с типами значений, NullReferenceExceptions не может возникнуть. Хотя при работе с ссылочными типами нужно соблюдать бдительность!

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

Типы ссылок (эти должны быть проверены):

  • динамический
  • объект
  • строка

Типы значений (вы можете просто игнорировать эти):

  • Числовые типы
  • Интегральные типы
  • Типы с плавающей точкой
  • десятичный
  • BOOL
  • Пользовательские структуры

Ответ 9

Другим случаем, когда NullReferenceExceptions может случиться, является (неправильное) использование оператора as:

class Book {
    public string Name { get; set; }
}
class Car { }

Car mycar = new Car();
Book mybook = mycar as Book;   // Incompatible conversion --> mybook = null

Console.WriteLine(mybook.Name);   // NullReferenceException

Здесь Book и Car являются несовместимыми типами; Car не может быть преобразован/брошен в Book. Когда этот бросок терпит неудачу, as возвращает null. Использование mybook после этого вызывает NullReferenceException.

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

Если вы ожидаете, что преобразование типа всегда будет успешным (т.е. Вы знаете, что объект должен быть впереди времени), тогда вы должны использовать бросок:

ComicBook cb = (ComicBook)specificBook;

Если вы не уверены в типе, но хотите попытаться использовать его как определенный тип, используйте его as:

ComicBook cb = specificBook as ComicBook;
if (cb != null) {
   // ...
}

Ответ 10

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

Пример:

string value = null;
if (value.Length == 0) // <-- Causes exception
{
    Console.WriteLine(value); // <-- Never reached
}

Ошибка исключения:

Необработанное исключение:

System.NullReferenceException: ссылка на объект не установлена ​​в экземпляр объекта. в Program.Main()

Ответ 11

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

В Visual Studio это обычно легко благодаря Visual Studio Debugger.


Во-первых, убедитесь, что правильная ошибка будет обнаружена — см. Как разрешить нарушение «Исключение System.NullReferenceException» в VS2010? Примечание 1

Затем либо » Начать с отладки» (F5), либо » Прикрепить» [отладчик VS] для выполнения процесса. Иногда может быть полезно использовать Debugger.Break, который предложит запустить отладчик.

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

Например, в следующей строке единственным кодом, который может вызвать исключение, является то, что myString значение null. Это можно проверить, посмотрев окно просмотра или выполнив выражения в окне Immediate.

var x = myString.Trim();

В более сложных случаях, таких как следующее, вам нужно будет использовать один из методов выше (Watch или Immediate Windows), чтобы проверить выражения, чтобы определить, было ли str1 пустым или str2 был нулевым.

var x = str1.Trim() + str2.Trim();

Когда место, где было выбрано исключение, было расположено, оно обычно тривиально по отношению к разуму назад, чтобы выяснить, где введено неверное значение —

Найдите время, необходимое для понимания причины исключения. Проверьте нулевые выражения. Проверьте предыдущие выражения, которые могли бы привести к таким нулевым выражениям. Добавьте точки останова и, по мере необходимости, выполните программу. Используйте отладчик.


1 Если Break on Throws является слишком агрессивным, и отладчик останавливается на NPE в библиотеке.NET или сторонних разработчиков, Break on User-Unhandled может использоваться для ограничения исключений. Кроме того, VS2012 представляет Just My Code, который я рекомендую также включить.

Если вы отлаживаетесь с включенным Just My Code, поведение немного отличается. При включенном Just My Code отладчик игнорирует исключения, связанные с привилегиями обычного языка (CLR), которые выходят за пределы My Code и не проходят через My Code

Ответ 12

Симон Моурир привел этот пример:

object o = null;
DateTime d = (DateTime)o;  // NullReferenceException

где преобразование (cast) unboxing из object (или из одного из классов System.ValueType или System.Enum или из типа интерфейса) в тип значения (кроме Nullable<>) само по себе дает NullReferenceException.

В другом направлении преобразование бокса из Nullable<> которое HasValue равно false для ссылочного типа, может дать null ссылку, которая затем может привести к NullReferenceException. Классический пример:

DateTime? d = null;
var s = d.ToString();  // OK, no exception (no boxing), returns ""
var t = d.GetType();   // Bang! d is boxed, NullReferenceException

Иногда бокс происходит по-другому. Например, с помощью этого не общего метода расширения:

public static void MyExtension(this object x)
{
  x.ToString();
}

следующий код будет проблематичным:

DateTime? d = null;
d.MyExtension();  // Leads to boxing, NullReferenceException occurs inside the body of the called method, not here.

Эти случаи возникают из-за специальных правил, которые использует среда выполнения при боксировании Nullable<> экземпляров.

Ответ 13

Добавление случая, когда имя класса для сущности, используемой в структуре сущности, такое же, как имя класса для файла с кодировкой веб-формы.

Предположим, что у вас есть веб-форма Contact.aspx, чей класс codebehind имеет контакт, и у вас есть контакт с именем организации.

Затем следующий код будет вызывать исключение NullReferenceException при вызове context.SaveChanges()

Contact contact = new Contact { Name = "Abhinav"};
var context = new DataContext();
context.Contacts.Add(contact);
context.SaveChanges(); // NullReferenceException at this line

Для полноты класса DataContext

public class DataContext : DbContext 
{
    public DbSet<Contact> Contacts {get; set;}
}

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

public partial class Contact 
{
    public string Name {get; set;}
}

Ошибка возникает, когда и класс entity, и codebehind находятся в одном пространстве имен.
Чтобы исправить это, переименуйте класс сущности или класс codebehind для Contact.aspx.

Причина
Я все еще не уверен в причине. Но всякий раз, когда какой-либо из классов сущностей расширяет System.Web.UI.Page, эта ошибка возникает.

Для обсуждения рассмотрим NullReferenceException в DbContext.saveChanges()

Ответ 14

Другой общий случай, когда можно получить это исключение, включает в себя насмешливые классы во время модульного тестирования. Независимо от используемой насмешливой структуры, вы должны убедиться, что все соответствующие уровни иерархии классов должным образом высмеиваются. В частности, все свойства HttpContext, на которые ссылается тестируемый код, должны быть изделены.

См. «Исключение NullReferenceException при проверке пользовательского атрибута AuthorizationAttribute» для несколько подробного примера.

Ответ 15

У меня есть другая перспектива ответить на это. Такие ответы «что еще можно сделать, чтобы избежать этого?»

При работе в разных слоях, например, в приложении MVC, контроллеру нужны службы для вызова бизнес-операций. В таких сценариях Контейнер инжекции зависимостей может использоваться для инициализации служб, чтобы избежать NullReferenceException. Это означает, что вам не нужно беспокоиться о проверке нулевого и просто вызвать службы с контроллера, как будто они всегда будут доступны (и инициализированы) как одиночный или прототип.

public class MyController
{
    private ServiceA serviceA;
    private ServiceB serviceB;

    public MyController(ServiceA serviceA, ServiceB serviceB)
    {
        this.serviceA = serviceA;
        this.serviceB = serviceB;
    }

    public void MyMethod()
    {
        // We don't need to check null because the dependency injection container 
        // injects it, provided you took care of bootstrapping it.
        var someObject = serviceA.DoThis();
    }
}

Ответ 16

В отношении «что мне делать с этим» может быть много ответов.

Более «формальный» способ предотвращения таких ошибок при разработке применяется дизайн по контракту в вашем коде. Это означает, что вам необходимо установить инварианты классов и/или даже предпосылки и постусловия функций и постусловий в вашей системе при разработке.

Короче говоря, инварианты класса гарантируют, что в вашем классе будут некоторые ограничения, которые не будут нарушены при нормальном использовании (и, следовательно, класс не попадет в несогласованное состояние). Предпосылки означают, что данные, данные как входные данные для функции/метода, должны соответствовать установленным ограничениям и никогда, а постусловия означает, что вывод функции/метода должен снова следовать установленным ограничениям, не нарушая их.
Условия контракта должны быть никогда нарушены во время выполнения программы без ошибок, поэтому дизайн по контракту проверяется на практике в режиме отладки, будучи отключенным в версиях, чтобы максимизировать развитую производительность системы.

Таким образом, вы можете избежать случаев NullReferenceException, которые являются результатом нарушения установленных ограничений. Например, если вы используете свойство объекта X в классе, а затем попытаетесь вызвать один из его методов, а X имеет нулевое значение, это приведет к NullReferenceException:

public X { get; set; }

public void InvokeX()
{
    X.DoSomething(); // if X value is null, you will get a NullReferenceException
}

Но если вы установите «свойство X никогда не должно иметь нулевого значения» в качестве предварительного условия метода, вы можете предотвратить описанный выше сценарий:

//Using code contracts:
[ContractInvariantMethod]
protected void ObjectInvariant () 
{
    Contract.Invariant ( X != null );
    //...
}

По этой причине существует проект Code Contracts для приложений .NET.

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

ОБНОВЛЕНИЕ: Стоит отметить, что этот термин был придуман Бертраном Мейером в связи с его дизайном языка программирования Эйфеля.

Ответ 17

A NullReferenceException возникает, когда мы пытаемся получить доступ к свойствам нулевого объекта или когда значение строки становится пустым и мы пытаемся получить доступ к строковым методам.

Например:

  • При использовании строкового метода пустой строки:

    string str = string.Empty;
    str.ToLower(); // throw null reference exception
    
  • Когда свойство нулевого объекта доступно:

    Public Class Person {
        public string Name { get; set; }
    }
    Person objPerson;
    objPerson.Name  /// throw Null refernce Exception 
    

Ответ 18

TL; DR: Попробуйте использовать Html.Partial вместо Renderpage


Я получал Object reference not set to an instance of an object, когда я пытался отобразить представление в представлении, отправив ему модель, например:

@{
    MyEntity M = new MyEntity();
}
@RenderPage("_MyOtherView.cshtml", M); // error in _MyOtherView, the Model was Null

Отладка показала, что модель была Null внутри MyOtherView. Пока я не изменил его на:

@{
    MyEntity M = new MyEntity();
}
@Html.Partial("_MyOtherView.cshtml", M);

И это сработало.

Кроме того, причина, по которой у меня не было Html.Partial, заключалась в том, что Visual Studio иногда генерирует строчные строки с ошибками в Html.Partial, если она находится внутри другого построенного цикла foreach, хотя это не ошибка:

@inherits System.Web.Mvc.WebViewPage
@{
    ViewBag.Title = "Entity Index";
    List<MyEntity> MyEntities = new List<MyEntity>();
    MyEntities.Add(new MyEntity());
    MyEntities.Add(new MyEntity());
    MyEntities.Add(new MyEntity());
}
<div>
    @{
        foreach(var M in MyEntities)
        {
            // Squiggly lines below. Hovering says: cannot convert method group 'partial' to non-delegate type Object, did you intend to envoke the Method?
            @Html.Partial("MyOtherView.cshtml");
        }
    }
</div>

Но я смог запустить приложение без проблем с этой «ошибкой». Я смог избавиться от ошибки, изменив структуру цикла foreach, чтобы выглядеть так:

@foreach(var M in MyEntities){
    ...
}

Хотя у меня такое чувство, что Visual Studio неправильно интерпретирует амперсанды и скобки.

Ответ 19

Что вы можете сделать с этим?

Здесь есть много хороших ответов, объясняющих, что такое пустая ссылка и как ее отладить. Но очень мало о том, как предотвратить проблему или, по крайней мере, сделать ее легче поймать.

Проверить аргументы

Например, методы могут проверять разные аргументы, чтобы увидеть, являются ли они нулевыми и выбрасывают ArgumentNullException, исключение, явно созданное для этой цели.

Конструктор для ArgumentNullException даже принимает имя параметра и сообщение в качестве аргументов, чтобы вы могли точно сказать разработчику, в чем проблема.

public void DoSomething(MyObject obj) {
    if(obj == null) 
    {
        throw new ArgumentNullException("obj", "Need a reference to obj.");
    }
}

Использовать инструменты

Есть также несколько библиотек, которые могут помочь. Например, «Resharper» может предоставить вам предупреждения при написании кода, особенно если вы используете их атрибут: NotNullAttribute

Там «Контракты кода Microsoft», где вы используете синтаксис типа Contract.Requires(obj != null), который дает вам проверку выполнения и компиляцию: Представление кодовых контрактов.

Там также «PostSharp», который позволит вам просто использовать такие атрибуты:

public void DoSometing([NotNull] obj)

Сделав это и сделав часть PostSharp вашего процесса сборки obj, будет проверяться на нуль во время выполнения. См.: Ошибка проверки PostSharp

Решение простого кода

Или вы всегда можете использовать свой собственный подход, используя простой старый код. Например, вот структура, которую вы можете использовать, чтобы поймать нулевые ссылки. Он смоделирован по той же концепции, что и Nullable<T>:

[System.Diagnostics.DebuggerNonUserCode]
public struct NotNull<T> where T: class
{
    private T _value;

    public T Value
    {
        get
        {
            if (_value == null)
            {
                throw new Exception("null value not allowed");
            }

            return _value;
        }
        set
        {
            if (value == null)
            {
                throw new Exception("null value not allowed.");
            }

            _value = value;
        }
    }

    public static implicit operator T(NotNull<T> notNullValue)
    {
        return notNullValue.Value;
    }

    public static implicit operator NotNull<T>(T value)
    {
        return new NotNull<T> { Value = value };
    }
}

Вы бы очень похожи на то, как вы использовали бы Nullable<T>, кроме как с целью достижения совершенно противоположного — не разрешать null. Вот несколько примеров:

NotNull<Person> person = null; // throws exception
NotNull<Person> person = new Person(); // OK
NotNull<Person> person = GetPerson(); // throws exception if GetPerson() returns null

NotNull<T> неявно отображается в и из T, поэтому вы можете использовать его в любом месте, где это необходимо. Например, вы можете передать объект Person методу, который принимает NotNull<Person>:

Person person = new Person { Name = "John" };
WriteName(person);

public static void WriteName(NotNull<Person> person)
{
    Console.WriteLine(person.Value.Name);
}

Как вы можете видеть выше, как с помощью nullable, вы получите доступ к базовому значению с помощью свойства Value. Кроме того, вы можете использовать явный или неявный листинг, вы можете увидеть пример с возвращаемым значением ниже:

Person person = GetPerson();

public static NotNull<Person> GetPerson()
{
    return new Person { Name = "John" };
}

Или вы даже можете использовать его, когда метод просто возвращает T (в данном случае Person), выполнив бросок. Например, следующий код будет похож на код выше:

Person person = (NotNull<Person>)GetPerson();

public static Person GetPerson()
{
    return new Person { Name = "John" };
}

Объединить с расширением

Объедините NotNull<T> с методом расширения, и вы можете охватить еще больше ситуаций. Вот пример того, как выглядит метод расширения:

[System.Diagnostics.DebuggerNonUserCode]
public static class NotNullExtension
{
    public static T NotNull<T>(this T @this) where T: class
    {
        if (@this == null)
        {
            throw new Exception("null value not allowed");
        }

        return @this;
    }
}

И вот пример того, как он может быть использован:

var person = GetPerson().NotNull();

GitHub

Для справки я сделал код выше, доступный на GitHub, вы можете найти его по адресу:

https://github.com/luisperezphd/NotNull

Функция родственного языка

С# 6.0 представил «нуль-условный оператор», который немного помогает в этом. С помощью этой функции вы можете ссылаться на вложенные объекты, и если какой-либо из них null, все выражение возвращает null.

Это уменьшает количество нулевых проверок, которые вы должны выполнить в некоторых случаях. Синтаксис заключается в том, чтобы поставить вопросительный знак перед каждой точкой. Возьмите следующий код, например:

var address = country?.State?.County?.City;

Предположим, что country является объектом типа country, который имеет свойство, называемое State и т.д. Если country, State, County или City есть null, то address will be null . Therefore you only have to check whether address is null`.

Это отличная функция, но она дает вам меньше информации. Это не делает очевидным, какой из 4 является нулевым.

Встроенный как Nullable?

С# имеет красивое сокращение для Nullable<T>, вы можете сделать что-то нулевое, поставив знак вопроса после такого типа int?.

Было бы неплохо, если бы у С# было что-то вроде структуры NotNull<T> выше и имела аналогичную стенографию, может быть, восклицательный знак (!), чтобы вы могли написать что-то вроде: public void WriteName(Person! person).

Ответ 20

Исключить исключение NullReferenceException можно с помощью Null-условных операторов в С# 6 и написать меньше кода для обработки нулевых проверок.

Он использовал для проверки значения null перед выполнением операции доступа к члену (?.) или индекса (? [).

Пример

  var name = p?.Spouse?.FirstName;

эквивалентно:

    if (p != null)
    {
        if (p.Spouse != null)
        {
            name = p.Spouse.FirstName;
        }
    }

В результате имя будет пустым, если p равно null или p.Spouse — null.

В противном случае имени переменной будет присвоено значение p.Spouse.FirstName.

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

Ответ 21

Строка ошибки «Ссылка на объект не установлена ​​в экземпляр объекта.» заявляет, что вы не присвоили экземпляр объекта объектной ссылке, и все же вы получаете доступ к свойствам/методам этого объекта.

например: допустим, что у вас есть класс с именем myClass, и он содержит одно свойство prop1.

public Class myClass
{
   public int prop1 {get;set;}
}

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

public class Demo
{
     public void testMethod()
     {
        myClass ref = null;
        ref.prop1 = 1;  //This line throws error
     }
}

выше строка выдает ошибку, потому что ссылка класса myClass объявлена, но не создана, или экземпляр объекта не назначен для referecne этого класса.

Чтобы исправить это, вы должны создать экземпляр (присвоить объект ссылке на этот класс).

public class Demo
{
     public void testMethod()
     {
        myClass ref = null;
        ref = new myClass();
        ref.prop1 = 1;  
     }
}

Ответ 22

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

Случай с краем # 1: одновременный доступ к словарю

Общие словари в .NET не являются потокобезопасными, и иногда они могут бросать NullReference или даже (чаще) a KeyNotFoundException при попытке получить доступ к ключу из двух параллельных потоков. Исключение в этом случае вводит в заблуждение.

Кромка # 2: небезопасный код

Если a NullReferenceException выбрано кодом unsafe, вы можете посмотреть на переменные указателя и проверить их на IntPtr.Zero или что-то еще. Это одно и то же ( «исключение нулевого указателя» ), но в небезопасном коде переменные часто переводятся в типы значений/массивы и т.д., И вы ударяете головой о стену, задаваясь вопросом, как тип значения может исключение.

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

Ответ 23

Ссылка NullReferenceException или Object, не установленная на экземпляр объекта, возникает, когда объект класса, который вы пытаетесь использовать, не создается.
Например:

Предположим, что у вас есть класс с именем Student.

public class Student
{
    private string FirstName;
    private string LastName;
    public string GetFullName()
    {
        return FirstName + LastName;
    }
}

Теперь рассмотрим другой класс, в котором вы пытаетесь получить полное имя студента.

public class StudentInfo
{      
    public string GetStudentName()
    {
        Student s;
        string fullname = s.GetFullName();
        return fullname;
    }        
}

Как видно из приведенного выше кода, утверждение
Студент s — объявляет только переменную типа Student, обратите внимание, что класс Student не создается в этой точке.
Следовательно, когда выполняется инструкция s.GetFullName(), она выкинет исключение NullReferenceException.

Ответ 24

Ну, простыми словами:

Вы пытаетесь получить доступ к объекту, который не создан или в данный момент отсутствует в памяти.

Итак, как справиться с этим:

  1. Отладьте и дайте отладчику сломаться… Он напрямую приведет вас к сломанной переменной… Теперь ваша задача просто исправить это… Используя ключевое слово new в соответствующем месте.

  2. Если это вызвано в некоторых командах базы данных, потому что объект не присутствует, тогда все, что вам нужно сделать, это сделать нулевую проверку и обработать ее:

    if (i == null) {
        // Handle this
    }
    
  3. Сложнее всего… если GC уже собрал объект… Обычно это происходит, если вы пытаетесь найти объект с помощью строк… То есть, находя его по имени объекта, может случиться так, что GC может уже Вычистил это… Это трудно найти и станет большой проблемой… Лучший способ справиться с этим — делать нулевые проверки везде, где это необходимо во время процесса разработки. Это сэкономит вам много времени.

Под поиском по имени я подразумеваю некоторую инфраструктуру, позволяющую вам FIndObjects использовать строки, и код может выглядеть следующим образом: FindObject («ObjectName»);

Ответ 25

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

Пример:

string postalcode=Customer.Address.PostalCode; 
//if customer or address is null , this will through exeption

здесь, если адрес равен NULL, тогда вы получите исключение NullReferenceException.

Итак, как практика, мы всегда должны использовать проверку нуля, прежде чем обращаться к свойствам в таких объектах (особенно в родовых)

string postalcode=Customer?.Address?.PostalCode;
//if customer or address is null , this will return null, without through a exception

Ответ 26

Буквально самый простой способ исправить NullReferenceExeption имеет два пути.
Если у вас есть GameObject, например, с прикрепленным script и переменной с именем rb (rigidbody), эта переменная начнет пустую, когда вы начнете игру.
Вот почему вы получаете NullReferenceExeption, потому что на компьютере нет данных, хранящихся в этой переменной.

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

  • Добавьте RigidBody к вашему объекту с помощью AddComponent > Физикa > Rigidbody
    Затем зайдите в свой script и введите rb = GetComponent<Rigidbody>();
    Эта строка кода лучше всего работает под вашими функциями Start() или Awake().
  • Вы можете добавить компонент программно и назначить переменную одновременно с одной строкой кода: rb = AddComponent<RigidBody>();

Дальнейшие примечания. Если вы хотите, чтобы единство добавляло компонент к вашему объекту, и вы, возможно, забыли добавить его, вы можете ввести [RequireComponent(typeof(RigidBody))] над объявлением класса (пробел ниже всех ваших приложений).
Наслаждайтесь и получайте удовольствие от игр!

Ответ 27

Для типов ссылок по умолчанию используется null, чтобы указать, что они не ссылаются на какой-либо объект. Следовательно, если вы попытаетесь получить доступ к объекту, на который ссылаетесь, и его нет, вы получите NullReferenceException.

Для Ex:

SqlConnection connection = null;
connection.Open();

Когда вы запустите этот код, вы получите:

System.NullReferenceException: Object reference not set to an instance of an object.

Вы можете избежать этой ошибки путем кодирования следующим образом:

if (connection != null){
    connection.Open();
}

Примечание. Чтобы избежать этой ошибки, вы всегда должны инициализировать свои объекты, прежде чем пытаться что-либо сделать с ними.

Ответ 28

Если вы получаете это сообщение во время сохранения или компиляции сборки, просто закройте все файлы, а затем откройте любой файл для компиляции и сохранения.

Для меня причина в том, что я переименовал файл, и старый файл все еще был открыт.

Ответ 29

Чтобы использовать методы и член объекта, вам сначала нужно создать этот объект. Если вы его не создали (переменная, которая должна содержать объект, не инициализируется), но вы пытаетесь использовать его методы или переменные, вы получите эту ошибку.

Иногда вы можете просто забыть инициализировать.

Отредактировано: new не может вернуть значение null, но исключение fire при неудаче. Давно это было на некоторых языках, но не больше. Спасибо @John Saunders за указание на это.

Ответ 30

Это означает, что вы пытаетесь манипулировать тем, что имеет ссылку, но еще не инициализировано

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

Используйте контрольные точки, часы, проверяйте свои значения varibale.

Следить за трассировкой стека и искать точные строки и столбцы, которые создают проблему.

Понравилась статья? Поделить с друзьями:
  • Error linker link exe not found
  • Error ld returned 1 exit status как исправить linux
  • Error ld returned 1 exit status как исправить codeblocks
  • Error ld returned 1 exit status code blocks
  • Error launching unity player