自动将 console.log 添加到每个函数

Posted

技术标签:

【中文标题】自动将 console.log 添加到每个函数【英文标题】:Adding console.log to every function automatically 【发布时间】:2011-06-29 08:42:51 【问题描述】:

有没有办法通过在某处注册一个全局钩子(即,不修改实际函数本身)或通过其他方式,使任何函数在调用时输出一个 console.log 语句?

【问题讨论】:

很好的问题,我很想知道这是否可能,但我很确定它不是...也许添加一个功能请求,以便将其添加到您的 js 引擎中最喜欢的浏览器? :-) 完美的问题,我需要类似的东西 【参考方案1】:

这是一种使用您选择的函数来扩充全局命名空间中所有函数的方法:

function augment(withFn) 
    var name, fn;
    for (name in window) 
        fn = window[name];
        if (typeof fn === 'function') 
            window[name] = (function(name, fn) 
                var args = arguments;
                return function() 
                    withFn.apply(this, args);
                    return fn.apply(this, arguments);

                
            )(name, fn);
        
    


augment(function(name, fn) 
    console.log("calling " + name);
);

一个缺点是调用augment 后创建的任何函数都不会有额外的行为。

【讨论】:

它是否正确处理函数的返回值? @SunnyShah 不,它不会:jsfiddle.net/Shawn/WnJQ5 但这个可以:jsfiddle.net/Shawn/WnJQ5/1 虽然我不确定它是否适用于所有情况......不同之处在于将fn.apply(this, arguments); 更改为return fn.apply(this, arguments); @Shawn @SunnyShah 已修复。只需要在最里面的函数中添加一个return 几乎可以正常工作,但是我在使用这个 jquery:call :if (jQuery.isFunction(lSrc)) 时遇到错误,它说 :TypeError: jQuery.isFunction is not a function 这个解决方案不使用jQuery【参考方案2】:

您实际上可以将自己的函数附加到 console.log 以获取所有加载的内容。

console.log = function(msg) 
    // Add whatever you want here
    alert(msg); 

【讨论】:

正确与否,它不回答问题。一点也不。 (以防有人仍然感到困惑)【参考方案3】:

这里有一些 javascript 代替了将 console.log 添加到 Javascript 中的每个函数;在Regex101 上玩它:

$re = "/function (.+)\\(.*\\)\\s*\\/m"; 
$str = "function example()"; 
$subst = "$& console.log(\"$1()\");"; 
$result = preg_replace($re, $subst, $str);

这是一个“快速而肮脏的 hack”,但我发现它对调试很有用。如果你有很多函数,要小心,因为这会添加很多代码。此外,RegEx 很简单,可能不适用于更复杂的函数名称/声明。

【讨论】:

【参考方案4】:

如果您想要更有针对性的日志记录,以下代码将记录特定对象的函数调用。您甚至可以修改对象原型,以便所有新实例也可以记录。我使用了 Object.getOwnPropertyNames 而不是 for...in,因此它适用于没有可枚举方法的 ECMAScript 6 类。

function inject(obj, beforeFn) 
    for (let propName of Object.getOwnPropertyNames(obj)) 
        let prop = obj[propName];
        if (Object.prototype.toString.call(prop) === '[object Function]') 
            obj[propName] = (function(fnName) 
                return function() 
                    beforeFn.call(this, fnName, arguments);
                    return prop.apply(this, arguments);
                
            )(propName);
        
    


function logFnCall(name, args) 
    let s = name + '(';
    for (let i = 0; i < args.length; i++) 
        if (i > 0)
            s += ', ';
        s += String(args[i]);
    
    s += ')';
    console.log(s);


inject(Foo.prototype, logFnCall);

【讨论】:

【参考方案5】:

记录函数调用的代理方法

在 JS 中有一种使用Proxy 的新方法来实现此功能。 假设我们希望在调用特定类的函数时拥有console.log

class TestClass 
  a() 
    this.aa = 1;
  
  b() 
    this.bb = 1;
  


const foo = new TestClass()
foo.a() // nothing get logged

我们可以用覆盖这个类的每个属性的代理来替换我们的类实例化。所以:

class TestClass 
  a() 
    this.aa = 1;
  
  b() 
    this.bb = 1;
  



const logger = className => 
  return new Proxy(new className(), 
    get: function(target, name, receiver) 
      if (!target.hasOwnProperty(name)) 
        if (typeof target[name] === "function") 
          console.log(
            "Calling Method : ",
            name,
            "|| on : ",
            target.constructor.name
          );
        
        return new Proxy(target[name], this);
      
      return Reflect.get(target, name, receiver);
    
  );
;



const instance = logger(TestClass)

instance.a() // output: "Calling Method : a || on : TestClass"

check that this actually works in Codepen


请记住,使用 Proxy 可以为您提供比仅记录控制台名称更多的功能。

此方法也适用于 Node.js

【讨论】:

不使用实例和类也能做到这一点吗?具体来说在 node.js 中? @Revadike 这应该会有所帮助:***.com/a/28708700/5284370【参考方案6】:

对我来说,这看起来是最优雅的解决方案:

(function() 
    var call = Function.prototype.call;
    Function.prototype.call = function() 
        console.log(this, arguments); // Here you can do whatever actions you want
        return call.apply(this, arguments);
    ;
());

【讨论】:

RangeError: 超出最大调用堆栈大小。节点js的任何替代品

以上是关于自动将 console.log 添加到每个函数的主要内容,如果未能解决你的问题,请参考以下文章

Angular中的单元测试点击事件

实用vscode插件:自动帮你写console.log —— Turbo Console Log

是否可以通过管道传输到 console.log?

JS六....

如何将每个项目添加到 listview android

bind的使用