C# Using Activator.CreateInstance

后端 未结 6 812
旧巷少年郎
旧巷少年郎 2020-12-04 05:42

I asked a question yesterday regarding using either reflection or Strategy Pattern for dynamically calling methods.

However, since then I have decided to change the

相关标签:
6条回答
  • 2020-12-04 05:58

    Essentially, it sounds like you want the factory pattern. In this situation, you define a mapping of input to output types and then instantiate the type at runtime like you are doing.

    Example:

    You have X number of classes, and they all share a common interface of IDoSomething.

    public interface IDoSomething
    {
         void DoSomething();
    }
    
    public class Foo : IDoSomething
    {
        public void DoSomething()
        {
             // Does Something specific to Foo
        }
    }
    
    public class Bar : IDoSomething
    {
        public void DoSomething()
        {
            // Does something specific to Bar
        }
    }
    
    public class MyClassFactory
    {
         private static Dictionary<string, Type> _mapping = new Dictionary<string, Type>();
    
         static MyClassFactory()
         {
              _mapping.Add("Foo", typeof(Foo));
              _mapping.Add("Bar", typeof(Bar));
         }
    
         public static void AddMapping(string query, Type concreteType)
         {
              // Omitting key checking code, etc. Basically, you can register new types at runtime as well.
              _mapping.Add(query, concreteType);
         }
    
         public IDoSomething GetMySomething(string desiredThing)
         {
              if(!_mapping.ContainsKey(desiredThing))
                  throw new ApplicationException("No mapping is defined for: " + desiredThing);
    
              return Activator.CreateInstance(_mapping[desiredThing]) as IDoSomething;
         }
    }
    
    0 讨论(0)
  • 2020-12-04 06:02

    Just as an example how to add initialization in the constructor:

    Something similar to:

    Activator.CreateInstance(Type.GetType("ConsoleApplication1.Operation1"), initializationData);
    but written with Linq Expression, part of code is taken here:

    public class Operation1 
    {
        public Operation1(object data)
        { 
        }
    }
    
    public class Operation2 
    {
        public Operation2(object data)
        {
        }
    }
    
    public class ActivatorsStorage
    {
        public delegate object ObjectActivator(params object[] args);
    
        private readonly Dictionary<string, ObjectActivator> activators = new Dictionary<string,ObjectActivator>();
    
        private ObjectActivator CreateActivator(ConstructorInfo ctor)
        {
            Type type = ctor.DeclaringType;
    
            ParameterInfo[] paramsInfo = ctor.GetParameters();
            ParameterExpression param = Expression.Parameter(typeof(object[]), "args");
    
            Expression[] argsExp = new Expression[paramsInfo.Length];
    
            for (int i = 0; i < paramsInfo.Length; i++)
            {
                Expression index = Expression.Constant(i);
                Type paramType = paramsInfo[i].ParameterType;
    
                Expression paramAccessorExp = Expression.ArrayIndex(param, index);
    
                Expression paramCastExp = Expression.Convert(paramAccessorExp, paramType);
    
                argsExp[i] = paramCastExp;
            }
    
            NewExpression newExp = Expression.New(ctor, argsExp);
    
            LambdaExpression lambda = Expression.Lambda(typeof(ObjectActivator), newExp, param);
    
            return (ObjectActivator)lambda.Compile();
        }
    
        private ObjectActivator CreateActivator(string className)
        {
            Type type = Type.GetType(className);
            if (type == null)
                throw new ArgumentException("Incorrect class name", "className");
    
            // Get contructor with one parameter
            ConstructorInfo ctor = type.GetConstructors()
                .SingleOrDefault(w => w.GetParameters().Length == 1 
                    && w.GetParameters()[0].ParameterType == typeof(object));
    
            if (ctor == null)
                    throw new Exception("There is no any constructor with 1 object parameter.");
    
            return CreateActivator(ctor);
        }
    
        public ObjectActivator GetActivator(string className)
        {
            ObjectActivator activator;
    
            if (activators.TryGetValue(className, out activator))
            {
                return activator;
            }
            activator = CreateActivator(className);
            activators[className] = activator;
            return activator;
        }
    }

    The usage is following:

    ActivatorsStorage ast = new ActivatorsStorage();
    var a = ast.GetActivator("ConsoleApplication1.Operation1")(initializationData);
    var b = ast.GetActivator("ConsoleApplication1.Operation2")(initializationData);

    The same can be implemented with DynamicMethods.

    Also, the classes are not required to be inherited from the same interface or base class.

    Thanks, Vitaliy

    0 讨论(0)
  • 2020-12-04 06:05

    One strategy that I use in cases like this is to flag my various implementations with a special attribute to indicate its key, and scan the active assemblies for types with that key:

    [AttributeUsage(AttributeTargets.Class)]
    public class OperationAttribute : System.Attribute
    { 
        public OperationAttribute(string opKey)
        {
            _opKey = opKey;
        }
    
        private string _opKey;
        public string OpKey {get {return _opKey;}}
    }
    
    [Operation("Standard deviation")]
    public class StandardDeviation : IOperation
    {
        public void Initialize(object originalData)
        {
            //...
        }
    }
    
    public interface IOperation
    {
        void Initialize(object originalData);
    }
    
    public class OperationFactory
    {
        static OperationFactory()
        {
            _opTypesByKey = 
                (from a in AppDomain.CurrentDomain.GetAssemblies()
                 from t in a.GetTypes()
                 let att = t.GetCustomAttributes(typeof(OperationAttribute), false).FirstOrDefault()
                 where att != null
                 select new { ((OperationAttribute)att).OpKey, t})
                 .ToDictionary(e => e.OpKey, e => e.t);
        }
        private static IDictionary<string, Type> _opTypesByKey;
        public IOperation GetOperation(string opKey, object originalData)
        {
            var op = (IOperation)Activator.CreateInstance(_opTypesByKey[opKey]);
            op.Initialize(originalData);
            return op;
        }
    }
    

    That way, just by creating a new class with a new key string, you can automatically "plug in" to the factory, without having to modify the factory code at all.

    You'll also notice that rather than depending on each implementation to provide a specific constructor, I've created an Initialize method on the interface I expect the classes to implement. As long as they implement the interface, I'll be able to send the "originalData" to them without any reflection weirdness.

    I'd also suggest using a dependency injection framework like Ninject instead of using Activator.CreateInstance. That way, your operation implementations can use constructor injection for their various dependencies.

    0 讨论(0)
  • 2020-12-04 06:08
    1. There's no error checking here. Are you absolutely sure that _class will resolve to a valid class? Are you controlling all the possible values or does this string somehow get populated by an end-user?
    2. Reflection is generally most costly than avoiding it. Performance issues are proportionate to the number of objects you plan to instantiate this way.
    3. Before you run off and use a dependency injection framework read the criticisms of it. =)
    0 讨论(0)
  • 2020-12-04 06:11

    When using reflection you should ask yourself a couple of questions first, because you may end up in an over-the-top complex solution that's hard to maintain:

    1. Is there a way to solve the problem using genericity or class/interface inheritance?
    2. Can I solve the problem using dynamic invocations (only .NET 4.0 and above)?
    3. Is performance important, i.e. will my reflected method or instantiation call be called once, twice or a million times?
    4. Can I combine technologies to get to a smart but workable/understandable solution?
    5. Am I ok with losing compile time type safety?

    Genericity / dynamic

    From your description I assume you do not know the types at compile time, you only know they share the interface ICalculation. If this is correct, then number (1) and (2) above are likely not possible in your scenario.

    Performance

    This is an important question to ask. The overhead of using reflection can impede a more than 400-fold penalty: that slows down even a moderate amount of calls.

    The resolution is relatively easy: instead of using Activator.CreateInstance, use a factory method (you already have that), look up the MethodInfo create a delegate, cache it and use the delegate from then on. This yields only a penalty on the first invocation, subsequent invocations have near-native performance.

    Combine technologies

    A lot is possible here, but I'd really need to know more of your situation to assist in this direction. Often, I end up combining dynamic with generics, with cached reflection. When using information hiding (as is normal in OOP), you may end up with a fast, stable and still well-extensible solution.

    Losing compile time type safety

    Of the five questions, this is perhaps the most important one to worry about. It is very important to create your own exceptions that give clear information about reflection mistakes. That means: every call to a method, constructor or property based on an input string or otherwise unchecked information must be wrapped in a try/catch. Catch only specific exceptions (as always, I mean: never catch Exception itself).

    Focus on TargetException (method does not exist), TargetInvocationException (method exists, but rose an exc. when invoked), TargetParameterCountException, MethodAccessException (not the right privileges, happens a lot in ASP.NET), InvalidOperationException (happens with generic types). You don't always need to try to catch all of them, it depends on the expected input and expected target objects.

    To sum it up

    Get rid of your Activator.CreateInstance and use MethodInfo to find the factory-create method, and use Delegate.CreateDelegate to create and cache the delegate. Simply store it in a static Dictionary where the key is equal to the class-string in your example code. Below is a quick but not-so-dirty way of doing this safely and without losing too much type safety.

    Sample code

    public class TestDynamicFactory
    {
        // static storage
        private static Dictionary<string, Func<ICalculate>> InstanceCreateCache = new Dictionary<string, Func<ICalculate>>();
    
        // how to invoke it
        static int Main()
        {
            // invoke it, this is lightning fast and the first-time cache will be arranged
            // also, no need to give the full method anymore, just the classname, as we
            // use an interface for the rest. Almost full type safety!
            ICalculate instanceOfCalculator = this.CreateCachableICalculate("RandomNumber");
            int result = instanceOfCalculator.ExecuteCalculation();
        }
    
        // searches for the class, initiates it (calls factory method) and returns the instance
        // TODO: add a lot of error handling!
        ICalculate CreateCachableICalculate(string className)
        {
            if(!InstanceCreateCache.ContainsKey(className))
            {
                // get the type (several ways exist, this is an eays one)
                Type type = TypeDelegator.GetType("TestDynamicFactory." + className);
    
                // NOTE: this can be tempting, but do NOT use the following, because you cannot 
                // create a delegate from a ctor and will loose many performance benefits
                //ConstructorInfo constructorInfo = type.GetConstructor(Type.EmptyTypes);
    
                // works with public instance/static methods
                MethodInfo mi = type.GetMethod("Create");
    
                // the "magic", turn it into a delegate
                var createInstanceDelegate = (Func<ICalculate>) Delegate.CreateDelegate(typeof (Func<ICalculate>), mi);
    
                // store for future reference
                InstanceCreateCache.Add(className, createInstanceDelegate);
            }
    
            return InstanceCreateCache[className].Invoke();
    
        }
    }
    
    // example of your ICalculate interface
    public interface ICalculate
    {
        void Initialize();
        int ExecuteCalculation();
    }
    
    // example of an ICalculate class
    public class RandomNumber : ICalculate
    {
        private static Random  _random;
    
        public static RandomNumber Create()
        {
            var random = new RandomNumber();
            random.Initialize();
            return random;
        }
    
        public void Initialize()
        {
            _random = new Random(DateTime.Now.Millisecond);
        }
    
        public int ExecuteCalculation()
        {
            return _random.Next();
        }
    }
    
    0 讨论(0)
  • 2020-12-04 06:16

    I suggest you give your factory implementation a method RegisterImplementation. So every new class is just a call to that method and you are not changing your factories code.

    UPDATE:
    What I mean is something like this:

    Create an interface that defines a calculation. According to your code, you already did this. For the sake of being complete, I am going to use the following interface in the rest of my answer:

    public interface ICalculation
    {
        void Initialize(string originalData);
        void DoWork();
    }
    

    Your factory will look something like this:

    public class CalculationFactory
    {
        private readonly Dictionary<string, Func<string, ICalculation>> _calculations = 
                            new Dictionary<string, Func<string, ICalculation>>();
    
        public void RegisterCalculation<T>(string method)
            where T : ICalculation, new()
        {
            _calculations.Add(method, originalData =>
                                      {
                                          var calculation = new T();
                                          calculation.Initialize(originalData);
                                          return calculation;
                                      });
        }
    
        public ICalculation CreateInstance(string method, string originalData)
        {
            return _calculations[method](originalData);
        }
    }
    

    This simple factory class is lacking error checking for the reason of simplicity.

    UPDATE 2:
    You would initialize it like this somewhere in your applications initialization routine:

    CalculationFactory _factory = new CalculationFactory();
    
    public void RegisterCalculations()
    {
        _factory.RegisterCalculation<Pivot>("Pivot");
        _factory.RegisterCalculation<GroupBy>("GroupBy");
        _factory.RegisterCalculation<StandardDeviation>("Standard deviation");
        _factory.RegisterCalculation<PhosphoPRASPercentage>("% phospho PRAS Protein");
        _factory.RegisterCalculation<AveragePPPperTreatment>("AveragePPPperTreatment");
        _factory.RegisterCalculation<AvgPPPNControl>("AvgPPPNControl");
        _factory.RegisterCalculation<PercentageInhibition>("PercentageInhibition");
    }
    
    0 讨论(0)
提交回复
热议问题