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

javascript - Replace all object values recursively - Stack Overflow

programmeradmin2浏览0评论

I am using angular 4 for one of my projects and i am in a situation where i have to replace all the {data} with {{data}}. ie if i have an object like this

{
    key: {
             key1: {
                       key2: "This is data {someData}",
                       key3: "This is data2 {someData2}"
                   }
         },
    key2: "This is data3 {someData3}"
}

the result should be

{
    key: {
             key1: {
                       key2: "This is data {{someData}}",
                       key3: "This is data2 {{someData2}}"
                   }
         },
    key2: "This is data3 {{someData3}}"
}

I am not so fortable with recursion as on how it works . So i am not able to show anything that i have tried. Please help. Any help would be appreciated.

I am using angular 4 for one of my projects and i am in a situation where i have to replace all the {data} with {{data}}. ie if i have an object like this

{
    key: {
             key1: {
                       key2: "This is data {someData}",
                       key3: "This is data2 {someData2}"
                   }
         },
    key2: "This is data3 {someData3}"
}

the result should be

{
    key: {
             key1: {
                       key2: "This is data {{someData}}",
                       key3: "This is data2 {{someData2}}"
                   }
         },
    key2: "This is data3 {{someData3}}"
}

I am not so fortable with recursion as on how it works . So i am not able to show anything that i have tried. Please help. Any help would be appreciated.

Share Improve this question edited Apr 22, 2018 at 4:41 31piy 23.9k6 gold badges51 silver badges68 bronze badges asked Apr 22, 2018 at 4:26 Vikhyath MaiyaVikhyath Maiya 3,2123 gold badges37 silver badges71 bronze badges
Add a ment  | 

2 Answers 2

Reset to default 8

Hopefully you can try something like this.

var orgObj = {
  key: {
    key1: {
      key2: "This is data {someData}",
      key3: "This is data2 {someData2}"
    }
  },
  key2: "This is data3 {someData3}"
}

function changeValue(obj) {
  if (typeof obj === 'object') {
    // iterating over the object using for..in
    for (var keys in obj) {
      //checking if the current value is an object itself
      if (typeof obj[keys] === 'object') {
        // if so then again calling the same function
        changeValue(obj[keys])
      } else {
        // else getting the value and replacing single { with {{ and so on
        let keyValue = obj[keys].replace('{', '{{').replace('}', '}}');
        obj[keys] = keyValue;
      }
    }
  }
  return obj;
}
console.log(changeValue(orgObj))

Answer by @brk is good but it is mutates the input object. If someone is working with a library like React where it is important to not mutate props, then we will have to do some additional things. We can either create new reference for all the objects that we find during recursive traversal. For this, we have to take care of null values since typeof null is also object

//typeof null is also object. But we cannot destructure it. So check that object is not null
const isValidObject = (obj) => typeof obj === 'object' && obj !== null

function changeValue(objFromProp) {
    let obj = objFromProp;

    if (isValidObject(objFromProp)) {
        //desctructure the object to create new reference
        obj = { ...objFromProp }
        // iterating over the object using for..in
        for (var keys in obj) {
            //checking if the current value is an object itself
            if (isValidObject(obj[keys])) {
                // if so then again calling the same function
                obj[keys] = changeValue(obj[keys])
            } else {
                // else getting the value and replacing single { with {{ and so on
                let keyValue = obj[keys].replace('{', '{{').replace('}', '}}');
                obj[keys] = keyValue;
            }
        }
    }
    return obj;
}

Or if we are replacing values conditionally and if we want to preserve object's reference if none of the values in an object is changed, then we can use another library like immer. Immer will change reference of only those objects whose at-least one field is changed. This will be a useful optimization. For this method, we should use version of changeValue provided by @brk. Immer will identify mutations and change necessary references.

import { produce } from 'immer';
const nextState = produce(objFromProp, draftState => {
    changeValue(draftState)
})
发布评论

评论列表(0)

  1. 暂无评论