最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - I am having a problem understanding the different behavior of $("button").click() and $(&quot

programmeradmin1浏览0评论

While I am trying to learn jquery I learned that $(selector) returns an object which has all the match of that selector and is iterable like arrays. eg $("button") will return an object that will have access to all the button tag of DOM in such way that to access the first button tag you can use $["button"][0] for second you can use $["button"][1] and so on.

So here below is code focus on mented line 1 and line 2.

   <body>
        <button>Click me</button>
        <script>
            $(document).ready(function() {
// line 1
               $("button").click(function() {
                    console.log("1");
// line 2
                    $("button").click(); 
                });
            });
        </script>
    </body>

line 2 inside line1 event handler function is set up an infinite loop as you can see that when I Click on "Click me" button it will trigger line1 inside of which is line 2 which too will again trigger line 1 and so on. now see the below code snippet with changed line2.

        <script>
            $(document).ready(function() {
// line 1
               $("button").click(function() {
                    console.log("1");
// line 2
                    $("button")[0].click();
                });
            });
        </script>

This time it is not setting up an infinite loop but prints to console "1" only two times why so?

While I am trying to learn jquery I learned that $(selector) returns an object which has all the match of that selector and is iterable like arrays. eg $("button") will return an object that will have access to all the button tag of DOM in such way that to access the first button tag you can use $["button"][0] for second you can use $["button"][1] and so on.

So here below is code focus on mented line 1 and line 2.

   <body>
        <button>Click me</button>
        <script>
            $(document).ready(function() {
// line 1
               $("button").click(function() {
                    console.log("1");
// line 2
                    $("button").click(); 
                });
            });
        </script>
    </body>

line 2 inside line1 event handler function is set up an infinite loop as you can see that when I Click on "Click me" button it will trigger line1 inside of which is line 2 which too will again trigger line 1 and so on. now see the below code snippet with changed line2.

        <script>
            $(document).ready(function() {
// line 1
               $("button").click(function() {
                    console.log("1");
// line 2
                    $("button")[0].click();
                });
            });
        </script>

This time it is not setting up an infinite loop but prints to console "1" only two times why so?

Share Improve this question edited Jun 24, 2019 at 5:24 user8795381 asked Jun 24, 2019 at 5:20 Yogesh_SinghYogesh_Singh 4251 gold badge5 silver badges9 bronze badges 6
  • What jQuery is doing is pretty understandable, but an addEventListener handler appears to run twice jsfiddle/xtu24ny5 – CertainPerformance Commented Jun 24, 2019 at 5:33
  • @CertainPerformance how what jquery is doing is understandable and also again why is your code only printing 'click' to console only two times and is not triggering an infinite loop? – Yogesh_Singh Commented Jun 24, 2019 at 5:58
  • 1 In the first snippet, jQuery's .click() triggers the jQuery-attached event, which subsequently calls .click() again recursively. An infinite loop isn't surprising. What's surprising is that calling the (built-in) .click method on the element runs twice. I'd have expected infinite recursion like with jQuery, or (if recursion is not permitted) a single log of 1. It looks like programmatic .click()s are only permitted to fire handlers once per... event loop? But I have no idea why, or any reference on this sort of behavior – CertainPerformance Commented Jun 24, 2019 at 6:04
  • Guessing here but I can only assume that HTMLElement.click() inspects the event and won't re-fire if it originated from .click(). It probably does this to specifically avoid infinite loops – Phil Commented Jun 24, 2019 at 6:42
  • @CertainPerformance yes, I too am saying that I can understand what snippet 1 is doing but can't the behavior of the second snippet, when you said what jquery is doing is understandable I thought you meant to say the second code snippet's output is expected and defined. so, sorry I messed it up there. Also, for now, I am considering the behavior of $[selector"][index].function() as undefined because when i did $["form"][0].submit(eventHandler); it doesn't goes to eventHandler and directly submits the form. – Yogesh_Singh Commented Jun 24, 2019 at 6:55
 |  Show 1 more ment

4 Answers 4

Reset to default 6

It is most likely the way in which the synthetic event is being invoked by JQuery.

Here is the JQuery snippet for event dispatch.

dispatch: function( nativeEvent ) {

        // Make a writable jQuery.Event from the native event object
        var event = jQuery.event.fix( nativeEvent );

        var i, j, ret, matched, handleObj, handlerQueue,
            args = new Array( arguments.length ),
            handlers = ( dataPriv.get( this, "events" ) || {} )[ event.type ] || [],
            special = jQuery.event.special[ event.type ] || {};

        // Use the fix-ed jQuery.Event rather than the (read-only) native event
        args[ 0 ] = event;

        for ( i = 1; i < arguments.length; i++ ) {
            args[ i ] = arguments[ i ];
        }

        event.delegateTarget = this;

        // Call the preDispatch hook for the mapped type, and let it bail if desired
        if ( special.preDispatch && special.preDispatch.call( this, event ) === false ) {
            return;
        }

        // Determine handlers
        handlerQueue = jQuery.event.handlers.call( this, event, handlers );

        // Run delegates first; they may want to stop propagation beneath us
        i = 0;
        while ( ( matched = handlerQueue[ i++ ] ) && !event.isPropagationStopped() ) {
            event.currentTarget = matched.elem;

            j = 0;
            while ( ( handleObj = matched.handlers[ j++ ] ) &&
                !event.isImmediatePropagationStopped() ) {

                // If the event is namespaced, then each handler is only invoked if it is
                // specially universal or its namespaces are a superset of the event's.
                if ( !event.rnamespace || handleObj.namespace === false ||
                    event.rnamespace.test( handleObj.namespace ) ) {

                    event.handleObj = handleObj;
                    event.data = handleObj.data;

                    ret = ( ( jQuery.event.special[ handleObj.origType ] || {} ).handle ||
                        handleObj.handler ).apply( matched.elem, args );

                    if ( ret !== undefined ) {
                        if ( ( event.result = ret ) === false ) {
                            event.preventDefault();
                            event.stopPropagation();
                        }
                    }
                }
            }
        }

        // Call the postDispatch hook for the mapped type
        if ( special.postDispatch ) {
            special.postDispatch.call( this, event );
        }

        return event.result;
    }

Reading the spec for synthetic and authentic click events:

When a user agent is to run post-click activation steps on an element, it must run the activation behavior defined for that element, if any. Activation behaviors can refer to the click event that was fired by the steps above leading up to this point.

As JQuery has implemented a custom dispatch handler for emulating click events it looks like they have added the following in order to emulate that aforementioned native html event lifecycle.

if ( special.postDispatch ) {
   special.postDispatch.call( this, event );
}

This call here is likely the suspect, proven when you look at the console output for the following. Note that you never see the console.log Done because the dispatch handler is never actually returning the event to plete it's native lifecycle (endless recursion), which I suspect is just frankly handled better by the native html implementation.

Infinite loop warning

<script src="https://cdnjs.cloudflare./ajax/libs/jquery/3.3.0/jquery.min.js"></script>
<body>
        <button>Click me</button>
        <script>
            $(document).ready(function() {
// line 1
               $("button").click(function() {
                    console.log("start");
// line 2
                    $("button").click(); 
                    console.log("Done");
                });
            });
        </script>
    </body>

Good find!

When you do $("button")[0] you are getting back the underlying native HTMLButtonElement object that is not a jQuery object.

$('button').click(fn) does things like below

    $("button").each(function() {
        //this === native button htmlElement.
        if (!this._event) this._event = {};
        if (!this._event.click) this._event.click = [];
        this._event.click.push(fn);
        if (!this.clickHandler) {
            this.clickHandler = e => {
                this._event.click.forEach(f => f.bind(this)(e));
            };
            this.addEventListener("click", this.clickHandler);
        }
    });

$('button').click() means

    $("button").each(function() {
        //this === native button htmlElement.
        if(this.clickHandler)this.clickHandler();
        if(typeof this.onclick === 'function') this.onclick();
    });

It's just an example, the jquery event source code is much more plex.

And why $("button")[0].click() prints '1' only twice? native click simulates a mouse click on the button element,so I assume that the loop is blocked by Explorer for a security reason.

bind events source code

trigger source code

$(“button”) returns a jq object,it calls methods on the jq object and $(“button”)[0] returns a DOM object,it calls the dom native method

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论