Please consider this C++ question:
#include <cstdio>
#include <iostream>
#include <vector>
using namespace std;
class ParseURL{
private:
int qualify; // 0 means we qualify
public:
string node;
string service;
bool primary;
bool health;
string epoch;
// methods
ParseURL(string URL);
~ParseURL();
};
ParseURL::ParseURL(string URL){
vector<string> g2 = {"node", "service", "primary", "health", "epoch"};
for (string tag : g2){
auto found = URL.find(tag);
if ( found != string::npos ){
auto cut_from = found + 1 + tag.size() ;
auto meh = URL.substr(cut_from, URL.substr(cut_from).find("&") );
// is there a way we can avoid these lines below?
if (tag.find("node",0) == 0){
this->node = meh;
} else if (tag.find("service",0) == 0 ){
this->service = meh;
} else if (tag.find("epoch",0) == 0) {
this->epoch = meh;
} else if (tag.find("health",0) == 0){
this->health = (meh == "OK");
} else if (tag.find("primary",0) == 0 ){
this->primary == (this->node == meh);
}
}
}
}
ParseURL::~ParseURL(){
cout << "Tearing Down the class\n";
}
int main(){
char req[] = "GET /register?node=hostrpi3&service=potatoservice&primary=node1&health=OK&epoch=1559345106 HTTP";
string Request = req;
Request = Request.substr(Request.find_first_of(" ") );
Request = Request.substr(0, Request.find(" HTTP"));
ParseURL *a = new ParseURL(Request);
cout.width(12);
cout << "a->node: " << a->node << endl;
cout.width(12);
cout << "a->service: " << a->service << endl;
cout.width(12);
cout << "a->epoch: " << a->epoch << endl;
cout.width(12);
cout << "a->health: " << a->health << endl;
cout.width(12);
cout << "a->primary: " << a->primary << endl;
delete(a);
return 0;
}
I need to represent a object based on URL Query Strings, the tags I need are represented in vector g2, and as you can see, I make datamembers for ParseURL
out of those tags.
The o/p looks like:
a->node: hostrpi3
a->service: potatoservice
a->epoch: 1559345106
a->health: 1
a->primary: 0
Tearing Down the class
Although this version works, it feels this can be improved much. Despite having it all in the Vector, I am repeatedly doing tag.find
for each of those attributes. There must be a better way, I hope. Could you please point in the right direction? Thanks!
Edit 1
Thanks, I updated the constructor following the suggestions:
ParseURL::ParseURL(string URL){
char tags[10][100] = {"node", "service", "primary", "health", "epoch"};
int i = 0;
for (auto tag : tags){
auto found = URL.find(tag);
if ( found != string::npos ){
auto cut_from = found + 1 + strlen(tag);
auto tag_info = URL.substr(cut_from, URL.substr(cut_from).find("&") );
switch (i){
case 0:
this->node = tag_info;
break;
case 1:
this->service = tag_info;
break;
case 2:
this->primary = (this->node == tag_info);
break;
case 3:
this->health = (tag_info == "OK");
break;
case 4:
this->epoch = tag_info;
break;
}
}
i++;
}
}
bash for example, offers indirect reference of variables - that is, one variable may contain the name of another variable:
$ cat indirect.sh
#!/bin/bash
values="value1 value2"
value1="this is value1"
value2="this is value2"
for i in $values; do
echo "$i has value ${!i}"
done
$ ./indirect.sh
value1 has value this is value1
value2 has value this is value2
I was hoping something similar exists for cases like this, nevertheless, I learned valuable lessons here. thank you!