This question is about the best practices to handle this pointer problem I’ve dug myself into.
I have an array of structures that is dynamically generated in a function that reads a csv.
int init_from_csv(instance **instances,char *path) {
... open file, get line count
*instances = (instance*) malloc( (size_t) sizeof(instance) * line_count );
... parse and set values of all instances
return count_of_valid_instances_read;
}
// in main()
instance *instances;
int ins_len = init_from_csv(&instances, "some/path/file.csv");
Now, I have to perform functions on this raw data, split it, and perform the same functions again on the splits. This data set can be fairly large so I do not want to duplicate the instances, I just want an array of pointers to structs that are in the split.
instance **split = (instance**) malloc (sizeof(instance*) * split_len_max);
int split_function(instance *instances, ins_len, instances **split){
int i, c;
c = 0;
for (i = 0; i < ins_len; i++) {
if (some_criteria_is_true) {
split[c++] = &instances[i];
}
return c;
}
Now my question what would be the best practice or most readable way to perform a function on both the array of structs and the array of pointers? For a simple example count_data().
int count_data (intances **ins, ins_len, float crit) {
int i,c;
c = 0;
for (i = 0; i < ins_len; i++) {
if ins[i]->data > crit) {
++c;
}
}
return c;
}
// code smell-o-vision going off by now
int c1 = count_data (split, ins_len, 0.05); // works
int c2 = count_data (&instances, ins_len, 0.05); // obviously seg faults
I could make my init_from_csv malloc an array of pointers to instances, and then malloc my array of instances. I want to learn how a seasoned c programmer would handle this sort of thing though before I start changing a bunch of code.
This might seem a bit grungey, but if you really want to pass that instances** pointer around and want it to work for both the main data set and the splits, you really need to make an array of pointers for the main data set too. Here’s one way you could do it…
Now you can always operate on an array of
instance*pointers, whether it’s your main list or a split. I know you didn’t want to use extra memory, but if yourinstancestruct is not trivially small, then allocating an extra pointer for each instance to prevent confusing code duplication is a good idea.When you’re done with your main instance list, you can do this:
I would be tempted to implement this as a struct:
That way, you can return this from your functions in a nicer way:
Now, your function that loads from CSV doesn’t have to focus on the details of creating this monster, so it can simply do the task it’s supposed to do. You can now return lists from other functions, whether they contain the data or simply point into other lists.
etc… Well, you get the idea. No guarantees this code will compile or work, but it should be close. I think it’s important, whenever you’re doing something with arrays that’s even slightly more complicated than just a simple array, it’s useful to make that tiny extra effort to encapsulate it. This is the major data structure you’ll be working with for your analysis or whatever, so it makes sense to give it a little bit of stature in that it has its own data type.
I dunno, was that overkill? =)