- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
示例用例:
我有一个带有属性“myProperty”的对象,具有 getter 和 setter(自 EcmaScript 5 起支持“Property Getters 和 Setters”:https://www.w3schools.com/js/js_es5.asp):
var obj = {
myProperty:'myPropertyValue',
get myProperty(){
return this.property;
},
set myProperty(value){
this.property=value;
}
};
我想将该属性绑定(bind)到 View ,这是名为 bindProperty
的自定义函数的任务。
为了将属性myProperty
传递给函数,我可以做类似的事情
bindProperty(obj, 'myProperty');
但是,我想避免将属性名称作为硬编码字符串传递。字符串有一个缺点,即在重构过程中属性名称更改时它们不会更新。
如果我会使用
bindProperty(obj, obj.myProperty);
该函数只知道属性值“myPropertyValue”,而不知道该值的来源。
=>如何在不使用字符串的情况下传递/识别属性本身?
A.使用反射?
我可以想象类似的事情
bindProperty(obj, ()=>obj.myProperty);
函数 bindProperty
然后必须执行一些反射魔法来找出 lambda 表达式主体中的属性名称(伪代码):
let attributeName = getNameofArgumentFromBodyOfLambdaExpression(lambda);
obj[attributeName] = 'newValue';
=>JavaScript 中是否可以使用反射计算 lambda 表达式的主体来获取属性名称?
(我知道这可以用 .NET 语言完成,例如
Private Sub BindProperty(Of T)(propertyExpression As Expression(Of Func(Of T)))
Dim propertyName As String = GetPropertyName(propertyExpression)
'...
)
B.使用复杂属性
另一种选择是我使用包装属性对象,拥有自己的 getter 和 setter。但是,那么我必须使用像
这样的属性obj.myProperty.set('newValue')
或
obj.myProperty('newValue') //similar to knockout observables
我仍然希望能够使用出色的 Getter/Setter 功能。换句话说:我想像普通属性一样使用我的属性:
obj.myProperty = 'newValue'
因此,这对我来说不是一个选择,我更愿意使用字符串而不是 B。
C.还有其他选择吗?
最佳答案
JavaScript 中的对象或多或少只是字符串或符号到值的映射。在运行时环境中没有可以调用的真正反射,可以让您从值向后移动到属性名称。
如果您需要的只是重构,那么执行此操作的一种方法是将您的 IDE 配置为通过 Flow 或 Typescript 或类似方式提供某种类型信息来识别字符串访问器(类型信息可能是是什么允许反射在 .NET 等语言中工作)。或者,您可以选择像“viewable_propName”这样的唯一前缀,然后在需要重命名时进行简单的查找和替换。
如果您确实专注于在没有类型信息和当前 ES6 语法的情况下使其工作,您可以执行以下操作:
function getNameofPropFromVal(obj, val){
for(let prop in obj){
if(obj[prop] === val) return prop;
}
}
obj[getNameofPropFromVal(obj, obj.myProp)] = 'newVal';
虽然这有缺点:(1) 不保证两个属性(property)不会共享相同的值(value)。(2) 引入了不必要的运行时开销。
最后,如果您愿意走在前沿并使用像 babel
这样的转换器,您可以在 bindProperty
方法中使用装饰器。这样您就可以在对象定义本身中进行绑定(bind)。这是an article解释要点,这里是 the more formal ECMAScript proposal .
关于javascript - 如何在不使用硬编码字符串的情况下传递javascript属性(名称)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52187142/
如何使用 SPListCollection.Add(String, String, String, String, Int32, String, SPListTemplate.QuickLaunchO
我刚刚开始使用 C++ 并且对 C# 有一些经验,所以我有一些一般的编程经验。然而,似乎我马上就被击落了。我试过在谷歌上寻找,以免浪费任何人的时间,但没有结果。 int main(int argc,
这个问题已经有答案了: In Java 8 how do I transform a Map to another Map using a lambda? (8 个回答) Convert a Map>
我正在使用 node + typescript 和集成的 swagger 进行 API 调用。我 Swagger 提出以下要求 http://localhost:3033/employees/sear
我是 C++ 容器模板的新手。我收集了一些记录。每条记录都有一个唯一的名称,以及一个字段/值对列表。将按名称访问记录。字段/值对的顺序很重要。因此我设计如下: typedef string
我需要这两种方法,但j2me没有,我找到了一个replaceall();但这是 replaceall(string,string,string); 第二个方法是SringBuffer但在j2me中它没
If string is an alias of String in the .net framework为什么会发生这种情况,我应该如何解释它: type JustAString = string
我有两个列表(或字符串):一个大,另一个小。 我想检查较大的(A)是否包含小的(B)。 我的期望如下: 案例 1. B 是 A 的子集 A = [1,2,3] B = [1,2] contains(A
我有一个似乎无法解决的小问题。 这里...我有一个像这样创建的输入... var input = $(''); 如果我这样做......一切都很好 $(this).append(input); 如果我
我有以下代码片段 string[] lines = objects.Split(new string[] { "\r\n", "\n" }, StringSplitOptions.No
这可能真的很简单,但我已经坚持了一段时间了。 我正在尝试输出一个字符串,然后输出一个带有两位小数的 double ,后跟另一个字符串,这是我的代码。 System.out.printf("成本:%.2
以下是 Cloud Firestore 列表查询中的示例之一 citiesRef.where("state", ">=", "CA").where("state", "= 字符串,我们在Stack O
我正在尝试检查一个字符串是否包含在另一个字符串中。后面的代码非常简单。我怎样才能在 jquery 中做到这一点? function deleteRow(locName, locID) { if
这个问题在这里已经有了答案: How to implement big int in C++ (14 个答案) 关闭 9 年前。 我有 2 个字符串,都只包含数字。这些数字大于 uint64_t 的
我有一个带有自定义转换器的 Dozer 映射: com.xyz.Customer com.xyz.CustomerDAO customerName
这个问题在这里已经有了答案: How do I compare strings in Java? (23 个回答) 关闭 6 年前。 我想了解字符串池的工作原理以及一个字符串等于另一个字符串的规则是
我已阅读 this问题和其他一些问题。但它们与我的问题有些无关 对于 UILabel 如果你不指定 ? 或 ! 你会得到这样的错误: @IBOutlet property has non-option
这两种方法中哪一种在理论上更快,为什么? (指向字符串的指针必须是常量。) destination[count] 和 *destination++ 之间的确切区别是什么? destination[co
This question already has answers here: Closed 11 years ago. Possible Duplicates: Is String.Format a
我有一个Stream一个文件的,现在我想将相同的单词组合成 Map这很重要,这个词在 Stream 中出现的频率. 我知道我必须使用 collect(Collectors.groupingBy(..)
我是一名优秀的程序员,十分优秀!