Loading...
djc_bg2015 avatar djc_bg2015 923 Точки

Създаване на обект с reflection - Въпрос

Здравейте,

някой може ли да предложи решение на следния проблем: 

Source code: https://gist.github.com/vdonchev/b55abe9a97f89b8c1dd5 (събрал съм всички класове в един файл за удобство)

Имам клас който създава обекти от даден тип чрез рефлекшън. Обектите които могат да се създадат имат конструктор, който приема един параметър (optional). Когато обаче създам обекта без да подавам аргумент (в опит да изпозлвам конструктура) ми се хвърля ексепшън:

System.MissingMethodException: No parameterless constructor defined for this object.

Излиза че немога да си позлвам конструктура , макар че погледнато реално той също може да конструира обекта без да му се подаде нито един аргумент?

 

Надявам се да съм обяснил ясно какъв е проблема, и ако някой може да предложи решение ще се радвам да го чуя.

Поздрави!

0
C# OOP Basics
RoYaL avatar RoYaL Trainer 6849 Точки

Ако искаш да извикаш конструктура с дефинираната дефолтна стойност: Използвай GetConstructor, и на него използвай GetParameters за да вземеш параметрите които приема конструктора и респективно тяхното default value. После подай това default value на CreateInstance.

class Person
{
    public Person(string name = "Pesho")
    {
        Console.WriteLine("hi, my name is " + name);
    }
}

В тази променлива симулирам твоето намиране на типа през асемблито, тъй като ми е клас в същия файл

Type type = Type.GetType("Person");

Събирам в един масив метаинформация за аргументите, които приема единствения конструктор в случая, за това го взимам от колекцията с конструктори с FirstOrDefault():

ParameterInfo[] arguments = type.GetConstructors().FirstOrDefault().GetParameters();

Activator.CreateInstance() очаква като втори аргумент освен типа и object масив от стойностите на параметрите, в същия ред, в който се даклариране в конструктора. Т.е. ако имаш string name, int age, то в object масива на нулева позиция трябва да е името, на първа - годините. „

Съответно първо правя един празен масив, дълъг толкова, колкото са аргументите на конструктора (колкото дължината на arguments масива)

object[] argumentsToPass = new object[arguments.Length];

След което обикалям arguments масива, и слагам DefaultValue-то на всеки аргумент, в object масива. Важно е в object масива (argumentsToPass) стойностите да са от такъв тип, какъвто ги очаква конструктора. Т.е. при string name, int age не може да вкараш в масива ["pesho", "6"]. Трябва да е ["pesho", 6].

За да кастна динмично всяка променлива към съответния тип, използвам Convert.ChangeType(). Тази функция приема като първи аргумент стойността, а като втори аргумент типа към който да я прекастне. Тъй като в ParameterInfo метакласа се съдържа информация и за това от какъв тип е аргументът, лесно можем да построим този код:

        int index = 0;
        foreach (ParameterInfo argument in arguments)
        {
            argumentsToPass[index] = Convert.ChangeType(
                argument.DefaultValue,
                argument.ParameterType
                );
            index++;
        }

След това вече конструирания и напълнен argumentsToPass масив, просто го подавам на активатора.

Person p = (Person)Activator.CreateInstance(Type.GetType(className), argumentsToPass);

Което директно тригърва конструктура и аутпутът е: hi, my name is Pesho

3
17/12/2015 11:18:05
Filkolev avatar Filkolev 4482 Точки

Друг вариант е да си направиш два конструктора. Така или иначе слагането на опционален параметър не се различава особено от овърлоудването на конструктора, а писането е доста малко ако ги chain-неш.

3
djc_bg2015 avatar djc_bg2015 923 Точки

Благодаря и на двамата ви за отговорите!

 

@RoYaL:

Наистина си е бая писане за да се подаде дефолтната стойност на коснтруктура, но пък работи точно както исках :)

 

@Filkolev:

Нещо неможах да разбера как да направя втори конструктор и по този начин да използвам този с дефолтната стойност?

 

Поздрави!

0
Filkolev avatar Filkolev 4482 Точки

Правиш конструктор, който не приема дефолтната стойност и правиш извикване към първия, като на него му подаваш въпросната константа. Май си забравил материала от първата лекция :)

public Student()
    :this("Name")
{
}

public Student(string name)
{
    this.Name = name;
}

 

1
djc_bg2015 avatar djc_bg2015 923 Точки

Хахах, явно да :P

Аз помислих, че някак си мога да кажа на празния конструктор да викне другия без да му подава дефолтната стойност :)

Мерси много!

0
Можем ли да използваме бисквитки?
Ние използваме бисквитки и подобни технологии, за да предоставим нашите услуги. Можете да се съгласите с всички или част от тях.
Назад
Функционални
Използваме бисквитки и подобни технологии, за да предоставим нашите услуги. Използваме „сесийни“ бисквитки, за да Ви идентифицираме временно. Те се пазят само по време на активната употреба на услугите ни. След излизане от приложението, затваряне на браузъра или мобилното устройство, данните се трият. Използваме бисквитки, за да предоставим опцията „Запомни Ме“, която Ви позволява да използвате нашите услуги без да предоставяте потребителско име и парола. Допълнително е възможно да използваме бисквитки за да съхраняваме различни малки настройки, като избор на езика, позиции на менюта и персонализирано съдържание. Използваме бисквитки и за измерване на маркетинговите ни усилия.
Рекламни
Използваме бисквитки, за да измерваме маркетинг ефективността ни, броене на посещения, както и за проследяването дали дадено електронно писмо е било отворено.