admin管理员组文章数量:1341401
I have an array of objects. Each object has a lot of keys (more than 100) and some of these keys can have special chars that I would like to remove.
I try to do what I want in this way:
const result = data.map(datum => {
const keys = Object.keys(datum)
const replacedKeys = keys.map(key => {
const newKey = key.replace(/[.|&;$%@%"<>+]/g, '')
})
// ??
})
But I'm sure it's not the right way..
I have an array of objects. Each object has a lot of keys (more than 100) and some of these keys can have special chars that I would like to remove.
I try to do what I want in this way:
const result = data.map(datum => {
const keys = Object.keys(datum)
const replacedKeys = keys.map(key => {
const newKey = key.replace(/[.|&;$%@%"<>+]/g, '')
})
// ??
})
But I'm sure it's not the right way..
Share edited Feb 9, 2019 at 13:29 John Conde 220k99 gold badges462 silver badges501 bronze badges asked Feb 9, 2019 at 9:57 user9944766user99447666 Answers
Reset to default 5You could map new object with a new key and create a single object with Object.assign
.
const result = data.map(datum => Object.assign(...Object
.keys(datum)
.map(key => ({ [key.replace(/[.|&;$%@%"<>+]/g, '')]: datum[key] }))
));
With the ES8 Object.fromEntries
method that has already found its way in FireFox, you can do:
const sanitiseKeys = o => Object.fromEntries(Object.entries(o).map(([k,v]) =>
[k.replace(/[.|&;$%@%"<>+]/g,""), v]));
// Example use:
var data = [{ "name#": "John" }, { "@key": 2 }];
data = data.map(sanitiseKeys);
console.log(data);
If not yet implemented, here is a polyfill:
Object.fromEntries = arr => Object.assign({}, ...arr.map( ([k, v]) => ({[k]: v}) ));
This solution relies on String.prototype.replace()
, so it can take String
s or RegExp
s as the source, and allows for subsitution. Keep in mind that it's not very performant, but it uses only pure functions:
const data = {
someKey: 1,
some0Key: 1,
some1Key: 1,
some2Key: 1,
some3Key: 1,
some4Key: 1,
some5Key: 1,
some6Key: 1,
some7Key: 1,
some8Key: 1,
some9Key: 1,
some10Key: 1,
some11Key: 1,
some12Key: 1,
some13Key: 1,
some14Key: 1,
some15Key: 1,
};
// simple equivalent of proposed Object.fromEntries()
const fromEntries = (entries) =>
entries.reduce((obj, [key, value]) => ({
[key]: value,
...obj
}), {});
const replaceObjectKeys = (obj, from, to) =>
fromEntries(
Object.entries(obj)
.map(([key, value]) => [key.replace(from, to), value]));
console.log(replaceObjectKeys(data, /Key$/, 'prop'));
fromEntries
could be easily rewritten into a faster implementation at the cost of introducing mutable variables.
You can convert the plain JavaScript object to JSON
using JSON.stringify()
and match the property of valid JSON
using String.prototype.replace()
, then convert back to a plain JavaScript object using JSON.parse()
.
Removed "
from character class as valid JSON
property is surrounded by double quotes "
.
The RegExp
([.|&;$%@%<>+]+)(?=([^\1]+|)":)
creates a capture group containing the character class and matches the character class followed by one or more characters not in the character class followed by closing double quote of the property name "
followed by colon character or double quotes followed by colon character.
The matched character class can be replaced with an empty string ''
or any other character.
let o = {"a.B|c&D;0$_%@q%<Z>5+":1};
console.log(o);
o = JSON.parse(JSON.stringify(o).replace(/([.|&;$%@%<>+]+)(?=([^\1]+|)":)/g, ''));
console.log(
JSON.stringify(o)
, /[.|&;$%@%<>+]+/.test(Object.keys(o)[0]) // false
);
Consider using Array#reduce()
to aggregate values for keys of your input object. The reason for this is that your key sanitization (ie removing unwanted characters from keys) may cause distinct key/value pairs of your input object to "reduce", so that a sanitized key effectively relates to multiple values. For instance an input object like:
const data = {
'key' : 'value0',
'key&&&' : 'value1',
'key$%<>' : 'value2'
}
would (based on your sanitation) yield an output object with a single key
relating to multiple values:
const data = {
'key' : 'value0', // what about value1, value2 ?
}
To address this, you might consider aggregating values with mon sanitized keys to an array as shown:
const data = {
'key' : 'value0',
'key&&&' : 'value1',
'key$%<>' : 'value2',
'foo' : 'value3'
}
const result = Object.entries(data).reduce((obj, [ key, value ]) => {
const sanitizedKey = key.replace(/[.|&;$%@%"<>+]/g, '');
const objValue = obj[ sanitizedKey ]
/*
Account for conflicting keys after santizing by grouping
values in a nested array
*/
if(objValue) {
obj[ sanitizedKey ] = [value].concat(objValue)
}
else {
obj[ sanitizedKey ] = value
}
return obj;
}, {});
console.log(result)
To extend @Nina's answer, here is the full representation:
data = [
{someKey: 1},
{some0Key: 1},
{some1Key: 1,
some2Key: 1},
{some3Key: 1,
some4Key: 1,
some5Key: 1,
some6Key: 1,
some7Key: 1,
some8Key: 1,
some9Key: 1,
some10Key: 1,
},
{some11Key: 1,
some12Key: 1,
some13Key: 1,
some14Key: 1,
some15Key: 1,}
];
result = data.map(datum => Object.assign(...Object
.keys(datum)
.map(key => ({ [key.replace(/some/g, 'bum')]: datum[key] }))
));
results:
result === [
{bumKey: 1},
{bum0Key: 1},
{bum1Key: 1,
bum2Key: 1},
{bum3Key: 1,
bum4Key: 1,
bum5Key: 1,
bum6Key: 1,
bum7Key: 1,
bum8Key: 1,
bum9Key: 1,
bum10Key: 1,
},
{bum11Key: 1,
bum12Key: 1,
bum13Key: 1,
bum14Key: 1,
bum15Key: 1,}
];
本文标签: javascriptUse regex to rename keys of array of objectsStack Overflow
版权声明:本文标题:javascript - Use regex to rename keys of array of objects - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743673818a2519974.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论