C++ return type overload hack

后端 未结 7 676
无人及你
无人及你 2021-02-04 09:02

I was bored and came up with such hack (pseudocode):

 1 struct proxy {
 2     operator int(); // int function
 3     operator double(); // double function
 4             


        
7条回答
  •  故里飘歌
    2021-02-04 10:03

    The call of the function "function" became kind of context sensitive. I suppose, this trick can be exploited to support subject-oriented programming.

    Subject-oriented programming is based on the observation that properties of an object are not inherent to the object itself, but depend on who perceive that object. For example, from the point of view of human, tree is not food, but from the point of view of termite, tree is food. Object-oriented paradigm doesn't support this observation directly, and people often come to complex unnatural designs, because they try to incorporate all different subjective views of an object into one entity ("class"), following thoughtlessly OOP guidelines.

    So, let's try to state subjective perceptions explicitly, using the trick in question to get context sensitivity.

    template
    class FoodFrom {};
    //forward declarations
    class Tree;
    class Termite;
    class Human;
    
    //property "food" of a tree
    template<>
    class FoodFrom
    {
    public:
        FoodFrom(Tree& _tree): tree(_tree) {}
    
        //termite perception of tree as food
        operator FoodFor()
        {
            int happiness_increase = 5;
            tree.mass -= 10;
            return FoodFor(happiness_increase);
        }
        //human perception of tree as food
        operator FoodFor()
        {
            int happiness_increase = 0;
            return FoodFor(happiness_increase);
        }
    private:
        Tree& tree;
    };
    //property "food" of a termite
    template<>
    class FoodFrom
    {
    public:
        FoodFrom(Termite& _termite): termite(_termite) {}
        //human perception of termite as food
        operator FoodFor()
        {
            int happiness_increase = -100;
            //apparently, the termite ought to be terminated due to such a violent act
            termite.~Termite();
            return FoodFor(happiness_increase);
        }
    private:
        Termite& termite;
    };
    
    //simple class FoodFor, just for demonstration purposes
    class FoodBase
    {
    public:
        FoodBase(int _value) : value(_value) {}
        int value;
    };
    template
    class FoodFor: public FoodBase
    {
    public:
        FoodFor(): FoodBase(0) {}
        FoodFor(int _value) : FoodBase(_value) {}
    };
    
    class AliveBeing
    {
    public:
        AliveBeing(): happiness(100) {}
        bool is_happy()
        {
            return happiness > 0;
        }
        void eat()
        {
            happiness += getMeal()->value;
        }
    private:
        int happiness;
        virtual FoodBase* getMeal() = 0;
    };
    class Tree: public AliveBeing
    {
    public:
        FoodFrom getFood(); //see definition below
        float mass;
        //...
    private:
        //we don't call getMeal for a tree in this demo
        virtual FoodBase* getMeal() { return NULL; }
    };
    
    class Termite: public AliveBeing
    {
    public:
        FoodFrom getFood(); //see definition below
        FoodFor meal;
    private:
        virtual FoodBase* getMeal() { return &meal; }
    };
    
    class Human: public AliveBeing
    {
    public:
        FoodFor meal;
    private:
        virtual FoodBase* getMeal() { return &meal; }
    };
    
    //return proxy "FoodFrom" to "overload" return type
    FoodFrom Tree::getFood()
    { return FoodFrom(*this); }
    FoodFrom Termite::getFood()
    { return FoodFrom(*this); }
    
    //usage
        Tree tree;
        Termite funny_bug;
        //funny_bug gets its perceived value of eating tree
        funny_bug.meal = tree.getFood();
        funny_bug.eat();
        if(funny_bug.is_happy())
            funny_bug.goFindThirdPlace();
    
        //...
    
        Human joel;
        //joel get its perceived value of eating tree
        joel.meal = tree.getFood();
        joel.eat();
    
        //...
    
        if(joel.see(funny_bug))
        {
            joel.meal = funny_bug.getFood();
            joel.eat();
        }
        if(joel.is_happy())
            joel.writeAnotherGreatArticle();
    

    Note that the tree doesn't know what eats it.

    (great question indeed, made me reflect upon it much)

提交回复
热议问题