I've created a generic class that needs to instantiate its implementation type, so the implementation type has to have an accessible parameter less constructor. Looks like the new() restriction could do the job but its enforcing the implementation type to have a public constructor when I have it internal (but accessible as both are on the same assembly).
Thanks in advance.
EDIT: The reason to do this is that I have a class X that must be used through a Singleton. The Singleton class is a generic class and I want to make the class X constructor internal to avoid external users accessing the object in a wrong way (calling the constructor).
This is disallowed by the C# language as outlined in section 4.4.3 of the specification Bound and Unbound Types.
If the constraint is the constructor constraint
new()
, the typeA
must not beabstract
and must have a public parameterless constructor. This is satisfied if one of the following is true.
A
is a value type, since all value types have a public default constructorA
is a type parameter having the cosntructor constraintA
is a type parameter having the value type constraintA
is a class that is notabstract
and contains an explicitly declaredpublic
constuctor with no parametersA
is notabstract
and has a default constructor.
A compiler error if any one of these conditions are not met. If you find yourself having types that are public but only with internal constructors, then most likely they should actually be internal types with public constructors.
I would recommend changing your types accessor to internal
and its constructor's to public
, and make it parameterless. Your public
parameterless constructor could then call through to a non-parameterless private
or internal
constructor to do any additional initialization work.
internal class C<T> where : T new()
{
public C() : this(new T()) {
}
private C(T t) {
// Do additional initialization
}
}
Mind you that pattern is limited, but there's nothing stopping you from using a private
method instead.
internal class C<T> where T : new() {
public C() {
T t = new T();
InitializeClass(t);
}
private void InitializeClass(T t) {
throw new NotImplementedException();
}
}
As per your update, here's a small example of a public singleton patter.
public class Singleton<T> where T : new()
{
public static Singleton<T> Current {
get;
private set;
}
internal Singleton() : this(new T()) {
}
private Singleton(T t) {
Current = this;
// Do whatever you need to with T
}
public String Name {
get;
set;
}
}
Usage
// Somewhere in your internal assembly
Singleton<String> singleton = new Singleton<String>();
// In an external assembly
Singleton.Current.Name = "SoMoS";
You don't even need to use constructors in that fashion either, you can just as easily do something simple.
public class Singleton<T> where T : new()
{
public static Singleton<T> Current {
get;
private set;
}
internal Singleton() {
T t = new T();
// Do stuff with T
}
public String Name {
get;
set;
}
}
Generics may not be the way to go if you can't design it to fit your requirements. Generics can only do so much and don't solve every problem. There are things like Factory Pattern, Injection, etc..