12-02-2023 06:46 AM - edited 12-02-2023 06:47 AM
Hi,
Suppose there are two LabVIEW classes:
Suppose that the Class A private data consists of a single string field named "String Field A", and that the Class B private data consists of a single string field named "String Field B".
Suppose that Class A contains a constructor VI named "Constructor A.vi". This constructor VI creates a new Class A object whose "String Field A" is set to "abc".
Suppose that Class B contains a constructor VI named "Constructor B.vi". This constructor VI creates a new Class B object whose "String Field B" is set to "def".
Main question
Secondary questions
I have tried a few variations of using the primitives mentioned above, and have read the following NI Community and LAVA posts, but have not quite managed to get it to work yet:
Side-note
In C#, by default a constructor of a child class automatically calls the constructor of the parent class. This default behaviour seems very useful (it can also be turned-off or avoided when not needed). I'm essentially trying to replicate that behaviour in LabVIEW.
Screenshot 1: Sample C# code
Screenshot 2: The result of running the code above. This shows that the Class A constructor was called implicitly inside the Class B constructor.
Many thanks!
Solved! Go to Solution.
12-02-2023 12:11 PM
All of your suggested options are workable, see attached example
12-02-2023 12:47 PM - edited 12-02-2023 12:59 PM
Hi Petru,
With LabVIEW classes, there is no such thing as a "constructor" as seen with OOP in text-based languages like C#. But you can imitate the constructor concept at the cost of some compromises.
The most simple solution I can think of would be:
Make a dynamic dispatch VI named "Create.vi" in your base class, then go to Properties -> Item Settings, select "Create.vi" and check option "Require overrides of this dynamic dispatch VI to always invoke the Call Parent Class Method node". Then, each time you override the "Create" VI in a subclass, you will have to unconditionally call the parent "Create". Note that you are not forced to override it for each class. For example an "abstract" intermediary class may not require specific initialization.
A drawback of using a dynamic dispatch VI for the "Create.vi" is that you must input a class constant to it, which is not very practical. So you can create a static wrapper of all your Create VIs named for example "Create <ClassName>.vi", which takes nothing as input and simply calls the dynamic "Create" of the same class. You can even make the dynamic "Create" protected to hide it from the exterior of the class hierarchy.
Here is an example project that implements this solution.
PS: What I described is basically what is used by some class creation toolkits such as GOOP or G#, except they also implement the concept of "by reference" objects.
Regards,
Raphaël.
12-02-2023 02:12 PM
Thanks Artem.SPb and Raphaël. The examples you attached are crystal clear and answer my question perfectly, thanks!
Also thanks for your detailed notes, Raphaël. I will create static wrappers of all the public "Create <ClassName>.vi" constructors, and set the "Create" dynamic dispatch methods to protected, such that the latter are abstracted away from the users of the class hierarchy.
Constructors that take only primitive data types (string, DBL, Boolean) as inputs, or that take no inputs at all, and output the "fully" constructed object (constructed by sequentially calling all ancestor constructors from most distant ancestor to immediate parent) was my goal.
Kind regards,
Petru