Let's say I have a vector class:
typedef struct vec3_s
{
float x, y, z;
}
vec3;
But, I would like to be able to iterate through it without converting it to an array of floats. While a cast is acceptable in this case, I'm curious to see if anything along the lines of C++ like functionality is doable in straight C. For example, in C++, since std::vector< T >
has the subscript []
operator overloaded, I can pass the address of its first index to a function taking a void*
.
i.e.,
void do_something_with_pointer_to_mem( void* mem )
{
// do stuff
}
int main( void )
{
std::vector< float > v;
// fill v with values here
// pass to do_something_with_pointer_to_mem
do_some_with_pointer_to_mem( &v[ 0 ] );
return;
}
Another, more concrete example is when calls to glBufferData(...) are made in OpenGL (when using C++):
glBufferData( GL_ARRAY_BUFFER, sizeof( somevector ), &somevector[ 0 ], GL_STREAM_DRAW );
So, is it possible to accomplish something similar in C using the subscript operator? If not, and I had to write a function (e.g., float vec3_value_at( unsigned int i )
), would it make sense to just static inline
it in the header file it's defined in?
You don't get the syntactic sugar of C++, but it's easy to write the function you'd write in C++ as operator[].
Now you can iterate through any vec3.
The problem in C with what you are trying to do is that you need to know how to move through a structure (i.e. you need to know the types). The reason
std::vector<T>
works the way it does is because it is using templates (aC++
concept). Now, that said, you could try something slightly different than what you suggested. If you do not want to use any arrays, you can store generic types. However, when retrieving the data and using it, the user will have to know what kind of data he or she is expecting. Below avoids arrays (although, a potentially cleaner solution exists in using them) and has a linked list implementation of something which gives you the nearly same flexibility ofstd::vector<T>
(performance benefits aside since this is a linked list withO(n)
operations for everything (you can be clever and reverse the list to achieve, perhaps,O(1)
insert, but this is merely for example)This code should compile right out of box. What you have here is a linked list which is your "vector" (particularly, a vec3 structure). Each node in the list (i.e. every element in the
std::vector<T>
sense) has 3 elements which are allvoid
pointers. Thus, you can store any data type you wish here. The only catch is that you need to allocate memory for those pointers to point to and when removing an element, you will need to free that memory (refer to thevec3_destroy
method for an example). Hope this helps a little more for understanding how these void pointers can work in your case.To retrieve data, you won't be able to use the
[]
notation, but you can use thevec3_get
method in the same way. Thedo_something
method is an example stub of some way you might be able to accomplish something similar to what you mentioned in the OP.If all of your structure fields are of the same type, you could use a union as following:
This way you could access to each x, y or z field independently or iterate over them using the vect3_a array. This solution cost nothing in term of memory or computation but we may be a bit far from a C++ like solution.