Extract specific positions from a char array
I have two arrays:
char input = "ughIuytLikeretC";
and
bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
My function takes these two arrays and returns the characters in input whose positions are true in mask such that in this example, the result being ILikeC.
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
char *filtArray(char input, bool mask, char *filtered) {
int j = 0;
int i;
for (i = 0; input[i]; i++) {
filtered[j] = input[i];
j += mask[i];
}
filtered[j] = 0;
return filtered;
}
filtArray
will run on billions of "input" strings of constant length and "mask" will be the same for all "input"s.
performance beginner c strings
New contributor
add a comment |
I have two arrays:
char input = "ughIuytLikeretC";
and
bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
My function takes these two arrays and returns the characters in input whose positions are true in mask such that in this example, the result being ILikeC.
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
char *filtArray(char input, bool mask, char *filtered) {
int j = 0;
int i;
for (i = 0; input[i]; i++) {
filtered[j] = input[i];
j += mask[i];
}
filtered[j] = 0;
return filtered;
}
filtArray
will run on billions of "input" strings of constant length and "mask" will be the same for all "input"s.
performance beginner c strings
New contributor
It's bad practice to write data to a result array that shouldn't be there. You write the contents tofiltered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.
– Lundin
11 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Is it a hard requirement thatmask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.
– Reinderien
9 hours ago
add a comment |
I have two arrays:
char input = "ughIuytLikeretC";
and
bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
My function takes these two arrays and returns the characters in input whose positions are true in mask such that in this example, the result being ILikeC.
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
char *filtArray(char input, bool mask, char *filtered) {
int j = 0;
int i;
for (i = 0; input[i]; i++) {
filtered[j] = input[i];
j += mask[i];
}
filtered[j] = 0;
return filtered;
}
filtArray
will run on billions of "input" strings of constant length and "mask" will be the same for all "input"s.
performance beginner c strings
New contributor
I have two arrays:
char input = "ughIuytLikeretC";
and
bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
My function takes these two arrays and returns the characters in input whose positions are true in mask such that in this example, the result being ILikeC.
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
char *filtArray(char input, bool mask, char *filtered) {
int j = 0;
int i;
for (i = 0; input[i]; i++) {
filtered[j] = input[i];
j += mask[i];
}
filtered[j] = 0;
return filtered;
}
filtArray
will run on billions of "input" strings of constant length and "mask" will be the same for all "input"s.
performance beginner c strings
performance beginner c strings
New contributor
New contributor
edited 21 mins ago
Jamal♦
30.2k11116226
30.2k11116226
New contributor
asked 13 hours ago
jregalad
313
313
New contributor
New contributor
It's bad practice to write data to a result array that shouldn't be there. You write the contents tofiltered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.
– Lundin
11 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Is it a hard requirement thatmask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.
– Reinderien
9 hours ago
add a comment |
It's bad practice to write data to a result array that shouldn't be there. You write the contents tofiltered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.
– Lundin
11 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Is it a hard requirement thatmask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.
– Reinderien
9 hours ago
It's bad practice to write data to a result array that shouldn't be there. You write the contents to
filtered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.– Lundin
11 hours ago
It's bad practice to write data to a result array that shouldn't be there. You write the contents to
filtered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.– Lundin
11 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Is it a hard requirement that
mask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.– Reinderien
9 hours ago
Is it a hard requirement that
mask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.– Reinderien
9 hours ago
add a comment |
2 Answers
2
active
oldest
votes
I see some things that may help you improve your code.
Provide complete code to reviewers
This is not so much a change to the code as a change in how you present it to other people. Without the full context of the code and an example of how to use it, it takes more effort for other people to understand your code. This affects not only code reviews, but also maintenance of the code in the future, by you or by others. One good way to address that is by the use of comments. Another good technique is to include test code showing how your code is intended to be used. Here's the test main
I used for your code:
int main() {
const char *input[2] = {
"ughIuytLikeretC",
"xxxExxxdwarxxxd",
};
const bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
char filt[100];
char maskstr[100];
// create the mask string
pmask(mask, maskstr);
printf("Orig: %snMask: %snFilt: %sn", input[0], maskstr, filtArray(input[0], mask, filt));
printf("Orig: %snMask: %snFilt: %sn", input[1], maskstr, filtArray(input[1], mask, filt));
for (int i = 0; i < 10000000; ++i) {
int n = rand() > RAND_MAX/2 ? 1 : 0;
printf("Orig: %snMask: %snFilt: %sn", input[n], maskstr, filtArray(input[n], mask, filt));
}
}
After it applies the function to two strings, it then iterates 10 million times, choosing one or the other test inputs randomly. This is for testing timing.
Use const
where practical
The filtArray
function does not (and should not) alter either the passed input
or mask
arrays and so both of those should be declared const
.
char *filtArray(const char input, const bool mask, char *filtered) {
Consider bounds checking
If the input strings have already been validated for length, the function you have is OK, but in general, it's good to make sure there is enough room to copy the masked characters. If there isn't enough room, that's the recipe for a buffer overflow vulnerability and must be eliminated, either by the calling routine or by this one.
Consider a custom copy
If the same mask is used for billions of strings, it would probably make sense to do things differently. For example, one alternative might look like this:
#include
char *filtArray(const char input, char *filtered) {
memcpy(&filtered[1], &input[7], 4);
filtered[0] = input[3];
filtered[5] = input[14];
filtered[6] = '';
return filtered;
}
Note that the mask
is no longer used in this version, because the code has implemented it implicitly. This is less flexible but offers better performance. For 10 million strings on my machine, your original version takes about 1.3 seconds, while the version shown here takes around 1.0 seconds (redirecting the output to /dev/null
on a Linux machine).
Use pointers rather than indexing for speed
Pointers are generally a faster way to access elements than using index variables. For example, your filtArray
routine could be written like this:
char *filtArray(const char *input, const bool *mask, char *filtered) {
char *beginning = filtered;
for ( ; *input; ++input, ++mask) {
if (mask) {
*filtered++ = *input;
}
}
*filtered = '';
return beginning;
}
Because you're just beginning, this may seem strange to you, but this kind of use of pointers is a very common idiom in C.
Compilers are good, but not quite that good yet
Because there's a tendency to assume the compiler will take care of it, here's compiler output comparison of the two approaches using gcc for ARM using the on-line compiler explorer: https://godbolt.org/z/H_a_o9
As can be seen in this case, the generated assembly code for the pointer version is much shorter. Shorter code is usually faster (and it is in this case according to my testing) but not always. For those who are expert in compiler design: The typical improvement is as likely to be the elimination of extra live variables as for the use of pointers per se, but the effect is nonetheless real.
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
add a comment |
The code compiled with no warnings and ran properly on first time which is great.
Let's see what can be improved.
Style
The indentation of the code seems a bit weird. I do not know if this is how it looked originally or if it got broken when it was copied here.
Also, it may be worth adding some documentation describing the inputs you are expecting (in your case, 3 arrays of same size, the first one being 0-terminated).
Do less
You use the mask only to know whether j
is to be incremented. Actually, you could rewrite:
j += mask[i];
as
if (mask[i])
j++;
which is more explicit but less concise.
The real benefic is when you realize than updating filtered
can be done only when we have mask[i]
. We can write:
if (mask[i])
{
filtered[j] = input[i];
j++;
}
or the equivalent:
if (mask[i])
filtered[j++] = input[i];
Null character
Instead of filtered[j] = 0;
, you could use the Null Character which is equivalent here but more usual and write: filtered[j] = '';
.
Signature
I am not sure if it is really useful to have the filtered
value returned as it is already known by the calling function. Also, filterArray
may be a better name.
Going further
Instead of definining a mask as an array of boolean, you could provide an array with the positions of the characters you are interested in.
In your case, you'd provide something like: {3, 7, 8, 9, 10, 14 }
.
This could be less efficient because we'd perform a smaller number of iterations. Here, we'd iterate over 6 elements instead of 15.
The corresponding mask could be converted manually (which is what I did here) if it is for a known value or you could write a function to pre-process the mask. This seems to be relevant in your case as the same mask is used many times on different inputs.
add a comment |
Your Answer
StackExchange.ifUsing("editor", function () {
return StackExchange.using("mathjaxEditing", function () {
StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
});
});
}, "mathjax-editing");
StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "196"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
jregalad is a new contributor. Be nice, and check out our Code of Conduct.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210044%2fextract-specific-positions-from-a-char-array%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
2 Answers
2
active
oldest
votes
2 Answers
2
active
oldest
votes
active
oldest
votes
active
oldest
votes
I see some things that may help you improve your code.
Provide complete code to reviewers
This is not so much a change to the code as a change in how you present it to other people. Without the full context of the code and an example of how to use it, it takes more effort for other people to understand your code. This affects not only code reviews, but also maintenance of the code in the future, by you or by others. One good way to address that is by the use of comments. Another good technique is to include test code showing how your code is intended to be used. Here's the test main
I used for your code:
int main() {
const char *input[2] = {
"ughIuytLikeretC",
"xxxExxxdwarxxxd",
};
const bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
char filt[100];
char maskstr[100];
// create the mask string
pmask(mask, maskstr);
printf("Orig: %snMask: %snFilt: %sn", input[0], maskstr, filtArray(input[0], mask, filt));
printf("Orig: %snMask: %snFilt: %sn", input[1], maskstr, filtArray(input[1], mask, filt));
for (int i = 0; i < 10000000; ++i) {
int n = rand() > RAND_MAX/2 ? 1 : 0;
printf("Orig: %snMask: %snFilt: %sn", input[n], maskstr, filtArray(input[n], mask, filt));
}
}
After it applies the function to two strings, it then iterates 10 million times, choosing one or the other test inputs randomly. This is for testing timing.
Use const
where practical
The filtArray
function does not (and should not) alter either the passed input
or mask
arrays and so both of those should be declared const
.
char *filtArray(const char input, const bool mask, char *filtered) {
Consider bounds checking
If the input strings have already been validated for length, the function you have is OK, but in general, it's good to make sure there is enough room to copy the masked characters. If there isn't enough room, that's the recipe for a buffer overflow vulnerability and must be eliminated, either by the calling routine or by this one.
Consider a custom copy
If the same mask is used for billions of strings, it would probably make sense to do things differently. For example, one alternative might look like this:
#include
char *filtArray(const char input, char *filtered) {
memcpy(&filtered[1], &input[7], 4);
filtered[0] = input[3];
filtered[5] = input[14];
filtered[6] = '';
return filtered;
}
Note that the mask
is no longer used in this version, because the code has implemented it implicitly. This is less flexible but offers better performance. For 10 million strings on my machine, your original version takes about 1.3 seconds, while the version shown here takes around 1.0 seconds (redirecting the output to /dev/null
on a Linux machine).
Use pointers rather than indexing for speed
Pointers are generally a faster way to access elements than using index variables. For example, your filtArray
routine could be written like this:
char *filtArray(const char *input, const bool *mask, char *filtered) {
char *beginning = filtered;
for ( ; *input; ++input, ++mask) {
if (mask) {
*filtered++ = *input;
}
}
*filtered = '';
return beginning;
}
Because you're just beginning, this may seem strange to you, but this kind of use of pointers is a very common idiom in C.
Compilers are good, but not quite that good yet
Because there's a tendency to assume the compiler will take care of it, here's compiler output comparison of the two approaches using gcc for ARM using the on-line compiler explorer: https://godbolt.org/z/H_a_o9
As can be seen in this case, the generated assembly code for the pointer version is much shorter. Shorter code is usually faster (and it is in this case according to my testing) but not always. For those who are expert in compiler design: The typical improvement is as likely to be the elimination of extra live variables as for the use of pointers per se, but the effect is nonetheless real.
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
add a comment |
I see some things that may help you improve your code.
Provide complete code to reviewers
This is not so much a change to the code as a change in how you present it to other people. Without the full context of the code and an example of how to use it, it takes more effort for other people to understand your code. This affects not only code reviews, but also maintenance of the code in the future, by you or by others. One good way to address that is by the use of comments. Another good technique is to include test code showing how your code is intended to be used. Here's the test main
I used for your code:
int main() {
const char *input[2] = {
"ughIuytLikeretC",
"xxxExxxdwarxxxd",
};
const bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
char filt[100];
char maskstr[100];
// create the mask string
pmask(mask, maskstr);
printf("Orig: %snMask: %snFilt: %sn", input[0], maskstr, filtArray(input[0], mask, filt));
printf("Orig: %snMask: %snFilt: %sn", input[1], maskstr, filtArray(input[1], mask, filt));
for (int i = 0; i < 10000000; ++i) {
int n = rand() > RAND_MAX/2 ? 1 : 0;
printf("Orig: %snMask: %snFilt: %sn", input[n], maskstr, filtArray(input[n], mask, filt));
}
}
After it applies the function to two strings, it then iterates 10 million times, choosing one or the other test inputs randomly. This is for testing timing.
Use const
where practical
The filtArray
function does not (and should not) alter either the passed input
or mask
arrays and so both of those should be declared const
.
char *filtArray(const char input, const bool mask, char *filtered) {
Consider bounds checking
If the input strings have already been validated for length, the function you have is OK, but in general, it's good to make sure there is enough room to copy the masked characters. If there isn't enough room, that's the recipe for a buffer overflow vulnerability and must be eliminated, either by the calling routine or by this one.
Consider a custom copy
If the same mask is used for billions of strings, it would probably make sense to do things differently. For example, one alternative might look like this:
#include
char *filtArray(const char input, char *filtered) {
memcpy(&filtered[1], &input[7], 4);
filtered[0] = input[3];
filtered[5] = input[14];
filtered[6] = '';
return filtered;
}
Note that the mask
is no longer used in this version, because the code has implemented it implicitly. This is less flexible but offers better performance. For 10 million strings on my machine, your original version takes about 1.3 seconds, while the version shown here takes around 1.0 seconds (redirecting the output to /dev/null
on a Linux machine).
Use pointers rather than indexing for speed
Pointers are generally a faster way to access elements than using index variables. For example, your filtArray
routine could be written like this:
char *filtArray(const char *input, const bool *mask, char *filtered) {
char *beginning = filtered;
for ( ; *input; ++input, ++mask) {
if (mask) {
*filtered++ = *input;
}
}
*filtered = '';
return beginning;
}
Because you're just beginning, this may seem strange to you, but this kind of use of pointers is a very common idiom in C.
Compilers are good, but not quite that good yet
Because there's a tendency to assume the compiler will take care of it, here's compiler output comparison of the two approaches using gcc for ARM using the on-line compiler explorer: https://godbolt.org/z/H_a_o9
As can be seen in this case, the generated assembly code for the pointer version is much shorter. Shorter code is usually faster (and it is in this case according to my testing) but not always. For those who are expert in compiler design: The typical improvement is as likely to be the elimination of extra live variables as for the use of pointers per se, but the effect is nonetheless real.
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
add a comment |
I see some things that may help you improve your code.
Provide complete code to reviewers
This is not so much a change to the code as a change in how you present it to other people. Without the full context of the code and an example of how to use it, it takes more effort for other people to understand your code. This affects not only code reviews, but also maintenance of the code in the future, by you or by others. One good way to address that is by the use of comments. Another good technique is to include test code showing how your code is intended to be used. Here's the test main
I used for your code:
int main() {
const char *input[2] = {
"ughIuytLikeretC",
"xxxExxxdwarxxxd",
};
const bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
char filt[100];
char maskstr[100];
// create the mask string
pmask(mask, maskstr);
printf("Orig: %snMask: %snFilt: %sn", input[0], maskstr, filtArray(input[0], mask, filt));
printf("Orig: %snMask: %snFilt: %sn", input[1], maskstr, filtArray(input[1], mask, filt));
for (int i = 0; i < 10000000; ++i) {
int n = rand() > RAND_MAX/2 ? 1 : 0;
printf("Orig: %snMask: %snFilt: %sn", input[n], maskstr, filtArray(input[n], mask, filt));
}
}
After it applies the function to two strings, it then iterates 10 million times, choosing one or the other test inputs randomly. This is for testing timing.
Use const
where practical
The filtArray
function does not (and should not) alter either the passed input
or mask
arrays and so both of those should be declared const
.
char *filtArray(const char input, const bool mask, char *filtered) {
Consider bounds checking
If the input strings have already been validated for length, the function you have is OK, but in general, it's good to make sure there is enough room to copy the masked characters. If there isn't enough room, that's the recipe for a buffer overflow vulnerability and must be eliminated, either by the calling routine or by this one.
Consider a custom copy
If the same mask is used for billions of strings, it would probably make sense to do things differently. For example, one alternative might look like this:
#include
char *filtArray(const char input, char *filtered) {
memcpy(&filtered[1], &input[7], 4);
filtered[0] = input[3];
filtered[5] = input[14];
filtered[6] = '';
return filtered;
}
Note that the mask
is no longer used in this version, because the code has implemented it implicitly. This is less flexible but offers better performance. For 10 million strings on my machine, your original version takes about 1.3 seconds, while the version shown here takes around 1.0 seconds (redirecting the output to /dev/null
on a Linux machine).
Use pointers rather than indexing for speed
Pointers are generally a faster way to access elements than using index variables. For example, your filtArray
routine could be written like this:
char *filtArray(const char *input, const bool *mask, char *filtered) {
char *beginning = filtered;
for ( ; *input; ++input, ++mask) {
if (mask) {
*filtered++ = *input;
}
}
*filtered = '';
return beginning;
}
Because you're just beginning, this may seem strange to you, but this kind of use of pointers is a very common idiom in C.
Compilers are good, but not quite that good yet
Because there's a tendency to assume the compiler will take care of it, here's compiler output comparison of the two approaches using gcc for ARM using the on-line compiler explorer: https://godbolt.org/z/H_a_o9
As can be seen in this case, the generated assembly code for the pointer version is much shorter. Shorter code is usually faster (and it is in this case according to my testing) but not always. For those who are expert in compiler design: The typical improvement is as likely to be the elimination of extra live variables as for the use of pointers per se, but the effect is nonetheless real.
I see some things that may help you improve your code.
Provide complete code to reviewers
This is not so much a change to the code as a change in how you present it to other people. Without the full context of the code and an example of how to use it, it takes more effort for other people to understand your code. This affects not only code reviews, but also maintenance of the code in the future, by you or by others. One good way to address that is by the use of comments. Another good technique is to include test code showing how your code is intended to be used. Here's the test main
I used for your code:
int main() {
const char *input[2] = {
"ughIuytLikeretC",
"xxxExxxdwarxxxd",
};
const bool mask = {
false, false, false, true, false,
false, false, true, true, true,
true, false, false, false, true,
};
char filt[100];
char maskstr[100];
// create the mask string
pmask(mask, maskstr);
printf("Orig: %snMask: %snFilt: %sn", input[0], maskstr, filtArray(input[0], mask, filt));
printf("Orig: %snMask: %snFilt: %sn", input[1], maskstr, filtArray(input[1], mask, filt));
for (int i = 0; i < 10000000; ++i) {
int n = rand() > RAND_MAX/2 ? 1 : 0;
printf("Orig: %snMask: %snFilt: %sn", input[n], maskstr, filtArray(input[n], mask, filt));
}
}
After it applies the function to two strings, it then iterates 10 million times, choosing one or the other test inputs randomly. This is for testing timing.
Use const
where practical
The filtArray
function does not (and should not) alter either the passed input
or mask
arrays and so both of those should be declared const
.
char *filtArray(const char input, const bool mask, char *filtered) {
Consider bounds checking
If the input strings have already been validated for length, the function you have is OK, but in general, it's good to make sure there is enough room to copy the masked characters. If there isn't enough room, that's the recipe for a buffer overflow vulnerability and must be eliminated, either by the calling routine or by this one.
Consider a custom copy
If the same mask is used for billions of strings, it would probably make sense to do things differently. For example, one alternative might look like this:
#include
char *filtArray(const char input, char *filtered) {
memcpy(&filtered[1], &input[7], 4);
filtered[0] = input[3];
filtered[5] = input[14];
filtered[6] = '';
return filtered;
}
Note that the mask
is no longer used in this version, because the code has implemented it implicitly. This is less flexible but offers better performance. For 10 million strings on my machine, your original version takes about 1.3 seconds, while the version shown here takes around 1.0 seconds (redirecting the output to /dev/null
on a Linux machine).
Use pointers rather than indexing for speed
Pointers are generally a faster way to access elements than using index variables. For example, your filtArray
routine could be written like this:
char *filtArray(const char *input, const bool *mask, char *filtered) {
char *beginning = filtered;
for ( ; *input; ++input, ++mask) {
if (mask) {
*filtered++ = *input;
}
}
*filtered = '';
return beginning;
}
Because you're just beginning, this may seem strange to you, but this kind of use of pointers is a very common idiom in C.
Compilers are good, but not quite that good yet
Because there's a tendency to assume the compiler will take care of it, here's compiler output comparison of the two approaches using gcc for ARM using the on-line compiler explorer: https://godbolt.org/z/H_a_o9
As can be seen in this case, the generated assembly code for the pointer version is much shorter. Shorter code is usually faster (and it is in this case according to my testing) but not always. For those who are expert in compiler design: The typical improvement is as likely to be the elimination of extra live variables as for the use of pointers per se, but the effect is nonetheless real.
edited 9 hours ago
answered 12 hours ago
Edward
45.8k377208
45.8k377208
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
add a comment |
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
4
4
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
"Pointers are generally a faster way to access elements than using index variables." This is very subjective. The opposite may as well be true, depending on system. We should not replace indexing with pointer arithmetic unless we have very good reasons - doing so for the sake of performance is pre-mature optimization. To truly optimize for speed, it might be better to drop the bool array in favour of true bit masks.
– Lundin
11 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
It's not actually subjective, but based on measurement and experience. On my machine it's faster, and for many embedded systems compilers (which is what I use often) it's generally faster. But what matters is whether it's faster for the author of the code. Only measurement on that system and with real data will tell whether it's faster or not.
– Edward
10 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
I've added more data and explanation to show why pointers are faster with this code.
– Edward
9 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
The resulting assembly is still very dependent on compiler as well. Switching to using the intel compiler on godbolt results in indexing generating shorter assembly.
– pseudonym117
7 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
@pseudonym117 At the risk of stating the obvious, the output assembly is always dependent on the compiler. In the example you mentioned, while there are fewer instructions, the assembled code is both longer and slower with the indexed version vs the pointer version. (42 bytes vs. 38 bytes for pointer version). godbolt.org/z/Sr1Bsq It's another example of why we must measure rather than guess.
– Edward
6 hours ago
add a comment |
The code compiled with no warnings and ran properly on first time which is great.
Let's see what can be improved.
Style
The indentation of the code seems a bit weird. I do not know if this is how it looked originally or if it got broken when it was copied here.
Also, it may be worth adding some documentation describing the inputs you are expecting (in your case, 3 arrays of same size, the first one being 0-terminated).
Do less
You use the mask only to know whether j
is to be incremented. Actually, you could rewrite:
j += mask[i];
as
if (mask[i])
j++;
which is more explicit but less concise.
The real benefic is when you realize than updating filtered
can be done only when we have mask[i]
. We can write:
if (mask[i])
{
filtered[j] = input[i];
j++;
}
or the equivalent:
if (mask[i])
filtered[j++] = input[i];
Null character
Instead of filtered[j] = 0;
, you could use the Null Character which is equivalent here but more usual and write: filtered[j] = '';
.
Signature
I am not sure if it is really useful to have the filtered
value returned as it is already known by the calling function. Also, filterArray
may be a better name.
Going further
Instead of definining a mask as an array of boolean, you could provide an array with the positions of the characters you are interested in.
In your case, you'd provide something like: {3, 7, 8, 9, 10, 14 }
.
This could be less efficient because we'd perform a smaller number of iterations. Here, we'd iterate over 6 elements instead of 15.
The corresponding mask could be converted manually (which is what I did here) if it is for a known value or you could write a function to pre-process the mask. This seems to be relevant in your case as the same mask is used many times on different inputs.
add a comment |
The code compiled with no warnings and ran properly on first time which is great.
Let's see what can be improved.
Style
The indentation of the code seems a bit weird. I do not know if this is how it looked originally or if it got broken when it was copied here.
Also, it may be worth adding some documentation describing the inputs you are expecting (in your case, 3 arrays of same size, the first one being 0-terminated).
Do less
You use the mask only to know whether j
is to be incremented. Actually, you could rewrite:
j += mask[i];
as
if (mask[i])
j++;
which is more explicit but less concise.
The real benefic is when you realize than updating filtered
can be done only when we have mask[i]
. We can write:
if (mask[i])
{
filtered[j] = input[i];
j++;
}
or the equivalent:
if (mask[i])
filtered[j++] = input[i];
Null character
Instead of filtered[j] = 0;
, you could use the Null Character which is equivalent here but more usual and write: filtered[j] = '';
.
Signature
I am not sure if it is really useful to have the filtered
value returned as it is already known by the calling function. Also, filterArray
may be a better name.
Going further
Instead of definining a mask as an array of boolean, you could provide an array with the positions of the characters you are interested in.
In your case, you'd provide something like: {3, 7, 8, 9, 10, 14 }
.
This could be less efficient because we'd perform a smaller number of iterations. Here, we'd iterate over 6 elements instead of 15.
The corresponding mask could be converted manually (which is what I did here) if it is for a known value or you could write a function to pre-process the mask. This seems to be relevant in your case as the same mask is used many times on different inputs.
add a comment |
The code compiled with no warnings and ran properly on first time which is great.
Let's see what can be improved.
Style
The indentation of the code seems a bit weird. I do not know if this is how it looked originally or if it got broken when it was copied here.
Also, it may be worth adding some documentation describing the inputs you are expecting (in your case, 3 arrays of same size, the first one being 0-terminated).
Do less
You use the mask only to know whether j
is to be incremented. Actually, you could rewrite:
j += mask[i];
as
if (mask[i])
j++;
which is more explicit but less concise.
The real benefic is when you realize than updating filtered
can be done only when we have mask[i]
. We can write:
if (mask[i])
{
filtered[j] = input[i];
j++;
}
or the equivalent:
if (mask[i])
filtered[j++] = input[i];
Null character
Instead of filtered[j] = 0;
, you could use the Null Character which is equivalent here but more usual and write: filtered[j] = '';
.
Signature
I am not sure if it is really useful to have the filtered
value returned as it is already known by the calling function. Also, filterArray
may be a better name.
Going further
Instead of definining a mask as an array of boolean, you could provide an array with the positions of the characters you are interested in.
In your case, you'd provide something like: {3, 7, 8, 9, 10, 14 }
.
This could be less efficient because we'd perform a smaller number of iterations. Here, we'd iterate over 6 elements instead of 15.
The corresponding mask could be converted manually (which is what I did here) if it is for a known value or you could write a function to pre-process the mask. This seems to be relevant in your case as the same mask is used many times on different inputs.
The code compiled with no warnings and ran properly on first time which is great.
Let's see what can be improved.
Style
The indentation of the code seems a bit weird. I do not know if this is how it looked originally or if it got broken when it was copied here.
Also, it may be worth adding some documentation describing the inputs you are expecting (in your case, 3 arrays of same size, the first one being 0-terminated).
Do less
You use the mask only to know whether j
is to be incremented. Actually, you could rewrite:
j += mask[i];
as
if (mask[i])
j++;
which is more explicit but less concise.
The real benefic is when you realize than updating filtered
can be done only when we have mask[i]
. We can write:
if (mask[i])
{
filtered[j] = input[i];
j++;
}
or the equivalent:
if (mask[i])
filtered[j++] = input[i];
Null character
Instead of filtered[j] = 0;
, you could use the Null Character which is equivalent here but more usual and write: filtered[j] = '';
.
Signature
I am not sure if it is really useful to have the filtered
value returned as it is already known by the calling function. Also, filterArray
may be a better name.
Going further
Instead of definining a mask as an array of boolean, you could provide an array with the positions of the characters you are interested in.
In your case, you'd provide something like: {3, 7, 8, 9, 10, 14 }
.
This could be less efficient because we'd perform a smaller number of iterations. Here, we'd iterate over 6 elements instead of 15.
The corresponding mask could be converted manually (which is what I did here) if it is for a known value or you could write a function to pre-process the mask. This seems to be relevant in your case as the same mask is used many times on different inputs.
answered 12 hours ago
Josay
24.8k13784
24.8k13784
add a comment |
add a comment |
jregalad is a new contributor. Be nice, and check out our Code of Conduct.
jregalad is a new contributor. Be nice, and check out our Code of Conduct.
jregalad is a new contributor. Be nice, and check out our Code of Conduct.
jregalad is a new contributor. Be nice, and check out our Code of Conduct.
Thanks for contributing an answer to Code Review Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
Use MathJax to format equations. MathJax reference.
To learn more, see our tips on writing great answers.
Some of your past answers have not been well-received, and you're in danger of being blocked from answering.
Please pay close attention to the following guidance:
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210044%2fextract-specific-positions-from-a-char-array%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
It's bad practice to write data to a result array that shouldn't be there. You write the contents to
filtered
regardless of if it should be there or not, then overwrite it. Avoid this. Suppose there is no correct data - you will then corrupt the result array. I would also avoid using booleans for arithmetic, it's quite ugly and hard to read.– Lundin
11 hours ago
@Lundin: I'm not clear on what you mean when you say the array "shouldn't be there." Seems to me it should always exist since that result is the point of calling the function in the first place.
– Edward
10 hours ago
Please fix the indentation in your sample code.
– Reinderien
9 hours ago
Is it a hard requirement that
mask
be an array of booleans? Because that's quite inefficient. You really should be using a binary mask in an integer.– Reinderien
9 hours ago