Reputation: 1795
I have such a structure
void Main()
{
var lol = ActionClass.GetTestTuple<Request, Response, RequestCallInfo>("lol", "user");
lol.Dump();
}
public class ActionClass
{
public static Tuple<TRes, TInfo> GetTestTuple<TReq, TRes, TInfo>(string resultMsg, string userName)
where TReq : BaseRequest, new()
where TRes : BaseResponse, new()
where TInfo : CallInfo<TReq>, new()
{
var response = new TRes { Message = resultMsg };
var eventsInfo = new TInfo();
eventsInfo.Data.UserName = userName;
return new Tuple<TRes, TInfo>(response, eventsInfo);
}
}
public class Request : BaseRequest
{
}
public class Response : BaseResponse
{
}
public class RequestCallInfo : CallInfo<Request>
{
public string Item { get; set; }
}
public class CallInfo<GenericType> : BaseCallInfo where GenericType : BaseRequest, new()
{
public GenericType Data { get; set; }
public CallInfo(GenericType x)
{
Data = x;
}
}
public class BaseCallInfo
{
public string CallItem { get; set; }
}
public class BaseRequest
{
public string UserName { get; set; }
}
public class BaseResponse
{
public string Message { get; set; }
}
and when I execute it, I get UserQuery.CallInfo<UserQuery.Request> does not contain a constructor that takes 0 arguments,
So i tried this way
public static Tuple<TRes, TInfo> GetTestTuple<TReq, TRes, TInfo>(string resultMsg, string userName)
where TReq : BaseRequest, new()
where TRes : BaseResponse, new()
where TInfo : CallInfo<TReq>, new()
{
var response = new TRes { Message = resultMsg };
var eventsInfo = new TInfo(new TReq { UserName = userName });
eventsInfo.Data.UserName = userName;
return new Tuple<TRes, TInfo>(response, eventsInfo);
}
but i'm getting 'TInfo': impossible to provide arguments when instantiating a type
How can I pass an instance of TReq
to TInfo
and keep only the parameter constructor in CallInfo<GenericType>
?
Upvotes: 3
Views: 386
Reputation: 71
You should use Activator.CreateInstance
method.
Working solution:
public class ActionClass
{
public static Tuple<TRes, TInfo> GetTestTuple<TReq, TRes, TInfo>(string resultMsg, string userName)
where TReq : BaseRequest, new()
where TRes : BaseResponse, new()
where TInfo : CallInfo<TReq>
{
var response = new TRes { Message = resultMsg };
var eventsInfo = (TInfo)Activator.CreateInstance(typeof(TInfo), new []{ new TReq() });
eventsInfo.Data.UserName = userName;
return new Tuple<TRes, TInfo>(response, eventsInfo);
}
}
public class BaseCallInfo
{
public string CallItem { get; set; }
}
public class BaseRequest
{
public string UserName { get; set; }
}
public class BaseResponse
{
public string Message { get; set; }
}
public class Request : BaseRequest
{
}
public class Response : BaseResponse
{
}
public class RequestCallInfo : CallInfo<Request>
{
public string Item { get; set; }
public RequestCallInfo(Request x) : base(x)
{
}
}
public class CallInfo<GenericType> : BaseCallInfo where GenericType : BaseRequest, new()
{
public GenericType Data { get; set; }
public CallInfo(GenericType x)
{
Data = x;
}
}
}
Upvotes: 0
Reputation: 19842
The new()
constraint only requires that a generic type have a public parameter-less constructor; there is no way to specify specific construct signatures on a generic type / constraint in C#. Given that, the compiler cannot know what constructors would be available by callers, so you can't instantiate a generic type using constructor parameters.
So you could use reflection here if you would like, but this solution seems simpler based on the code you've provided:
var eventsInfo = new TInfo() { Data = new TReq { UserName = userName } };
Upvotes: 1