我有一些代码,执行时会抛出NullReferenceException
,说:
你调用的对象是空的。
这是什么意思,我该怎么做才能解决此错误?
我有一些代码,执行时会抛出NullReferenceException
,说:
你调用的对象是空的。
这是什么意思,我该怎么做才能解决此错误?
Answers:
您正在尝试使用null
(或Nothing
在VB.NET中)。这意味着您要么将其设置为null
,要么根本不将其设置为任何东西。
像其他任何东西一样,null
被传遍了。如果null
在方法“ A”中,则可能是方法“ B”将a传递null
给方法“ A”。
null
可以具有不同的含义:
NullReferenceException
。null
来指示没有可用的有意义的值。请注意,C#具有变量可为空的数据类型的概念(例如数据库表可以具有可为空的字段)-您可以分配null
给它们以指示其中没有存储任何值,例如int? a = null;
,问号指示允许在其中存储null。可变的a
。您可以使用if (a.HasValue) {...}
或使用进行检查if (a==null) {...}
。像a
本示例一样,可空变量允许通过a.Value
显式访问值,或者像通过正常一样访问值a
。a.Value
抛出InvalidOperationException
,而不是NullReferenceException
如果a
ISnull
-您应该事先进行检查,即,如果您有另一个on-nullable变量,int b;
则应该进行like if (a.HasValue) { b = a.Value; }
或更短的赋值if (a != null) { b = a; }
。本文的其余部分将更详细地说明错误,并指出许多程序员经常犯的错误,这些错误可能导致程序错误NullReferenceException
。
该runtime
扔NullReferenceException
总是意味着同样的事情:你要使用的引用,引用未初始化(或它一次初始化,但不再初始化)。
这意味着引用为null
,并且您无法通过null
引用访问成员(例如方法)。最简单的情况:
string foo = null;
foo.ToUpper();
这将NullReferenceException
在第二行抛出,因为您无法ToUpper()
在string
指向的引用上调用实例方法null
。
您如何找到来源NullReferenceException
?除了查看将要引发的异常本身之外,Visual Studio中的常规调试规则也适用:放置战略断点并检查变量(通过将鼠标悬停在其名称上,然后打开(快速)监视窗口或使用各种调试面板(例如本地和自动)。
如果要找出参考的位置或未设置的位置,请右键单击其名称,然后选择“查找所有参考”。然后,可以在每个找到的位置放置一个断点,并在连接了调试器的情况下运行程序。调试器每次在这样的断点处中断时,都需要确定您是否希望引用为非空,检查变量,并在期望时验证它是否指向实例。
通过以这种方式遵循程序流程,您可以找到实例不应为null的位置以及未正确设置实例的原因。
可能引发异常的一些常见方案:
ref1.ref2.ref3.member
如果ref1或ref2或ref3为空,则将获得一个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.
}
}
如果要避免子(Person)空引用,可以在父(Book)对象的构造函数中对其进行初始化。
嵌套对象初始化器也是如此:
Book b1 = new Book
{
Author = { Age = 45 }
};
这转化为
Book b1 = new Book();
b1.Author.Age = 45;
使用new
关键字时,它只会创建的新实例Book
,而不创建的新实例Person
,因此Author
该属性仍然是null
。
public class Person
{
public ICollection<Book> Books { get; set; }
}
public class Book
{
public string Title { get; set; }
}
嵌套集合的Initializers
行为相同:
Person p1 = new Person
{
Books = {
new Book { Title = "Title1" },
new Book { Title = "Title2" },
}
};
这转化为
Person p1 = new Person();
p1.Books.Add(new Book { Title = "Title1" });
p1.Books.Add(new Book { Title = "Title2" });
该new Person
只创建一个实例Person
,但Books
集合仍然是null
。集合Initializer
语法不会为创建一个集合p1.Books
,它仅转换为p1.Books.Add(...)
语句。
int[] numbers = null;
int n = numbers[0]; // numbers is null. There is no array to index.
Person[] people = new Person[5];
people[0].Age = 20 // people[0] is null. The array was allocated but not
// initialized. There is no Person to set the Age for.
long[][] array = new long[1][];
array[0][0] = 3; // is null because only the first dimension is yet initialized.
// Use array[0] = new long[2]; first.
Dictionary<string, int> agesForNames = null;
int age = agesForNames["Bob"]; // agesForNames is null.
// There is no Dictionary to perform the lookup.
public class Person
{
public string Name { get; set; }
}
var people = new List<Person>();
people.Add(null);
var names = from p in people select p.Name;
string firstName = names.First(); // Exception is thrown here, but actually occurs
// on the line above. "p" is null because the
// first element we added to the list is null.
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
}
}
###Bad Naming Conventions:
If you named fields differently from locals, you might have realized that you never initialized the field.
公共类Form1 {私人客户客户;
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;
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!";
}
}
// if the "FirstName" session value has not yet been set,
// then this line will throw a NullReferenceException
string firstName = Session["FirstName"].ToString();
如果引用的属性时出现异常@Model
的一个ASP.NET MVC View
,你需要了解的是,Model
获取你的操作方法设定,当您return
的视图。当从控制器返回空模型(或模型属性)时,视图访问它时会发生异常:
// Controller
public class Restaurant:Controller
{
public ActionResult Search()
{
return View(); // Forgot the provide a Model here.
}
}
// Razor view
@foreach (var restaurantSearch in Model.RestaurantSearch) // Throws.
{
}
<p>@Model.somePropertyName</p> <!-- Also throws -->
WPF
控件在调用过程InitializeComponent
中按照它们在视觉树中的显示顺序创建。对于NullReferenceException
带有事件处理程序等的早期创建的控件,将引发A ,在事件InitializeComponent
引用期间将触发A。
例如 :
<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
,而忽略设计哲学问题)至少可以解决NullReferenceException
此处的问题。
as
var myThing = someObject as Thing;
当强制转换失败(并且本身为null)时,此方法不会抛出an,InvalidCastException
但会返回a 。因此请注意。null
someObject
FirstOrDefault()
和SingleOrDefault()
普通版本First()
并Single()
在没有内容时引发异常。在这种情况下,“ OrDefault”版本返回null。因此请注意。
foreach
尝试迭代null集合时引发。通常由null
返回集合的方法的意外结果引起。
List<int> list = null;
foreach(var v in list) { } // exception
更实际的示例-从XML文档中选择节点。如果未找到节点,但初始调试显示所有属性均有效,则会抛出该异常:
foreach (var node in myData.MyXml.DocumentNode.SelectNodes("//Data"))
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
方法调用并引发自定义异常。您还可以抛出一个自定义异常,仅在调用代码中将其捕获:
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()
的nullable
值类型提供一个默认值,当他们null
。DateTime? appointment = null;
Console.WriteLine(appointment.GetValueOrDefault(DateTime.Now));
// Will display the default value provided (DateTime.Now), because appointment is null.
appointment = new DateTime(2022, 10, 20);
Console.WriteLine(appointment.GetValueOrDefault(DateTime.Now));
// Will display the appointment date, not the default
??
[C#]或If()
[VB]。null
遇到a时提供默认值的简写:
IService CreateService(ILogger log, Int32? frobPowerLevel)
{
var serviceImpl = new MyService(log ?? NullLog.Instance);
// Note that the above "GetValueOrDefault()" can also be rewritten to use
// the coalesce operator:
serviceImpl.FrobPowerLevel = frobPowerLevel ?? 5;
}
?.
或?[x]
用于数组(在C#6和VB.NET 14中可用):有时也称为安全导航或Elvis(形状正确)操作员。如果运算符左侧的表达式为null,则不会对右侧进行求值,而是返回null。这意味着像这样的情况:
var title = person.Title.ToUpper();
如果此人没有标题,这将引发异常,因为它试图调用ToUpper
具有空值的属性。
在C# 5
下面,可以用以下方法保护它:
var title = person.Title == null ? null : person.Title.ToUpper();
现在,title变量将为null而不是引发异常。C#6为此引入了一个较短的语法:
var title = person.Title?.ToUpper();
这将导致title变量为null
,ToUpper
如果person.Title
为,则不会调用null
。
当然,您仍然必须检查title
null或将null条件运算符与null合并运算符(??
)一起使用以提供默认值:
// regular null check
int titleLength = 0;
if (title != null)
titleLength = title.Length; // If title is null, this would throw NullReferenceException
// combining the `?` and the `??` operator
int titleLength = title?.Length ?? 0;
同样,对于数组,您可以使用?[i]
以下方法:
int[] myIntArray=null;
var i=5;
int? elem = myIntArray?[i];
if (!elem.HasValue) Console.WriteLine("No value");
这将执行以下操作:如果myIntArray
为null,则表达式返回null,您可以安全地对其进行检查。如果包含数组,则将执行以下操作:
elem = myIntArray[i];
并返回该i<sup>th</sup>
元素。
其中引入了C# 8
null上下文和nullable引用类型,它们对变量执行静态分析,并在值可能为null或已设置为null时向编译器发出警告。可为空的引用类型允许将类型明确地允许为null。
可以使用文件中的Nullable
元素为项目设置可为空的注释上下文和可为空的警告上下文csproj
。该元素配置编译器如何解释类型的可空性以及生成什么警告。有效设置为:
使用与可为空的值类型相同的语法来记录可为空的引用类型:将a ?
附加到变量的类型。
C#
支持“迭代器块”(在其他一些流行语言中称为“生成器”)。由于延迟执行,在迭代器块中调试空引用异常可能特别棘手:
public IEnumerable<Frob> GetFrobs(FrobFactory f, int count)
{
for (int i = 0; i < count; ++i)
yield return f.MakeFrob();
}
...
FrobFactory factory = whatever;
IEnumerable<Frobs> frobs = GetFrobs();
...
foreach(Frob frob in frobs) { ... }
如果whatever
结果null
那么MakeFrob
就会抛出。现在,您可能会认为正确的做法是:
// DON'T DO THIS
public IEnumerable<Frob> GetFrobs(FrobFactory f, int count)
{
if (f == null)
throw new ArgumentNullException("f", "factory must not be null");
for (int i = 0; i < count; ++i)
yield return f.MakeFrob();
}
为什么会这样呢?因为迭代器块直到!才实际运行foreach
。对的调用GetFrobs
仅返回一个对象,该对象在迭代时将运行迭代器块。
通过编写空校验这样可以防止空取消引用,但您将空参数例外点迭代,不给点呼叫,那就是非常混乱调试。
正确的解决方法是:
// 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
调用when时,将立即GetFrobsForReal
执行空检查,然后在序列迭代时执行。
如果您检查LINQ
对象的参考源,您会发现整个使用了这种技术。编写起来有点笨拙,但是它使调试null错误更加容易。 优化代码以方便调用者,而不是作者。
C#
顾名思义,它具有“不安全”模式,这是非常危险的,因为不执行提供内存安全和类型安全的常规安全机制。除非您对内存的工作原理有透彻和深入的了解,否则不要编写不安全的代码。
在不安全模式下,您应该意识到两个重要事实:
要了解为什么会这样,它有助于首先了解.NET如何产生null取消引用异常。(这些详细信息适用于Windows上运行的.NET;其他操作系统使用类似的机制。)
内存虚拟化在Windows
;每个进程都会获得由操作系统跟踪的许多“页面”内存的虚拟内存空间。内存的每个页面上都设置有标志,这些标志确定如何使用它:读取,写入,执行等。该最低页面被标记为“产生的任何方式,如果使用过的错误”。
空指针和in C#
中的空引用都在内部表示为数字零,因此任何尝试将其取消引用到其相应的内存中的操作都会导致操作系统产生错误。然后,.NET运行时将检测到此错误,并将其转变为null取消引用异常。
这就是为什么同时取消引用空指针和空引用会产生相同的异常的原因。
那第二点呢?取消引用落在虚拟内存最低页中的任何无效指针会导致相同的操作系统错误,从而导致相同的异常。
为什么这有意义?好吧,假设我们有一个包含两个int的结构,以及一个等于null的非托管指针。如果尝试取消引用结构中的第二个int,CLR
则不会尝试访问零位置的存储;它将访问第四位置的存储。但从逻辑上讲,这是一个空取消引用,因为我们要通过空到达该地址。
如果您使用的是不安全的代码,并且会收到null解除引用异常,则请注意,有问题的指针不必为null。它可以在最低页面的任何位置,并且将产生此异常。
在NullReference Exception
为Visual Basic中是从一个在任何不同的C# 。毕竟,它们都报告了它们都使用的.NET Framework中定义的相同异常。Visual Basic特有的原因很少(也许只有一个)。
该答案将使用Visual Basic术语,语法和上下文。使用的示例来自大量过去的Stack Overflow问题。这是通过使用帖子中经常出现的各种情况来最大化相关性。还为可能需要的人提供了更多解释。此处很可能列出与您类似的示例。
注意:
NullReferenceException
(NRE)的原因,查找方法,修复方法以及避免方法。NRE可以通过多种方式引起,因此这不可能是您唯一遇到的问题。消息“对象未设置为对象的实例”表示您正在尝试使用尚未初始化的对象。归结为以下之一:
由于问题是一个对象引用,即Nothing
,答案是检查它们以找出哪个对象。然后确定为什么不初始化。将鼠标悬停在各个变量上,Visual Studio(VS)将显示其值-罪魁祸首是Nothing
。
您还应该从相关代码中删除任何Try / Catch块,尤其是在Catch块中没有任何内容的地方。尝试使用的对象时,这将导致您的代码崩溃Nothing
。这就是您想要的,因为它将识别出问题的确切位置,并允许您识别导致问题的对象。
MsgBox
捕获中显示的A Error while...
几乎没有帮助。这种方法还会导致非常糟糕的堆栈溢出问题,因为您无法描述实际的异常,所涉及的对象甚至发生异常的代码行。
您还可以使用Locals Window
(Debug-> Windows-> Locals)检查对象。
一旦知道了问题所在和出处,通常比发布新问题要容易得多,而且速度也更快。
也可以看看:
Dim reg As CashRegister
...
TextBox1.Text = reg.Amount ' NRE
问题是Dim
没有创建CashRegister 对象;它只声明一个名为reg
该类型的变量。声明对象变量和创建实例是两件事。
补救
该New
操作通常可以用来当你把它声明创建实例:
Dim reg As New CashRegister ' [New] creates instance, invokes the constructor
' Longer, more explicit form:
Dim reg As CashRegister = New CashRegister
仅在以后创建实例时:
Private reg As CashRegister ' Declare
...
reg = New CashRegister() ' Create instance
注意:请勿Dim
在过程中再次使用,包括构造函数(Sub New
):
Private reg As CashRegister
'...
Public Sub New()
'...
Dim reg As New CashRegister
End Sub
这将创建一个局部变量,reg
仅在该上下文(子)中存在。您将在其他地方使用的reg
具有模块级别的变量Scope
仍然存在Nothing
。
New
NullReference Exceptions
在所检查的堆栈溢出问题中缺少操作员是第一位原因。Visual Basic尝试使用
New
以下方法反复使过程清晰:使用New
运算符创建一个新对象,并调用Sub New
构造函数,您的对象可以在其中执行任何其他初始化。
明确地说,Dim
(或Private
)仅声明一个变量及其Type
。变量的作用域 -是否在整个模块/类中存在,还是过程的局部变量-由声明位置确定。Private | Friend | Public
定义访问级别,而不是Scope。
有关更多信息,请参见:
数组也必须实例化:
Private arr as String()
仅声明了此数组,未创建。有几种初始化数组的方法:
Private arr as String() = New String(10){}
' or
Private arr() As String = New String(10){}
' For a local array (in a procedure) and using 'Option Infer':
Dim arr = New String(10) {}
注意:从VS 2010开始,使用文字和初始化本地数组时Option Infer
,As <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
使用a 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
有关更多信息,请参见:
.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
也不会创建internal的实例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)
Class。
与数据库礼物的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会警告您可能会发生这种情况:
当声明为模块/类级别的变量时(如所示)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
vs 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
也可能什么都不是。的补救是在一个时间测试整个链或对象的一个的路径:
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)
除其他外,此代码无法预期用户可能未在一个或多个UI控件中选择某些内容。 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的相当普遍的方法。在C#中,根据其编码方式,IDE将报告Controls
当前上下文中不存在的内容,或“无法引用非静态成员”。因此,在某种程度上,这是仅VB的情况。这也很复杂,因为它可能导致失败的级联。
数组和集合不能以这种方式初始化。此初始化代码将在构造函数创建Form
或之前运行Controls
。结果是:
somevar
分配将导致立即NRE因为没有什么不具有.Text
财产稍后引用数组元素将导致NRE。如果您在中执行此操作Form_Load
,则由于一个奇怪的错误,IDE 可能不会在发生异常时报告该异常。当您的代码尝试使用该数组时,该异常将在以后弹出。这篇文章详细介绍了这种 “无声例外” 。就我们的目的而言,关键是当创建表单(Sub New
或Form Load
事件)时发生灾难性事件时,可能不会报告异常,代码将退出过程并仅显示表单。
由于NRE之后,您的Sub New
或Form Load
事件中不会再有其他代码运行,因此许多其他事情可以保留未初始化。
Sub Form_Load(..._
'...
Dim name As String = NameBoxes(2).Text ' NRE
' ...
' More code (which will likely not be executed)
' ...
End Sub
请注意,这适用于所有和所有控件和组件引用,这些引用使它们在以下位置非法:
Public Class Form1
Private myFiles() As String = Me.OpenFileDialog1.FileName & ...
Private dbcon As String = OpenFileDialog1.FileName & ";Jet Oledb..."
Private studentName As String = TextBox13.Text
部分补救
奇怪的是,VB没有提供警告,但补救措施是在窗体级别声明容器,但在控件确实存在时在窗体加载事件处理程序中初始化它们。只要您的代码在调用之后,就可以完成此操作:Sub New
InitializeComponent
' Module level declaration
Private NameBoxes as TextBox()
Private studentName As String
' Form Load, Form Shown or Sub New:
'
' Using the OP's approach (illegal using OPTION STRICT)
NameBoxes = New TextBox() {Me.Controls("TextBox1"), Me.Controls("TestBox2"), ...)
studentName = TextBox32.Text ' For simple control references
阵列代码可能还没有走出困境。容器控件中的任何控件(如a GroupBox
或Panel
)都不会在中找到Me.Controls
;它们将位于该Panel或GroupBox的Controls集合中。拼写错误的控件名称("TeStBox2"
)也不会返回控件。在这种情况下,Nothing
将再次存储在这些数组元素中,并且在您尝试引用它时将产生NRE。
现在,您知道要查找的内容时,应该很容易找到它们:
“ Button2”位于 Panel
补救
而不是使用表单的Controls
集合通过名称进行间接引用,而应使用控件引用:
' Declaration
Private NameBoxes As TextBox()
' Initialization - simple and easy to read, hard to botch:
NameBoxes = New TextBox() {TextBox1, TextBox2, ...)
' Initialize a List
NamesList = New List(Of TextBox)({TextBox1, TextBox2, TextBox3...})
' or
NamesList = New List(Of TextBox)
NamesList.AddRange({TextBox1, TextBox2, TextBox3...})
Private bars As New List(Of Bars) ' Declared and created
Public Function BarList() As List(Of Bars)
bars.Clear
If someCondition Then
For n As Integer = 0 to someValue
bars.Add(GetBar(n))
Next n
Else
Exit Function
End If
Return bars
End Function
在这种情况下,IDE会警告您“ 并非所有路径都会返回值,并且NullReferenceException
可能会导致结果 ”。您可以通过替换Exit Function
为来禁止显示警告,Return Nothing
但这不能解决问题。任何尝试使用when的返回值someCondition = False
都会导致NRE:
bList = myFoo.BarList()
For Each b As Bar in bList ' EXCEPTION
...
补救
用替换Exit Function
功能Return bList
。返回空 List
与returning是不同的Nothing
。如果返回的对象可能是Nothing
,请在使用它之前进行测试:
bList = myFoo.BarList()
If bList IsNot Nothing Then...
实施不当的“尝试/捕获”(Try / Catch)可能会隐藏问题所在并导致新问题:
Dim dr As SqlDataReader
Try
Dim lnk As LinkButton = TryCast(sender, LinkButton)
Dim gr As GridViewRow = DirectCast(lnk.NamingContainer, GridViewRow)
Dim eid As String = GridView1.DataKeys(gr.RowIndex).Value.ToString()
ViewState("username") = eid
sqlQry = "select FirstName, Surname, DepartmentName, ExtensionName, jobTitle,
Pager, mailaddress, from employees1 where username='" & eid & "'"
If connection.State <> ConnectionState.Open Then
connection.Open()
End If
command = New SqlCommand(sqlQry, connection)
'More code fooing and barring
dr = command.ExecuteReader()
If dr.Read() Then
lblFirstName.Text = Convert.ToString(dr("FirstName"))
...
End If
mpe.Show()
Catch
Finally
command.Dispose()
dr.Close() ' <-- NRE
connection.Close()
End Try
这是一种未按预期创建对象的情况,但也证明了empty的反作用Catch
。
SQL中有一个多余的逗号(在“ mailaddress”之后),导致处出现异常.ExecuteReader
。在Catch
不Finally
执行任何操作之后,尝试执行清理,但是由于您不能Close
为空DataReader
对象,因此将产生全新的NullReferenceException
结果。
一个空Catch
块是魔鬼的游乐场。这个OP困惑于他为什么要获得NRE的困惑Finally
。在其他情况下,清空Catch
可能会导致更进一步的下游混乱,并导致您花费时间在错误的位置上寻找错误的地方以解决问题。(上述“静默异常”提供相同的娱乐价值。)
补救
不要使用空的Try / Catch块-让代码崩溃,以便您可以a)查明原因b)查明位置并c)采取适当的补救措施。Try / Catch块无意于向唯一有资格修复它们的人员(开发人员)隐藏异常。
For Each row As DataGridViewRow In dgvPlanning.Rows
If Not IsDBNull(row.Cells(0).Value) Then
...
该IsDBNull
函数用于测试值是否等于System.DBNull
: 来自MSDN:
System.DBNull值指示该Object表示缺少或不存在的数据。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
with chkName
(或中存在GroupBox
),则chk
它将为Nothing,并且尝试引用任何属性都将导致异常。
补救
If (chk IsNot Nothing) AndAlso (chk.Checked) Then ...
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
具有has AutoGenerateColumns = True
,它将创建列,但不会命名它们,因此当按名称引用它们时,上面的代码将失败。
补救
手动命名列,或通过索引引用:
dgvBooks.Columns(0).Visible = True
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
when IsNewRow
为true时。
在某些情况下,试图使用从项目My.Settings
这是一种StringCollection
可导致NullReference你使用它的第一次。解决方案是相同的,但不是很明显。考虑:
My.Settings.FooBars.Add("ziggy") ' foobars is a string collection
由于VB正在为您管理“设置”,因此可以预期它会初始化集合。它将,但前提是您之前已在集合中添加了初始条目(在“设置”编辑器中)。由于添加时(显然)初始化了集合,因此Nothing
当“设置”编辑器中没有要添加的项目时,集合将保留。
补救
Load
如有必要,在表单的事件处理程序中初始化设置集合:
If My.Settings.FooBars Is Nothing Then
My.Settings.FooBars = New System.Collections.Specialized.StringCollection
End If
通常,Settings
仅在应用程序第一次运行时才需要初始化集合。另一种解决方法是在“ 项目”->“设置” |“设置”中为您的集合添加初始值。FooBars,保存项目,然后删除假值。
您可能忘记了New
操作员。
要么
您认为可以完美执行的操作可以将已初始化的对象返回到您的代码中,而事实并非如此。
永远不要忽略编译器警告,永远不要忽略使用Option Strict On
。
另一种情况是将空对象转换为值类型时。例如,下面的代码:
object o = null;
DateTime d = (DateTime)o;
它将抛出一个NullReferenceException
演员表。在上面的示例中,这似乎很明显,但是这可能发生在更多的“后期绑定”复杂场景中,其中空对象是从您不拥有的某些代码中返回的,而强制转换例如是由某些自动系统生成的。
一个示例就是带有Calendar控件的简单ASP.NET绑定片段:
<asp:Calendar runat="server" SelectedDate="<%#Bind("Something")%>" />
在这里,SelectedDate
其实是一个性质-的DateTime
类型-的的Calendar
Web控件类型,结合可完全恢复一些空。隐式ASP.NET生成器将创建一段与上面的强制转换代码等效的代码。这将引起一个NullReferenceException
很难发现的问题,因为它位于ASP.NET生成的代码中,可以很好地进行编译...
DateTime x = (DateTime) o as DateTime? ?? defaultValue;
这意味着所讨论的变量没有指向任何对象。我可以这样生成:
SqlConnection connection = null;
connection.Open();
这将引发错误,因为在声明变量“ connection
”时,它没有指向任何内容。当我尝试调用成员“ Open
”时,没有可解决的引用,它将引发错误。
为避免此错误:
object == null
。JetBrains的Resharper工具将识别代码中可能存在空引用错误的每个位置,从而使您可以进行空检查。此错误是错误的第一来源,恕我直言。
这意味着您的代码使用了设置为null的对象引用变量(即,它没有引用实际的对象实例)。
为避免该错误,应在使用可能为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
}
抛出此异常的一个示例是:当您尝试检查某些内容时,该值为null。
例如:
string testString = null; //Because it doesn't have a value (i.e. it's null; "Length" cannot do what it needs to do)
if (testString.Length == 0) // Throws a nullreferenceexception
{
//Do something
}
当您尝试对尚未实例化的内容(即上面的代码)执行操作时,.NET运行时将抛出NullReferenceException。
与ArgumentNullException相比,如果方法期望传递给它的内容不为null,则通常将其作为防御措施。
有关更多信息,请参见C#NullReferenceException和Null参数。
C#8.0引入了可为空的引用类型和不可为空的引用类型。因此,仅必须检查可为空的引用类型,以避免NullReferenceException。
如果尚未初始化引用类型,并且要设置或读取其属性之一,它将抛出NullReferenceException。
例:
Person p = null;
p.Name = "Harry"; // NullReferenceException occurs here.
您可以通过检查变量是否不为null来简单地避免这种情况:
Person p = null;
if (p!=null)
{
p.Name = "Harry"; // Not going to run to this point
}
要完全理解为什么会引发NullReferenceException,了解值类型和[引用类型] [3] 之间的区别非常重要。
因此,如果要处理值类型,则不会发生NullReferenceExceptions 。尽管在处理引用类型时需要保持警惕!
顾名思义,只有引用类型才能保存引用或从字面上指向任何内容(或“空”)。而值类型始终包含一个值。
引用类型(必须检查这些类型):
值类型(您可以简单地忽略这些类型):
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) {
// ...
}
您正在使用包含空值引用的对象。因此,它给出了一个空异常。在示例中,字符串值为null,并且在检查其长度时,发生了异常。
例:
string value = null;
if (value.Length == 0) // <-- Causes exception
{
Console.WriteLine(value); // <-- Never reached
}
异常错误是:
未处理的异常:
System.NullReferenceException:对象引用未设置为对象的实例。在Program.Main()
虽然什么导致NullReferenceException异常和办法,以避免/修复这样的例外在其他的答案已经得到解决,很多程序员还没有学会的是如何独立调试开发过程中这样的例外。
在Visual Studio中,这通常很容易,这要归功于Visual Studio调试器。
首先,请确保将捕获正确的错误-请参阅 VS2010中的“ System.NullReferenceException”如何允许中断? 注1
然后,从调试开始(F5)或将[VS调试器]附加到正在运行的进程。有时使用可能会有用Debugger.Break
,它将提示启动调试器。
现在,当抛出(或未处理)NullReferenceException时,调试器将在发生异常的行上停止(是否记住上面设置的规则?)。有时,错误很容易发现。
例如,在以下行中,唯一会导致异常的代码是if myString
评估为null。可以通过查看监视窗口或在立即窗口中运行表达式来验证这一点。
var x = myString.Trim();
在更高级的情况下,例如以下情况,您需要使用上面的一种技术(“监视”或“即时Windows”)来检查表达式以确定是否str1
为空或是否str2
为空。
var x = str1.Trim() + str2.Trim();
一旦找到了引发异常的位置,通常不容易进行倒推以找出[错误地]引入了null值的位置-
花点时间了解异常原因。检查空表达式。检查可能导致这种空表达式的先前表达式。添加断点并根据需要逐步执行该程序。使用调试器。
1 如果“抛出时中断”过于激进,并且调试器在.NET或第3方库中的NPE上停止,则可以使用“ 用户未处理时中断”来限制捕获的异常。另外,VS2012引入了“我的代码”,我建议也启用它。
如果在启用“仅我的代码”的情况下进行调试,则行为会稍有不同。启用“仅我的代码”后,调试器将忽略在“我的代码”之外抛出且不会通过“我的代码”的第一时间公共语言运行时(CLR)异常
object o = null;
DateTime d = (DateTime)o; // NullReferenceException
从(或从一个类或,或从接口类型)到值类型(非)的拆箱转换(广播)本身给出。 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<>
实例进行装箱时使用的特殊规则。
添加一种情况,即实体框架中使用的实体的类名与Web表单代码隐藏文件的类名相同。
假设您有一个Web窗体Contact.aspx,其代码背后的类是Contact,并且您有一个实体名称Contact。
然后,当您调用context.SaveChanges()时,以下代码将引发NullReferenceException。
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;}
}
和Contact实体类。有时,实体类是局部类,因此您也可以在其他文件中扩展它们。
public partial class Contact
{
public string Name {get; set;}
}
当实体和codebehind类都在同一名称空间中时,将发生错误。若要解决此问题,请为Contact.aspx重命名实体类或代码隐藏类。
原因 我仍然不确定原因。但是,只要任何实体类将扩展System.Web.UI.Page,就会发生此错误。
可能会收到此异常的另一种常见情况涉及在单元测试期间模拟类。无论使用哪种模拟框架,都必须确保正确模拟了类层次结构的所有适当级别。特别是,HttpContext
必须模拟被测试代码引用的所有属性。
请参见“ 测试自定义AuthorizationAttribute时抛出的NullReferenceException ”,以获取详细说明。
我对此有不同的看法。这种回答“我还能做些什么来避免呢? ”
当跨不同的层工作时(例如在MVC应用程序中),控制器需要服务来调用业务操作。在这种情况下,可以使用依赖项注入容器来初始化服务以避免NullReferenceException。因此,这意味着您无需担心检查null的情况,只需从控制器调用服务,就好像它们将始终作为单例或原型可用(并初始化)一样。
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();
}
}
关于“我该怎么办”,可能会有很多答案。
防止此类错误情况发生的更“正式”方法是在代码中按合同应用设计。这意味着在开发时,您需要在系统上设置类不变式,甚至函数/方法的前提条件和后置条件。
简而言之,类不变式可确保您的类中存在一些在正常使用中不会受到违反的约束(因此,该类不会处于不一致的状态)。前提条件意味着作为函数/方法的输入提供的数据必须遵循一定的约束条件,并且永不违反它们;而后置条件意味着函数/方法的输出必须再次遵循设置的约束条件,而又从未违反它们。合同条件应该永远无bug的程序的执行过程中被侵犯,因此契约式设计在调试模式实践中被选中,同时在发布禁用,最大限度地开发的系统性能。
这样,您可以避免NullReferenceException
由于违反约束集而导致的情况。例如,如果您X
在类中使用对象属性,然后稍后尝试调用其方法之一并X
具有null值,则将导致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 );
//...
}
因此,存在用于.NET应用程序的Code Contracts项目。
或者,可以使用断言来应用按合同设计。
更新:值得一提的是,该术语是由Bertrand Meyer 与其Eiffel编程语言设计相关的。
NullReferenceException
当我们尝试访问空对象的属性或字符串值变为空并且我们尝试访问字符串方法时,将引发A。
例如:
当访问空字符串的字符串方法时:
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
String.Empty.ToLower()
不会引发空引用异常。它代表一个实际的字符串,尽管是一个空字符串(即""
)。由于有一个要调用的对象ToLower()
,因此在此处抛出空引用异常是没有意义的。
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
调试显示MyOtherView中的模型为Null。直到我将其更改为:
@{
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误读了“&”号和括号。
Html.Partial
,没有@Html.Partial
Null
),所以我知道错误是与我是如何在发送模式。
你能为这个做什么?
这里有很多很好的答案,解释什么是空引用以及如何调试它。但是,如何预防或至少更容易发现问题却很少。
检查参数
例如,方法可以检查不同的参数以查看它们是否为null并抛出ArgumentNullException
,显然是为此目的而创建的异常。
ArgumentNullException
偶数的构造函数将参数的名称和一条消息作为参数,以便您可以确切地告诉开发者问题出在哪里。
public void DoSomething(MyObject obj) {
if(obj == null)
{
throw new ArgumentNullException("obj", "Need a reference to obj.");
}
}
使用工具
也有几个库可以提供帮助。例如,“ Resharper”可以在编写代码时为您提供警告,尤其是在使用其属性的情况下:NotNullAttribute
在“ Microsoft Code Contracts”中,您可以使用类似语法的语法Contract.Requires(obj != null)
,它可以为您提供运行时和编译检查:Code Contracts简介。
还有“ PostSharp”,将允许您仅使用如下属性:
public void DoSometing([NotNull] obj)
这样,obj
将在运行时检查PostSharp是否为构建过程的一部分。请参阅: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
相关语言功能
C#6.0引入了“空条件运算符”,它对此有所帮助。使用此功能,您可以引用嵌套对象,如果其中任何一个嵌套null
整个表达式,则返回null
。
这样可以减少在某些情况下必须执行的空检查次数。语法是在每个点之前打一个问号。以以下代码为例:
var address = country?.State?.County?.City;
想象一下,这country
是一个Country
具有称为State
等属性的类型的对象。如果country
,State
,County
,或者City
是null
那么address will be
空. Therefore you only have to check whether
地址is
null`。
这是一个很棒的功能,但是它给您的信息较少。并没有使4中的哪个为空并不明显。
像Nullable一样内置?
C#有一个很好的速记方式Nullable<T>
,您可以通过在这样的类型后面加上一个问号使某些内容为空int?
。
如果C#具有NotNull<T>
上面的struct之类的东西并且具有类似的速记(也许是感叹号(!)),那么您会写成类似以下内容将是很好的public void WriteName(Person! person)
。
有趣的是,此页面上的答案均未提及这两种情况,如果我添加它们,希望没人介意:
.NET中的通用词典不是线程安全的,因此当您尝试从两个并发线程访问键时,它们有时可能会抛出a NullReference
甚至(更常见)的错误KeyNotFoundException
。在这种情况下,例外情况非常容易引起误解。
如果代码NullReferenceException
抛出a,则unsafe
可能会查看指针变量,并检查它们的内容IntPtr.Zero
。这是同一件事(“空指针异常”),但是在不安全的代码中,变量通常被强制转换为值类型/数组等,然后您将头撞墙,想知道值类型如何抛出该值例外。
(顺便说一句,除非您需要,否则不使用不安全代码的另一个原因)
null
哪种方法不同?
您可以使用c#6中的Null条件运算符以干净的方式修复NullReferenceException,并编写较少的代码来处理null检查。
在执行成员访问(?。)或索引(?[)操作之前,它用于测试null。
例
var name = p?.Spouse?.FirstName;
等效于:
if (p != null)
{
if (p.Spouse != null)
{
name = p.Spouse.FirstName;
}
}
结果是,当p为null或p.Spouse为null时,名称将为null。
否则,将为变量名分配p.Spouse.FirstName的值。
有关更多详细信息:空条件运算符
错误行“对象引用未设置为对象的实例。”表明您尚未将实例对象分配给对象引用,而您仍在访问该对象的属性/方法。
例如:假设您有一个名为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;
}
}
当未实例化要使用的类的对象时,会发生NullReferenceException或未设置为对象实例的对象引用。例如:
假设您有一个名为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;
}
}
如上面的代码所示,语句 Student s-仅声明了Student类型的变量,请注意,此时尚未实例化Student类。因此,当执行s.GetFullName()语句时,它将抛出NullReferenceException。
好吧,简单来说:
您正在尝试访问未创建或当前不在内存中的对象。
那么如何解决这个问题:
调试并让调试器中断...它将直接带您到已损坏的变量...现在您的任务是简单地解决此问题。. 在适当的位置使用new关键字。
如果它是由于不存在该对象而在某些数据库命令上引起的,那么您所需要做的就是进行空检查并处理它:
if (i == null) {
// Handle this
}
最难的一个..如果GC已经收集了对象...如果您尝试使用字符串查找对象,这通常会发生...也就是说,按对象名称查找它,则可能发生GC可能已经发生的情况。清理它...这很难找到,并且将成为一个大问题。解决此问题的更好方法是在开发过程中的必要位置进行空检查。这样可以节省您很多时间。
通过名称查找,我的意思是某些框架允许您使用字符串来查找对象,并且代码可能如下所示:FindObject(“ ObjectName”);
从字面上看,修复NullReferenceExeption的最简单方法有两种。例如,如果您有一个GameObject,它带有附加的脚本和一个名为rb(rigidbody)的变量,则在您开始游戏时,该变量将开始为null。
这就是为什么您获得NullReferenceExeption的原因,因为计算机没有在该变量中存储数据。
我将使用RigidBody变量作为示例。
实际上,我们可以通过以下几种方式轻松地添加数据:
rb = GetComponent<Rigidbody>();
Start()
或Awake()
函数下效果最佳。 rb = AddComponent<RigidBody>();
附加说明:如果要统一向对象添加组件,并且可能忘记添加一个组件,则可以[RequireComponent(typeof(RigidBody))]
在类声明上方(所有用法下方的空格)键入内容。
享受游戏乐趣!
如果我们考虑可能引发此异常的常见情况,请在顶部使用对象访问属性。
例如:
string postalcode=Customer.Address.PostalCode;
//if customer or address is null , this will through exeption
在这里,如果address为null,那么您将获得NullReferenceException。
因此,作为一种做法,在访问此类对象的属性(特别是通用属性)之前,我们应始终使用null检查
string postalcode=Customer?.Address?.PostalCode;
//if customer or address is null , this will return null, without through a exception
这基本上是一个Null引用异常。作为微软明─
当您尝试访问其值为null的类型的成员时,将引发NullReferenceException异常。
这意味着,如果任何一个成员没有任何价值,而我们正在使该成员执行某些任务,那么系统无疑会抛出一条消息并说:
“嘿,等等,该成员没有值,因此它无法执行您要移交给它的任务。”
异常本身表明正在引用某些内容,但未设置其值。因此,这表明它仅在使用引用类型时发生,因为值类型不可为空。
如果我们使用值类型成员,则不会发生NullReferenceException。
class Program
{
static void Main(string[] args)
{
string str = null;
Console.WriteLine(str.Length);
Console.ReadLine();
}
}
上面的代码显示了一个简单的字符串,该字符串分配了一个空值。
现在,当我尝试打印字符串str的长度时,我得到了一条'System.NullReferenceException'类型的未处理的异常消息,因为成员str指向null并且不能有任何长度的null。
当我们忘记实例化引用类型时,也会发生' NullReferenceException '。
假设我有一个类和成员方法。我没有实例化我的班级,而只是命名了我的班级。现在,如果我尝试使用该方法,编译器将抛出错误或发出警告(取决于编译器)。
class Program
{
static void Main(string[] args)
{
MyClass1 obj;
obj.foo(); //Use of unassigned local variable 'obj'
}
}
public class MyClass1
{
internal void foo()
{
Console.WriteLine("hello from foo");
}
}
上面代码的编译器会引发一个错误,即未分配变量obj,这表明我们的变量具有空值或什么都没有。上面代码的编译器会引发一个错误,即未分配变量obj,这表明我们的变量具有空值或什么都没有。
NullReferenceException的出现是由于我们没有检查对象的值而导致的错误。我们经常在代码开发中不检查对象值。
当我们忘记实例化对象时,也会出现这种情况。使用可以返回或设置空值的方法,属性,集合等也可能是导致此异常的原因。
有多种方法和方法可以避免这种著名的异常:
显式检查:我们应遵循检查对象,属性,方法,数组和集合是否为空的传统。可以使用if-else if-else等条件语句简单地实现。
异常处理:管理此异常的重要方法之一。使用简单的try-catch-finally块,我们可以控制此异常并对其进行维护。当您的应用程序处于生产阶段时,这可能非常有用。
空运算符:在将值设置为对象,变量,属性和字段时,也可以方便地使用空合并运算符和空条件运算符。
调试器:对于开发人员而言,我们拥有进行调试的强大武器。如果在开发过程中遇到NullReferenceException,则可以使用调试器来获取异常源。
内置方法:GetValueOrDefault(),IsNullOrWhiteSpace()和IsNullorEmpty()之类的系统方法检查是否为空,如果有空值,则分配默认值。
这里已经有很多好的答案。您还可以在我的博客上查看带有示例的更详细的描述。
希望这也能有所帮助!
如果在保存或编译内部版本时收到此消息,请关闭所有文件,然后打开任何文件进行编译和保存。
对我来说,原因是我已重命名该文件,而旧文件仍处于打开状态。