有没有和谐的'从ES6类实例获取类名的方法?
以外someClassInstance.constructor.name
目前我依靠Traceur实施。似乎Babel有Function.name
的polyfill而Traceur没有。
总结一下:ES6 / ES2015 / Harmony中没有其他方式,ES中没有预期的ATM.Next。
它可以为未经管理的服务器端应用程序提供有用的模式,但在用于浏览器/桌面/移动设备的应用程序中是不需要的。
Babel uses core-js
到polyfill Function.name
,应该根据需要手动加载Traceur和TypeScript应用程序。
答案 0 :(得分:166)
someClassInstance.constructor.name
正是这样做的正确方法。运营商可能不支持这一点,但它是符合规范的标准方式。 (通过ClassDeclaration产生声明的函数的name
属性在14.5.15中设置,步骤6。)
答案 1 :(得分:44)
正如@Domenic所说,使用someClassInstance.constructor.name
。 @Esteban在评论中提及
someClassInstance.constructor
是一个函数。所有函数都有name
属性...
因此,要静态访问类名,请执行以下操作(这适用于我的Babel版本BTW。根据对@Domenic的评论,您的里程可能会有所不同)。
class SomeClass {
constructor() {}
}
var someClassInstance = new SomeClass();
someClassInstance.constructor.name; // === 'SomeClass'
SomeClass.name // === 'SomeClass'
<强>更新强>
Babel很好,但uglify / minification确实最终导致了我的问题。我正在制作游戏,并且正在创建池化Sprite资源的哈希值(其中键是函数名称)。缩小后,每个函数/类都被命名为t
。这会杀死哈希。我在这个项目中使用Gulp
,在阅读gulp-uglify docs后我发现有一个参数可以防止这个局部变量/函数名称发生变形。所以,在我的gulpfile中,我改变了
.pipe($.uglify())
至.pipe($.uglify({ mangle: false }))
这里有性能与可读性的权衡。不破坏名称将导致(略微)更大的构建文件(更多网络资源)和可能更慢的代码执行(需要引用 - 可能是BS)。另一方面,如果我保持相同,我必须在每个ES6类上手动定义getClassName
- 在静态和实例级别。不,谢谢!
<强>更新强>
在评论中讨论之后,似乎避免使用.name
约定来支持定义这些函数是一个很好的范例。它只需要几行代码,并且可以完全缩小代码的一般性和通用性(如果在库中使用)。所以我想我会改变主意并在我的课程上手动定义getClassName
。谢谢@estus!。与直接变量访问相比,Getter / Setter通常是一个好主意,特别是在基于客户端的应用程序中。
class SomeClass {
constructor() {}
static getClassName(){ return 'SomeClass'; }
getClassName(){ return SomeClass.getClassName(); }
}
var someClassInstance = new SomeClass();
someClassInstance.constructor.getClassName(); // === 'SomeClass' (static fn)
someClassInstance.getClassName(); // === 'SomeClass' (instance fn)
SomeClass.getClassName() // === 'SomeClass' (static fn)
答案 2 :(得分:5)
先前的回答解释说someClassInstance.constructor.name
可以很好地工作,但是如果您需要以编程方式将类名转换为字符串,并且不想为此创建实例,请记住:
typeof YourClass === "function"
而且,由于每个函数都有一个name
属性,所以使用类名获取字符串的另一种不错的方法是:
YourClass.name
下面是一个很好的例子,说明为什么这样做很有用。
MDN documentation教我们,这就是您加载Web组件的方式:
customElements.define("your-component", YourComponent);
YourComponent
是从HTMLElement
扩展的类。由于在组件标签本身之后命名组件的类是一种好习惯,因此最好编写一个辅助函数,所有组件都可以使用该函数来注册自己。这是该函数:
function registerComponent(componentClass) {
const componentName = upperCamelCaseToSnakeCase(componentClass.name);
customElements.define(componentName, componentClass);
}
所以您要做的就是:
registerComponent(YourComponent);
这很不错,因为它比自己编写component标签要容易出错。总结一下,这是upperCamelCaseToSnakeCase()
函数:
// converts `YourString` into `your-string`
function upperCamelCaseToSnakeCase(value) {
return value
// first char to lower case
.replace(/^([A-Z])/, $1 => $1.toLowerCase())
// following upper chars get preceded with a dash
.replace(/([A-Z])/g, $1 => "-" + $1.toLowerCase());
}
答案 3 :(得分:1)
如果您将Babel与@babel/preset-env
一起使用,则可以保留类定义而无需将它们转换为函数(这会删除constructor
属性)
您可以在babel.config / babelrc
中删除与此配置之间的旧浏览器兼容性:
{
"presets": [
["@babel/preset-env", {"targets": {"browsers": ["> 2%"]}}]
]
}
有关
targets
的更多信息:https://babeljs.io/docs/en/babel-preset-env#targets
看来目前没有简单的解决方案...我们需要研究如何处理排除问题。