Changing a macro at runtime in C

后端 未结 5 1324
野趣味
野趣味 2020-11-30 12:29

I have a macro defined. But I need to change this value at run time depending on a condition. How can I implement this?

相关标签:
5条回答
  • 2020-11-30 12:36

    You can't. Macros are expanded by the Preprocessor, which happens even before the code is compiled. It is a purely textual replacement.

    If you need to change something at runtime, just replace your macro with a real function call.

    0 讨论(0)
  • 2020-11-30 12:39

    Depending on what you want to do, you might do it several ways.

    Global variable instead of macro

    // someincludefile.h
    extern static int foo;
    
    // someincludefile.c
    static int foo = 5;
    
    // someotherfile.c
    #include "someincludefile.h"
    printf("%d\n", foo); // >> 5
    foo = -5;
    printf("%d\n", foo); // >> -5
    

    Condition you can toggle

    // someincludefile.h
    extern static int condition;
    #define FOO1 (5)
    #define FOO2 (-5)
    #define FOO (condition ? (FOO1) : (FOO2))
    
    // someincludefile.c
    static int condition = 1;
    
    // someotherfile.c
    #include "someincludefile.h"
    printf("%d\n", FOO); // >> 5
    condition = 0;
    printf("%d\n", FOO); // >> -5
    

    Condition that's locally and dynamically evaluated

    // someincludefile.h
    #define CONDITION (bar >= 0)
    #define FOO1 (5)
    #define FOO2 (-5)
    #define FOO ((CONDITION) ? (FOO1) : (FOO2))
    
    // someotherfile.c
    #include "someincludefile.h"
    int bar = 1;
    printf("%d\n", FOO); // >> 5
    bar = -1;
    printf("%d\n", FOO); // >> -5
    

    In that last one the CONDITION will be evaluated as if its code were in your local scope, so you can use local variables and/or parameters in it, but you can also use global variables if you want.

    0 讨论(0)
  • 2020-11-30 12:48

    You can't.

    As a macro is resolved by the preprocessor before the compilation itself, its content is directly copied where you use it.

    You can still use parameters to insert a conditional statement depending on what you want, or use a call-scope accessible variable.

    If you want to change a single value, better use global scope variable, even if such behavior is discouraged. (as the intensive use of macro)

    0 讨论(0)
  • 2020-11-30 12:50

    You can't change the macro itself, i.e. what it expands to, but potentially you can change the value of an expression involving the macro. For a very silly example:

    #include <stdio.h>
    
    #define UNCHANGEABLE_VALUE 5
    #define CHANGEABLE_VALUE foo
    
    int foo = 5;
    
    int main() {
        printf("%d %d\n", UNCHANGEABLE_VALUE, CHANGEABLE_VALUE);
        CHANGEABLE_VALUE = 10;
        printf("%d %d\n", UNCHANGEABLE_VALUE, CHANGEABLE_VALUE);
    }
    

    So the answer to your question depends on what kind of effect you want your change to have on code that uses the macro.

    Of course 5 is a compile-time constant, while foo isn't, so this doesn't work if you planned to use CHANGEABLE_VALUE as a case label or whatever.

    Remember there are two (actually more) stages of translation of C source. In the first (of the two we care about), macros are expanded. Once all that is done, the program is "syntactically and semantically analyzed", as 5.1.1.2/2 puts it. These two steps are often referred to as "preprocessing" and "compilation" (although ambiguously, the entire process of translation is also often referred to as "compilation"). They may even be implemented by separate programs, with the "compiler" running the "preprocessor" as required, before doing anything else. So runtime is way, way too late to try to go back and change what a macro expands to.

    0 讨论(0)
  • 2020-11-30 13:00

    Macros are replaced by the preprocessor by their value before your source file even compiles. There is no way you'd be able to change the value of the macro at runtime.

    If you could explain a little more about the goal you are trying to accomplish undoubtedly there is another way of solving your problem that doesn't include macros.

    0 讨论(0)
提交回复
热议问题