c#(java)中,参数传递都是传递的参数本身的值,
对于值类型,传递的是值本身。
对于引用类型,定义引用类型变量的时候,一个是在栈中的变量,存储的是一个指针,指向在堆中分配的对象实例的地址,当然,如果对象没有实例化,给null值的时候例外。
传递引用类型变量的时候,传递的也是值,
但它的值是内存地址,地址指定堆中的对象。
所以当我们在方法中改变对象内容的时候,我们外围 的引用类型变量操作的对象也发生了变化,因为他们指向的是同一个。
但是如果我们在操作的方法中对 对象进行了新的实例化,则就会在堆中新生成一个对象,和原来的就是两个不同的对象了,在操作的方法结束,这个对象如果没有特别处理,就没有变量指向它了,它就会被销毁。
示例:
new Thread(() => { try { StringBuilder sb = null; //addline(ref sb); addline(sb); rtb_log.InvokeIfRequired(()=> rtb_log.AppendText(sb.ToString()+"\r\n")); } catch (Exception ex) { rtb_log.InvokeIfRequired(() => rtb_log.AppendText(ex.Message + "\r\n")); } finally { if (conn != null && conn.State == ConnectionState.Open) conn.Close(); } }).Start(); void addline( StringBuilder sb) { if (sb == null) sb = new StringBuilder(); sb.Append("hello world!"); }
调用时会发生 对象空引用的异常。
原因是因为在方法中进行了对象的初始化,但是外围的sb和方法中的sb是两个不同的变量,在方法中分配了对象实例后,外围 的sb指定的还是null.
如果想解除这个异常,有几个方法,第一个是对对象进行有效初始化,不要置为null值,如果这里可以直接new StringBuilder()。不要在方法内部再实例化。第二个是如果对象在外围无法初始化,像接口对象,需要延迟到方法中进行初始化的,可以使用返回值的方式或者是ref参数的方法。
例如:
void addline(ref StringBuilder sbx) { if (sbx == null) sbx = new StringBuilder(); sbx.Append("hello world!"); } 或者 StringBuilder addline2(StringBuilder sbx) { if (sbx == null) sbx = new StringBuilder(); sbx.Append("hello world!"); return sbx; }
时间: 2024-10-05 08:10:13