Method vs Computed in Vue

后端 未结 7 459
说谎
说谎 2020-11-28 01:00

What is the main difference between a method and a computed value in Vue.js?

They look the same and interchangeable.

相关标签:
7条回答
  • 2020-11-28 01:34

    Computed Properties

    Computed properties are called computed value as well. It means, they update and can be changed anytime. Also, it caches the data until it changes. When the Vue is instantiated, computed properties are converted into a property.

    One more thing I want to share, You cannot pass any parameter in the computed properties that's why while calling any computer property no parenthesis required.

    Methods

    Methods are the same as function and work the same way. Besides, a method does nothing unless you call it. Also, like all javascript functions, it accepts parameters and will be re-evaluated every time it’s called. After that, they can’t cache values

    In the method calling parenthesis is there and you can send one or more parameter in that.

    0 讨论(0)
  • 2020-11-28 01:36

    Here’s a breakdown of this question.

    When to use methods

    • To react to some event happening in the DOM
    • To call a function when something happens in your component.
    • You can call a method from computed properties or watchers.

    When to use computed properties

    • You need to compose new data from existing data sources
    • You have a variable you use in your template that’s built from one or more data properties
    • You want to reduce a complicated, nested property name to a more readable and easy to use one (but update it when the original property changes)
    • You need to reference a value from the template. In this case, creating a computed property is the best thing, because it’s cached.
    • You need to listen to changes of more than one data property
    0 讨论(0)
  • 2020-11-28 01:37

    From the docs

    ..computed properties are cached based on their dependencies. A computed property will only re-evaluate when some of its dependencies have changed.

    If you want data to be cached use Computed properties on the other hand if you don't want data to be cached use simple Method properties.

    0 讨论(0)
  • 2020-11-28 01:44

    Computed values and methods are very different in Vue and are definitely not interchangeable in most cases.

    Computed Property

    A more appropriate name for a computed value is a computed property. In fact, when the Vue is instantiated, computed properties are converted into a property of the Vue with a getter and sometimes a setter. Basically you can think of a computed value as a derived value that will be automatically updated whenever one of the underlying values used to calculate it is updated. You don't call a computed and it doesn't accept any parameters. You reference a computed property just like you would a data property. Here's the classic example from the documentation:

    computed: {
      // a computed getter
      reversedMessage: function () {
        // `this` points to the vm instance
        return this.message.split('').reverse().join('')
      }
    }
    

    Which is referenced in the DOM like this:

    <p>Computed reversed message: "{{ reversedMessage }}"</p>
    

    Computed values are very valuable for manipulating data that exists on your Vue. Whenever you want to filter or transform your data, typically you will use a computed value for that purpose.

    data:{
        names: ["Bob", "Billy", "Mary", "Jane"]
    },
    computed:{
        startsWithB(){
            return this.names.filter(n => n.startsWith("B"))
        }
    }
    
    <p v-for="name in startsWithB">{{name}}</p>
    

    Computed values are also cached to avoid repetitively calculating a value that doesn't need to be re-calculated when it hasn't changed (as it might not be in a loop for example).

    Method

    A method is just a function bound to the Vue instance. It will only be evaluated when you explicitly call it. Like all javascript functions it accepts parameters and will be re-evaluated every time it's called. Methods are useful in the same situations any function is useful.

    data:{
        names: ["Bob", "Billy", "Mary", "Jane"]
    },
    computed:{
        startsWithB(){
            return this.startsWithChar("B")
        },
        startsWithM(){
            return this.startsWithChar("M")
        }
    },
    methods:{
        startsWithChar(whichChar){
            return this.names.filter(n => n.startsWith(whichCharacter))
        }
    }
    

    Vue's documentation is really good and easily accessible. I recommend it.

    0 讨论(0)
  • 2020-11-28 01:46

    As @gleenk asked for a practical example to make evident the cache and dependency differences between methods and computed properties, I'll show a simple scenario:

    app.js

    new Vue({
        el: '#vue-app',
        data: {
            a: 0,
            b: 0,
            age: 20
        },
        methods: {
            addToAmethod: function(){
                console.log('addToAmethod');
                return this.a + this.age;
            },
            addToBmethod: function(){
                console.log('addToBmethod');
                return this.b + this.age;
            }
        },
        computed: {
            addToAcomputed: function(){
                console.log('addToAcomputed');
                return this.a + this.age;
            },
            addToBcomputed: function(){
                console.log('addToBcomputed');
                return this.b + this.age;
            }
        }
    });
    

    Here we have 2 methods and 2 computed properties that perform the same task. The methods addToAmethod & addToBmethod and the computed properties addToAcomputed & addToBcomputed all add +20 (i.e. the age value) to either a or b. Regarding the methods, they are both called every time an action has been performed on any of the listed properties, even if the dependencies for one specific method have not changed. For the computed properties, the code is executed only when a dependency has changed; for example, one of the specific property values that refers to A or B will trigger addToAcomputed or addToBcomputed, respectively.

    The method and computed descriptions seem pretty similar, but as @Abdullah Khan has already specified it, they are not the same thing! Now let's try to add some html to execute everything together and see where the difference is.

    The Method case demo

    new Vue({
        el: '#vue-app',
        data: {
            a: 0,
            b: 0,
            age: 20
        },
        methods: {
            addToAmethod: function(){
                console.log('addToAmethod');
                return this.a + this.age;
            },
            addToBmethod: function(){
                console.log('addToBmethod');
                return this.b + this.age;
            }
        }
    });
    <!DOCTYPE html>
        <html>
            <head>
                <meta charset="utf-8">
                <title>VueJS Methods - stackoverflow</title>
                <link href="style.css" rel="stylesheet" />
                <script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.11/vue.min.js"></script>
        
            </head>
            <body>
                <div id="vue-app">
                    <h1>Methods</h1>
                    <button v-on:click="a++">Add to A</button>
                    <button v-on:click="b++">Add to B</button>
                    <p>Age + A = {{ addToAmethod() }}</p>
                    <p>Age + B = {{ addToBmethod() }}</p>
                </div>
            </body>
            
            <script src="app.js"></script>
        </html>

    The explained result

    When I click on the button "Add to A", all the methods are called (see the console log screen result above), the addToBmethod() is also executed but I didn't press the "Add to B" button; the property value that refers to B has not changed. The same behaviour comes if we decide to click the button "Add to B", because again both the methods will be called independently of dependency changes. According to this scenario this is bad practice because we are executing the methods every time, even when dependencies have not changed. This is really resource consuming because there is not a cache for property values that have not changed.

    method button method

    The Computed property case demo

    new Vue({
        el: '#vue-app',
        data: {
            a: 0,
            b: 0,
            age: 20
        },
    
        computed: {
            addToAcomputed: function(){
                console.log('addToAcomputed');
                return this.a + this.age;
            },
            addToBcomputed: function(){
                console.log('addToBcomputed');
                return this.b + this.age;
            }
        }
    });
    <!DOCTYPE html>
        <html>
            <head>
                <meta charset="utf-8">
                <title>VueJS Computed properties - stackoverflow</title>
                <link href="style.css" rel="stylesheet" />
                <script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.11/vue.min.js"></script>
            </head>
            <body>
                <div id="vue-app">
                    <h1>Computed Properties</h1>
                    <button v-on:click="a++">Add to A</button>
                    <button v-on:click="b++">Add to B</button>
                    <p>Age + A = {{ addToAcomputed }}</p>
                    <p>Age + B = {{ addToBcomputed }}</p>
                </div>
            </body>
            
            <script src="app.js"></script>
        </html>

    The explained result

    When I click on the button "Add to A", only the computed property addToAcomputed is called because, as we already said, the computed properties are executed only when a dependency has changed. And since I didn't press the button "Add to B" and the age property value for B has not changed, there is no reason to call and execute the computed property addToBcomputed. So, in a certain sense, the computed property is maintaining the "same unchanged" value for the B property like a kind of cache. And in this circumstance this is consider good practice.

    computed button computed

    0 讨论(0)
  • 2020-11-28 01:48

    Stumbled upon the same question. To me it's more clear like this:

    1. When Vue.js sees the v-on directive followed by a method, it knows exactly which method to call and when to call it.
    <button v-on:click="clearMessage">Clear message</button> // @click
    // method clearMessage is only called on a click on this button
    
    <input v-model="message" @keyup.esc="clearMessage" @keyup.enter="alertMessage" />
    /* The method clearMessage is only called on pressing the escape key
    and the alertMessage method on pressing the enter key */
    
    1. When a method is called without the v-on directive it will be called every time an event is triggered on the page that updates the DOM (or simply needs to re-render a part of the page). Even when that method has nothing to do with the event being triggered.
    <p>Uppercase message: {{ messageUppercase() }}</p>
    methods: {
       messageUppercase() {
          console.log("messageUpercase");
          return this.message.toUpperCase();
       }
    }
    /* The method `messageUppercase()` is called on every button click, mouse hover 
    or other event that is defined on the page with the `v-on directive`. So every
    time the page re-renders.*/
    
    1. A Computed property is only called when a property value is changed that is being referenced by the this word in its function definition.
    <p>Uppercase message: {{ messageUppercase }}</p> 
    data() {
     return {
        message: "I love Vue.js"
       }
     },
    computed: {
     messageUppercase() {
        console.log("messageUpercase");
        return this.message.toUpperCase();
     }
    }
    /* The computed property messageUppercase is only called when the propery message is
    changed. Not on other events (clicks, mouse hovers,..) unless of course a specific 
    event changes the value of message.  */
    

    The take away here is that it's best practice to use the computed properties in case a method is not being called with the v-on directive.

    0 讨论(0)
提交回复
热议问题