I have been trying to serialize an array across an AppDomain
boundary, using the following code:
public int Read(byte[] buffer, int offset, int count)
{
return base.Read(buffer, offset, count);
}
As a guess, after noticing the attributes elsewhere, I marked the method's parameters with [In]
and [Out]
attributes, which seemed to cause the parameters to behave as if they were passed by reference.
For example:
public int Read([In, Out] byte[] buffer, int offset, int count)
{
return base.Read(buffer, offset, count);
}
Before I added the attributes, the contents of the buffer
variable were lost after returning from the method across an AppDomain
boundary.
The class (SslStream
) was inheriting from MarshalByRefObject
but not marked with the Serializable
attribute. Is this the only way to make a parameter pass-by-value? Are these attributes being recognised somehow by .NET when the class is being serialised? And do they truly cause the parameter to be passed by reference, or are the contents just copied?
In C#, attributes are classes that inherit from the Attribute base class. Any class that inherits from Attribute can be used as a sort of "tag" on other pieces of code. For instance, there is an attribute called ObsoleteAttribute . This is used to signal that code is obsolete and shouldn't be used anymore.
Variables passed as out arguments do not have to be initialized before being passed in a method call. However, the called method is required to assign a value before the method returns.
Attributes are used in C# to convey declarative information or metadata about various code elements such as methods, assemblies, properties, types, etc. Attributes are added to the code by using a declarative tag that is placed using square brackets ([ ]) on top of the required code element.
To note, there has been additional documentation added since the original accepted answer was posted.
The definition of the directional attributes is:
So the key uses of explicitly modifying these are:
The latter of these is particularly useful for reference types which pass by value by default, such as arrays of value types - e.g. byte[].
This is a remarkably poorly documented feature of .NET Remoting. It doesn't have anything to do with whether your class is [Serializable] or derived from MarshalByRefObject. At issue here is how the argument is marshaled across the AppDomain boundary. The call itself is made under the hood by Remoting. Arrays do not automatically get marshaled back after the call, clearly a performance optimization. Only the [Out] attribute is required, [In] is implied. I could not find any relevant documentation about this in MSDN, just a blog post from somebody that ran into the same issue (scroll down to "Using OutAttribute in Remoting").
Some code to play with:
using System;
using System.Runtime.InteropServices;
class Program {
static void Main(string[] args) {
var ad = AppDomain.CreateDomain("second");
var t = (Test)ad.CreateInstanceAndUnwrap(typeof(Test).Assembly.FullName, typeof(Test).FullName);
var b = new byte[] { 1 };
t.Read(b);
System.Diagnostics.Debug.Assert(b[0] == 2);
}
}
class Test : MarshalByRefObject {
public void Read([Out]byte[] arg) {
arg[0] *= 2;
}
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With