- Get class methods in typescript
- More advanced
- Extract methods
- Reflection in TypeScript
- Get Functions (Methods) of a Class
- How do I get list of methods in a Python class?
- How to get all functions of es6 class instance
- Get class methods in typescript
- More advanced
- Extract methods
- Reflection in TypeScript
- Getting all class methods in classes in current file in Python?
- Get methods of class in JavaScript
- Get return type of class method via method name in Typescript
- Can I get all methods of a class?
- Get function reference of class (not object)
- Get class methods in typescript
- More advanced
- Extract methods
- Reflection in TypeScript
- Solution 2
- Solution 3
Get class methods in typescript
I had a very hard time getting other answers to work. They also don’t cover a way to do it without an instance, if that’s helpful to you as it was for me.
This is what I came up with:
import < route >from "../../lib/route_decorator"; export class SomeClass < index() < console.log("here"); >>
let ctrl = require("./filepath/filename"); // This is because angular exports as `exports.SomeClass = SomeClass;` ctrl = ctrl[Object.keys(ctrl)[0]]; let ctrlObj = new ctrl(); // Access from Class w/o instance console.log(Reflect.ownKeys(ctrl.prototype)); // Access from instance console.log(Reflect.ownKeys(Object.getPrototypeOf(ctrlObj)));
[ 'constructor', 'index' ] [ 'constructor', 'index' ]
You forget that TypeScript is Javascript. Remember that the TypeScript compiler compiles your code into Javascript.
So, as you normally do in Javascript, you can enumerate members on an object like this:
UtilityClass myclass = . ; for (var member in myclass) < /* do something */ >
More advanced
If you want to make sure you don’t get inherited members:
for (var member in myclass) < if (myclass.hasOwnProperty(member)) < /* do something */ >>
Extract methods
If you want to make sure you get only methods (functions):
for (var member in myclass) < // For each member of the dictionary if (typeof myclass[member] == "function") < // Is it a function? if (myclass.hasOwnProperty(member)) < // Not inherited // do something. >> >
Reflection in TypeScript
As you can see the approaches require an instance to work on. You don’t work on the class. Reflection is what you are trying to achieve in the context of OOP; however Javascript (which is not OOP) deals with it in a different way.
Get Functions (Methods) of a Class
This function will get all functions. Inherited or not, enumerable or not. All functions are included.
function getAllFuncs(toCheck) const props = [];
let obj = toCheck;
do props.push(. Object.getOwnPropertyNames(obj));
> while (obj = Object.getPrototypeOf(obj));
return props.sort().filter((e, i, arr) => <
if (e!=arr[i+1] && typeof toCheck[e] == 'function') return true;
>);
>
["constructor", "toString", "toLocaleString", "join", "pop", "push", "concat", "reverse", "shift", "unshift", "slice", "splice", "sort", "filter", "forEach", "some", "every", "map", "indexOf", "lastIndexOf", "reduce", "reduceRight", "entries", "keys", "constructor", "toString", "toLocaleString", "valueOf", "hasOwnProperty", "isPrototypeOf", "propertyIsEnumerable", "__defineGetter__", "__lookupGetter__", "__defineSetter__", "__lookupSetter__"]
It doesn’t return functions defined via symbols;
How do I get list of methods in a Python class?
An example (listing the methods of the optparse.OptionParser class):
>>> from optparse import OptionParser
>>> import inspect
#python2
>>> inspect.getmembers(OptionParser, predicate=inspect.ismethod)
[([('__init__', ),
.
('add_option', ),
('add_option_group', ),
('add_options', ),
('check_values', ),
('destroy', ),
('disable_interspersed_args',
),
('enable_interspersed_args',
),
('error', ),
('exit', ),
('expand_prog_name', ),
.
]
# python3
>>> inspect.getmembers(OptionParser, predicate=inspect.isfunction)
.
Notice that getmembers returns a list of 2-tuples. The first item is the name of the member, the second item is the value.
You can also pass an instance to getmembers :
>>> parser = OptionParser()
>>> inspect.getmembers(parser, predicate=inspect.ismethod)
.
How to get all functions of es6 class instance
You have to call Object.getOwnPropertyNames() on the prototype property of the class.
class Test < methodA() <>methodB() <>>
console.log(Object.getOwnPropertyNames(Test.prototype))
Get class methods in typescript
You forget that TypeScript is Javascript. Remember that the TypeScript compiler compiles your code into Javascript.
So, as you normally do in Javascript, you can enumerate members on an object like this:
UtilityClass myclass = . ;
for (var member in myclass) < /* do something */ >
More advanced
If you want to make sure you don’t get inherited members:
for (var member in myclass) if (myclass.hasOwnProperty(member)) /* do something */
>
>
Extract methods
If you want to make sure you get only methods (functions):
for (var member in myclass) < // For each member of the dictionary
if (typeof myclass[member] == "function") < // Is it a function?
if (myclass.hasOwnProperty(member)) < // Not inherited
// do something.
>
>
>
Reflection in TypeScript
As you can see the approaches require an instance to work on. You don’t work on the class. Reflection is what you are trying to achieve in the context of OOP; however Javascript (which is not OOP) deals with it in a different way.
Getting all class methods in classes in current file in Python?
On Python 3, calling inspect.ismethod on an attribute of a class that happens to be a function will always be False, because it will just be a plain function. function.__get__ only returns a method object when accessed as an attribute of an instance.
If you want to get all «methods» of the class just use inspect.isfunction .
>>> class A:
. def __init__(self): pass
.
>>> A.__init__
>>> inspect.ismethod(A.__init__)
False
>>> inspect.isfunction(A.__init__)
True
>>> inspect.ismethod(A().__init__)
True
Get methods of class in JavaScript
You can use Object.getOwnPropertyNames and filter the instance and static methods:
class c < methodA()<>static methodB()<>
log() static logStatic()>const instanceOnly = Object.getOwnPropertyNames(c.prototype) .filter(prop => prop != "constructor");console.log(instanceOnly);const staticOnly = Object.getOwnPropertyNames(c) .filter(prop => typeof c[prop] === "function");console.log(staticOnly);
Get return type of class method via method name in Typescript
Afaik ,there is no safe way to do what you want without changing function body or using type assertion.
In order to validate function arguments, first of all we need to obtain all method keys from Foo :
class Foo var1: string = ‘var1’;
var2: string = ‘var2’;
hello(request: string) < >
world(request: number) < >
>
// This type reflects any function/method
type Fn = (. args: any[]) => any
type ObtainMethods = [Prop in keyof T]: T[Prop] extends Fn ? Prop : never
>Typescript get class methods
// «hello» | «world»
type AllowedMethods = ObtainMethods
const executeFoo = >(
methodName: Method
) => < >
executeFoo('hello') // ok
executeFoo('world') // ok
executeFoo('var1') // expected error
However, there is a problem with second argument:
const executeFoo = >(
methodName: Method, parameter: Parameters[0]
) => // Argument of type 'string | number' is not assignable to parameter of type 'never'. Type 'string' is not assignable to type 'never'.
foo[methodName](parameter)
>
As you might have noticed, there is an error.
Argument of type 'string | number' is not assignable to parameter of type 'never'.
Type 'string' is not assignable to type 'never'.
It is very important. If you try to call foo[methodName]() you will see that this function expects never as a type for first argument. This is because
Likewise, multiple candidates for the same type variable in contra-variant positions causes an intersection type to be inferred.
You can find more in my article, in the first part. This is because TS does not know which methodName you are using exactly. Hence, TS compiler intersects all parameters from methods: string & number because this is the only safe way to make function signature safe.
SO, it is very important what type of argument are you expect in your methods.
How to fix it ?
In this particular example, I believe using type assertion is justified:
const executeFoo = >(
methodName: Method, parameter: Parameters[0]
) => (foo[methodName] as (arg: Parameters[0]) => void)(parameter)
>
executeFoo('hello', 'str') // ok
executeFoo('world', 42) // ok
executeFoo('world', "42") // expected error
executeFoo('var1') // expected error
If you are interested in function argument inference you can check my blog
It is also possible to use conditional statement for type narrowing (works in TS >= 4.6)
type Fn = (. args: any[]) => any
type ObtainMethods = [Prop in keyof T]: T[Prop] extends Fn ? Prop : never
>Typescript get class methods
// "hello" | "world"
type AllowedMethods = ObtainMethods
type Values = TTypescript get class methods
type AllowedArguments = [Method in AllowedMethods]: [Method, Parameters[0]]
>
const foo = new Foo();
const executeFoo = (
. [name, arg]: Values
) => if (name === 'hello') foo[name](arg)
> else foo[name](arg)
>
>
executeFoo('hello', 'str') // ok
executeFoo('world', 42) // ok
executeFoo('world', "42") // expected error
executeFoo('var1') // expected error
but it does not make much sense.
Can I get all methods of a class?
To know about all methods use this statement in console:
javap -cp jar-file.jar packagename.classname
javap class-file.class packagename.classname
Get function reference of class (not object)
The methods defined with method syntax in the body of a class construct that aren’t marked static are prototype methods and so they’re on Temp.prototype , not Temp itself. So that’s where you’d update them:
Temp.prototype.hi = modifyMethod(Temp.prototype.hi);
Only static methods end up on Temp itself.
You may see other functions created within the class body using the class fields proposal’s syntax:
Those are instance methods. They’re created by the constructor, and re-created for each instance, roughly as though they’d been written like this:¹
class Temp constructor() this.hi = () => //
>;
>
>
You can’t wrap those until/unless an instance is created, as they’re instance-specific.
class Temp static staticMethod() // .
>
prototypeMethod() // .
>
instanceMethod = () => // .
>;
constructor() this.anotherInstanceMethod = () => // .
>;
this.yetAnotherInstanceMethod = function // .
>;
>
>
That class shows the three types of methods:
- Static Methods, such as staticMethod , which you’ll find on Temp (e.g., Temp.staticMethod );
- Prototype Methods, such as prototypeMethod , which you’ll find on Temp.prototype (e.g., Temp.prototype.prototypeMethod ); and
- Instance Methods, such as instanceMethod , anotherInstanceMethod , and yetAnotherInstanceMethod , which you’ll find on the instances themselves, if/when any instances are created
¹ Technically, they’re created as though with Object.defineProperty like this:
class Temp constructor() Object.defineProperty(this, "hi", value: () => //
>,
writable: true,
configurable: true,
enumerable: true
>);
>
>
. not via simple assignment. I used simple assignment in the example to keep it. simple. 🙂
Get class methods in typescript
You forget that TypeScript is Javascript. Remember that the TypeScript compiler compiles your code into Javascript.
So, as you normally do in Javascript, you can enumerate members on an object like this:
UtilityClass myclass = . ; for (var member in myclass) < /* do something */ >
More advanced
If you want to make sure you don’t get inherited members:
for (var member in myclass) < if (myclass.hasOwnProperty(member)) < /* do something */ >>
Extract methods
If you want to make sure you get only methods (functions):
for (var member in myclass) < // For each member of the dictionary if (typeof myclass[member] == "function") < // Is it a function? if (myclass.hasOwnProperty(member)) < // Not inherited // do something. >> >
Reflection in TypeScript
As you can see the approaches require an instance to work on. You don’t work on the class. Reflection is what you are trying to achieve in the context of OOP; however Javascript (which is not OOP) deals with it in a different way.
Solution 2
I had a very hard time getting other answers to work. They also don’t cover a way to do it without an instance, if that’s helpful to you as it was for me.
This is what I came up with:
import < route >from "../../lib/route_decorator"; export class SomeClass < index() < console.log("here"); >>
let ctrl = require("./filepath/filename"); // This is because angular exports as `exports.SomeClass = SomeClass;` ctrl = ctrl[Object.keys(ctrl)[0]]; let ctrlObj = new ctrl(); // Access from Class w/o instance console.log(Reflect.ownKeys(ctrl.prototype)); // Access from instance console.log(Reflect.ownKeys(Object.getPrototypeOf(ctrlObj)));
[ 'constructor', 'index' ] [ 'constructor', 'index' ]
Solution 3
Andry’s answer works well for ES3/ES5, but things get weird if you use ES6 and inheritance.
Here is a little more complex example, and how to retrieve each type of method in case of ES6:
class ParentTest < parentFoo = function() < // do stuff >parentBar() < // do stuff >> class Test extends ParentTest < foo = function() < // do stuff >bar() < // do stuff >> let instance = new Test(); // These two works equally for every method declared as a property of type "function" for(let method in instance) < console.log(method); // foo, foo2 >console.log(Object.keys(instance)); // Array [ "parentFoo", "foo" ] // This works for method of the class let protoOfTest = Object.getPrototypeOf(instance); console.log(Object.getOwnPropertyNames(protoOfTest)); // Array [ "constructor", "bar" ] // This works for methods of the extended class let protoOfParentTest = Object.getPrototypeOf(protoOfTest); console.log(Object.getOwnPropertyNames(protoOfParentTest)); // Array [ "constructor", "parentBar" ]