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

javascript - Get the class function name within itself - Stack Overflow

programmeradmin3浏览0评论

I'm trying to setup logging across my typescript program, using log4javascript.

However I have no idea how to retrieve the function names using reflection (rather than typed manually).

Ideally I want to emulate what I do in C#:

public class Foo 
{
    private static readonly log4net.ILog logger = log4net.LogManager.GetLogger(typeof(Foo));

    public Foo()
    {
    }

    public FooMethod()
    {
        try {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Trace, "Entering" + MethodBase.GetCurrentMethod().Name, null);
            // code
        }
        catch (e) {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Debug, ex.Message, null);
        }
        finally {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Trace, "Exiting" + MethodBase.GetCurrentMethod().Name, null);
        }
    }
}

How can I do this in Typescript? All I can do is get the class name.

class Foo {
    private static logger: log4javascript.Logger = log4javascript.getLogger(getName(Foo));

    constructor() {
    }

    FooFunction() {
        try {
            SymDataSource.logger.trace("Entering: " + getName(Foo.prototype.FooFunction));
            // code
        } catch (e) {
            SymDataSource.logger.debug("Exception: " + getName(Foo.prototype.FooFunction), e);
        } finally {
            SymDataSource.logger.trace("Exiting: " + getName(Foo.prototype.FooFunction));
        }
    }
}

function getName(obj: any): string {
   if (obj.name) {
       return obj.name;
   }
   var funcNameRegex = /function (.{1,})\(/;
   var results = (funcNameRegex).exec((<any> obj).constructor.toString());
   return (results && results.length > 1) ? results[1] : "";
}

The class name returns correctly, but the functions return as "Function".

I'm trying to setup logging across my typescript program, using log4javascript.

However I have no idea how to retrieve the function names using reflection (rather than typed manually).

Ideally I want to emulate what I do in C#:

public class Foo 
{
    private static readonly log4net.ILog logger = log4net.LogManager.GetLogger(typeof(Foo));

    public Foo()
    {
    }

    public FooMethod()
    {
        try {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Trace, "Entering" + MethodBase.GetCurrentMethod().Name, null);
            // code
        }
        catch (e) {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Debug, ex.Message, null);
        }
        finally {
            logger.Logger.Log(this.GetType(), log4net.Core.Level.Trace, "Exiting" + MethodBase.GetCurrentMethod().Name, null);
        }
    }
}

How can I do this in Typescript? All I can do is get the class name.

class Foo {
    private static logger: log4javascript.Logger = log4javascript.getLogger(getName(Foo));

    constructor() {
    }

    FooFunction() {
        try {
            SymDataSource.logger.trace("Entering: " + getName(Foo.prototype.FooFunction));
            // code
        } catch (e) {
            SymDataSource.logger.debug("Exception: " + getName(Foo.prototype.FooFunction), e);
        } finally {
            SymDataSource.logger.trace("Exiting: " + getName(Foo.prototype.FooFunction));
        }
    }
}

function getName(obj: any): string {
   if (obj.name) {
       return obj.name;
   }
   var funcNameRegex = /function (.{1,})\(/;
   var results = (funcNameRegex).exec((<any> obj).constructor.toString());
   return (results && results.length > 1) ? results[1] : "";
}

The class name returns correctly, but the functions return as "Function".

Share Improve this question asked Oct 1, 2013 at 10:42 TimTim 3,0486 gold badges31 silver badges56 bronze badges 7
  • possible duplicate of How do I get the name of an object's type in JavaScript? or Get an objects Class name at runtime in TypeScript – Fenton Commented Oct 1, 2013 at 14:50
  • Are the definitelytyped typings out for log4javascript?? I've been waiting for those! – Allen Rice Commented Nov 22, 2013 at 18:52
  • I've written them myself. Would upload to the repository but haven't had the time to write the required test classes. – Tim Commented Nov 25, 2013 at 9:40
  • If you want my copies or want to do that then I'll email it to you if you wish. – Tim Commented Nov 25, 2013 at 9:41
  • Tim, would love to get hold of the typings for log4javascript. Do you have them available? Couldn't see an email address in your profile. – Dunco Commented May 2, 2014 at 6:27
 |  Show 2 more ments

2 Answers 2

Reset to default 6

For member functions it is actually not possible to use the simple toString parsing approach because of the way prototype functions are created e.g. :

Foo.prototype.FooFunction = function () {
    return getName(this.FooFunction);
};

Here toString would give you :

function () {
    return getName(this.FooFunction);
};

And you can see that the function name is not present between function () {.

What you can do is create an inline function (using the awesome fat arrow e.g. return getName(()=>this.FooFunction);) and parse this local function body to find the function name. Notice that in this way it is still TypeSafe and if you do a rename refactoring there are no magic strings that will go out of sync. So the plete implementation bees :

class Foo { 
    static className = getName(Foo);    
    constructor() {
    }
    FooFunction() {        
            return getName(()=>this.FooFunction);            
    }
}

function getName(obj: any): string {    
    if (obj.name) {
        return obj.name;
    }

    var funcNameRegex = /function (.{1,})\(/;   
    var results = (funcNameRegex).exec(obj.toString());
    var result = results && results.length > 1 && results[1];

    // Check to see custom implementation
    if(!result){
        funcNameRegex = /return _this.(.*);/;
        results = (funcNameRegex).exec(obj.toString());
        result = results && results.length > 1 && results[1];
    }
    return result || "";
}

console.log(Foo.className);
var foo = new Foo();
console.log(foo.FooFunction());

which will output :

Foo
FooFunction

I've ended up making two functions. 1 for current top method in stack, other for the type name.

  export function getName(): string {
     var result: string = "";
     var error: any = new Error("getStack");
     var stack: string[] = error.stack
        .replace(/^[^\(]+?[\n$]/gm, '') // Remove first useless line
        .replace(/^\s+at\s+/gm, '') // Remove 'at'
        .replace(/^Object.<anonymous>\s*\(/gm, '{anonymous}()@') // Replace Object.<anon> with {anon}
        .replace(/\s\(.*:[0-9]+:[0-9]+\)/gm, '') // Remove filename, col and row
        .split('\n');
     return stack[1]; // Previous method
  }

  export function getType(obj: any): string {
     if (obj.name) {
        return obj.name;
     }
     var results = /function (.{1,})\(/.exec(obj.constructor.toString());
     return (results && results.length > 1) ? results[1] : "";
  }
发布评论

评论列表(0)

  1. 暂无评论