Reflex is a powerful C library that provides reflection capabilities for C data structures. It allows you to introspect C structures at runtime, enabling you to access and manipulate the structure members dynamically. This library is particularly useful for tasks like serialization, deserialization, data validation, and generic data handling.
As a C library, Reflex aims to add dynamic features to the otherwise static C language. It allows developers to work with C data structures without knowing their exact layout at compile-time, making the code more flexible and adaptable to changes.
- Runtime Reflection: Introspect C data structures at runtime.
- Serialization/Deserialization: Easily convert data structures to and from various formats.
- Data Validation: Validate input data against the defined structure.
- Dynamic Access: Access structure members dynamically using their names.
- Custom Type Support: Define custom data types for more complex structures.
- Portable: Works on various architectures and platforms.
- Easy Integration: Simple API for easy integration into existing projects.
- Cross-Platform: Compatible with different operating systems.
To get started with Reflex, follow these steps:
-
Prerequisites: Make sure you have a C compiler and standard C library installed on your system.
-
Installation: See Installation section.
-
Include Header: Include the "Reflex.h" header file in your C source code.
-
Define Data Structures: Define the data structures you want to work with.
-
Runtime Reflection: Use Reflex to introspect and manipulate your data structures dynamically.
To install Reflex, follow these steps:
$ git clone https://github.com/Ali-Mirghasemi/Reflex.git
$ cd Reflex
To use Reflex in your C projects, include the "Reflex.h" header in your source code:
#include "Reflex.h"
Then, define your C data structures that you want to work with and set up the appropriate schema.
Here's a simple example demonstrating how to use Reflex:
#include <stdio.h>
#include <string.h>
#include "Reflex.h"
typedef struct {
int32_t id;
char name[50];
float price;
} Product;
// Define the Product data structure
const Reflex_TypeParams productParams[] = {
REFLEX_TYPE_PARAMS(Reflex_Type_Primary_Int32),
REFLEX_TYPE_PARAMS(Reflex_Type_Array_Char, 50),
REFLEX_TYPE_PARAMS(Reflex_Type_Primary_Float),
};
const Reflex_Schema productSchema = REFLEX_SCHEMA(Reflex_FormatMode_Param, productParams);
Reflex_Result Reflex_onFiled(Reflex* reflex, void* value, const Reflex_TypeParams* fmt) {
// Do whatever you want to do with your fields, for example simple serialize
switch (fmt->Type) {
case Reflex_Type_Primary_Int32:
printf("%d, ", *(int32_t*) value);
break;
case Reflex_Type_Array_Char:
printf("%s, ", (char*) value);
break;
case Reflex_Type_Primary_Float:
printf("%f, ", *(float*) value);
break;
default:
// Not handled
break;
}
return REFLEX_OK;
}
int main() {
// Define Reflex handler instance
Reflex reflex;
// Create a Product instance
Product product;
product.id = 1;
strcpy(product.name, "Example Product");
product.price = 19.99;
// Initialize reflex
Reflex_init(&reflex, &productSchema);
Reflex_setCallback(&reflex, Reflex_onFiled);
// Serialize a Product with scan function
Reflex_scan(&reflex, &product);
return 0;
}
For more examples and detailed documentation, please refer to the "Examples" directory and the API documentation.
We welcome contributions from the community to enhance and improve the Reflex library. If you find any issues or have suggestions for new features, please submit them as GitHub issues or create pull requests.
Reflex is licensed under the MIT License. See the LICENSE file for details.
If you have any questions or need further assistance, you can reach us at [email protected]