I came up with the following, but I'm not sure if it'll be faster than your method since there's a few more steps involved. Also it's a bit longer...
var toObj = function (str) {
var arr = str.split(","), obj = {};
for (var i = 0; i < arr.length; i++) {
var keyval = arr[i].split("=", 2);
obj[keyval[0]] = keyval[1];
}
return obj;
};
var objToString = function (obj) {
var stringBuilder = [];
for (var k in obj) {
stringBuilder.push(k+"="+obj[k]);
}
return stringBuilder.join(",");
};
var merge = function (obj1, obj2) { // merge obj1 into obj2.. obj2 will be modified.
for (var k in obj1) {
obj2[k] = obj2[k] || obj1[k];
}
return obj2;
};
var newStr = objToString(merge(toObj(str1), toObj(str2));
Benchmarking
I was bored, so I decided to do a quick benchmark of these 3 different solutions offered.
Benchmarking function/test:
function benchmark(name, fn, n) {
console.time(name);
for(var i = 0; i < n; i++)
fn();
console.timeEnd(name);
}
Results:
Andres (OP):
benchmark("Andres", function () { paramStrAndres("width=800,height=600,resizable=no,titlebar=no", "width=100,height=100"); }, 10000);
Andres: 505ms
Ryan/RWT:
benchmark("RWT", function () { paramStrRWT("width=800,height=600,resizable=no,titlebar=no", "width=100,height=100"); }, 10000);
RWT: 86ms
Daniel:
benchmark("Daniel", function () { paramStrDaniel("width=800,height=600,resizable=no,titlebar=no", "width=100,height=100"); }, 10000);
Daniel: 98ms
Sean:
benchmark("Sean", function () { paramStrSean("width=800,height=600,resizable=no,titlebar=no", "width=100,height=100"); }, 10000);
Sean: 40ms
Functions Tested:
Andres:
function paramStrAndres(str1, str2) {
sArray = str2.split(",");
for( var i = 0; i < sArray.length; i++ ) {
var key = sArray[i].match(/(\w+)=/gi).toString().replace("=", ""),
in_str1 = str1.search(key),
replace_pattern = new RegExp(key+"=(\\w+)", "gi");
if(in_str1 !== -1){
str1 = str1.replace(replace_pattern, sArray[i]);
} else {
str1 = str1 + "," + sArray[i];
}
}
return str1;
}
Ryan/RWT:
function paramStrRWT (str1, str2) {
function parseParamString(str, obj) {
var pairs = str.split(","),
i = 0, l = pairs.length,
pair;
obj || (obj = {});
for (; l > i; ++i) {
pair = pairs[i].split("=", 2);
obj[pair[0]] = pair[1];
}
return obj;
}
function buildParamString(obj) {
var pairs = [], i;
for (i in obj) {
if (obj.hasOwnProperty(i)) {
pairs.push(i + "=" + obj[i]);
}
}
return pairs.join(",");
}
var resultObj = parseParamString(str2, parseParamString(str1));
return buildParamString(resultObj);
}
Mine:
function paramStrDaniel(str1, str2) {
var toObj = function (str) {
var arr = str.split(","), obj = {};
for (var i = 0; i < arr.length; i++) {
var keyval = arr[i].split("=", 2);
obj[keyval[0]] = keyval[1];
}
return obj;
};
var objToString = function (obj) {
var stringBuilder = [];
for (var k in obj) {
stringBuilder.push(k+"="+obj[k]);
}
return stringBuilder.join(",");
};
var merge = function (obj1, obj2) { // merge obj1 into obj2.. obj2 will be modified.
for (var k in obj1) {
obj2[k] = obj2[k] || obj1[k];
}
return obj2;
};
return objToString(merge(toObj(str1), toObj(str2)));
}
Sean:
function paramStrSean(str1, str2) {
var str3 = str2 + "," + str1, temp;
while ((temp = str3.replace(/\b([a-z]+)(=.*)\b\1=[^,]*,?/, "$1$2")) != str3) {
str3 = temp;
}
return str3;
}
EDIT: I found it odd that my version was faster than Ryan's, considering they were similar. After looking into it, I see I made a boo-boo on the for loop (length -> arr.length
). I've updated the benchmarks. That's what I get for not testing my code.