Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск  
 
 
Рейтинг 4.60/48: Рейтинг темы: голосов - 48, средняя оценка - 4.60
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10

Использование строк в макросах С++

11.02.2013, 13:19. Показов 10512. Ответов 23
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго времени суток всем!
Возникла небольшая проблема - не могу понять, возможно ли сделать нечто подобное, и, если возможно, то как.
Есть макрос, объявляющий функцию:

C++
1
2
3
4
#define DECLARE_FUNC(name1, body, type_cl, ...)\
    __declspec(dllexport) void type_cl name1##_func(__VA_ARGS__){\
    body;\
}\
Используется следующим образом:
C++
1
DECLARE_FUNC(testDeclare, "debug_msg(\"TestDeclare\",\"hello!\")", void, void);
где строка - вызов функции debug_msg(запись в файл форматированной строки).

Собственно, вопрос - как это сделать правильно? В текущем варианте нет никакой реакции на вызов функции testDeclare_func().

P.S. В следующем варианте
C++
1
2
3
4
5
#define DECLARE_FUNC(name1, body, type_cl, ...)\
    __declspec(dllexport) void type_cl name1##_func(__VA_ARGS__){\
    debug_msg("TestDeclare","hello!");\
    body;\
}\
вывод из debug_msg("TestDeclare","hello!"); вполне адекватно отрабатывает.
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
11.02.2013, 13:19
Ответы с готовыми решениями:

Использование строк.Использование структур
Задачка: Дана строка,состоящая из групп нулей и едениц. Найти и вывести на экран группы с нечетным количеством символов. И еще одна: ...

Метки в макросах в ассемблерной вставке
В C++ написан макрос с ассемблерной вставкой, который содержит в себе метку "doit" и команду перехода на неё: #define MYMACRO(x, y,...

Использование строк
Помогите с алгоритмом программы: вводим текст происходит проверка на наличие цифр прописью, заменяет их и выводит исправленый...

23
Evg
Эксперт CАвтор FAQ
 Аватар для Evg
21281 / 8305 / 637
Регистрация: 30.03.2009
Сообщений: 22,660
Записей в блоге: 30
13.02.2013, 17:35
Студворк — интернет-сервис помощи студентам
Напомню ещё раз. Правильно поставленный вопрос - это половина ответа. Если ты хочешь, чтобы тебе ответили - потрать 10 минут и нормально опиши проблему. Мне, если честно, уже становится впадлу читать по три раза твои посты, чтобы догадаться, что же ты имеешь ввиду
0
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
14.02.2013, 22:00  [ТС]
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
#define RF_IMP(ret, type_cl, name1, format, adres, ...)\
    typedef ret (type_cl * t##name1 ) (va_list args);\
    t##name1 name1##_Detour = ( t##name1 ) ( adres );\
    void type_cl name1##_hook(char *txt, ...);\
    BYTE name1##_var[6];\
    void type_cl name1##_hook (char *txt, ...) {\
    txt=format;\
    va_list arg;\
    va_start(arg, txt);\
    name1##_Detour(arg);\
    debug_msg("ParseArgs",txt, arg);\
    va_end(arg);\
}\
Написал вышеприведенный код.
Вроде бы все хорошо, без возврата в оригинальную функцию выводит полученные аргументы, бла-бла-бла, всемирное добро и счастье.
Но вот как перейти к оригинальной функции - вот тут-то и затык. Возникает ошибка , предположительно связанная с тем, что переменное число аргументов не может работать с calling conventions, очищающими стек(__stdcall, к примеру). Впрочем, и с __cdecl'ом та же проблема возникла.
На самом деле, va_list здесь не нужен, достаточно бы передать аргументы из макроса(__VA_ARGS__) - но тут возникает проблема: вызов функции не получается провести с __VA_ARGS__(насколько понимаю, внутри содержится список аргументов с типами, в результате чего компилятор ругается на то, что нельзя вызвать функцию name_Detour(int a1).

Попробую четко сформулировать вопрос - как, собственно, вернуть управление оригинальной функции, передав при этом ей список аргументов, объявленных в макросе?

Пример использования:
C++
1
2
3
RF_IMP(int, WINAPI, send2, "%d, %d, 0x%X, %d, %d, %d, %d", 0x17E4D18, SOCKET s, LPWSABUF lpBuffers, DWORD dwBufferCount, LPDWORD lpNumberOfBytesSent, DWORD dwFlags, 
       LPWSAOVERLAPPED lpOverlapped, 
       LPWSAOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine);
Благодарю за внимание!
0
Evg
Эксперт CАвтор FAQ
 Аватар для Evg
21281 / 8305 / 637
Регистрация: 30.03.2009
Сообщений: 22,660
Записей в блоге: 30
15.02.2013, 18:25
Сначала был пример:

Цитата Сообщение от Evg Посмотреть сообщение
Код

C++
1
2
3
4
5
6
#define DECLARE_FUNC(name1, body, type_cl, ...) \
  __declspec(dllexport) type_cl name1##_func(__VA_ARGS__) {\
    body; \
  }
 
DECLARE_FUNC (testDeclare, debug_msg ("TestDeclare", "hello!"), void, int x, int y)
после препроцессора превратится в

C++
1
__declspec(dllexport) void testDeclare_func(int x, int y) { debug_msg ("TestDeclare", "hello!"); }
Затем был вопрос:

Цитата Сообщение от vxg Посмотреть сообщение
я думал что препроцессор споткнется об запятую перед hello, но пример работает. как это возможно?
Вот что на это говорит стандарт:

ISO/IEC 9899:1999 (E)
...
6.10.3 Macro replacement
...
11. The sequence of preprocessing tokens bounded by the outside-most matching parentheses forms the list of arguments or the function-like macro. The individual arguments within the list are separated by comma preprocessing tokens, but comma preprocessing tokens between matching inner parentheses do not separate arguments. If there are sequences of preprocessing tokens within the list of arguments that would otherwise act as preprocessing directives, the behavior is undefined.
Вот как надо поступать, если мы хотим вставить круглые скобки в качестве параметра макроса:

C
#define LB (
#define RB )
#define A(x1, x2, x3) x1 qqq x2 qqq x3
A (LB , x , RB)
после препроцессора превратится

C
( qqq x qqq )
0
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
15.02.2013, 19:46  [ТС]
Любопытно. Спасибо за информацию.
А что насчет передачи аргументов в функцию(Использование строк в макросах С++
И есть ли какой-либо способ перехвата функциидля различных соглашений о вызовах(__stdcall, к примеру) без указания аргументов функции(к примеру, объявление общего прототипа с va_list в качестве аргумента)?

Тобишь, к примеру, есть 3 функции:
C++
1
2
3
int WINAPI WSASend(Socket s, blablabla...);
void __stdcall WhatDidThisFunctionDoAnyImagine(int a1, char *a2, bool a3); 
void * __cdecl BlaBlaReallyBla(void *arg);
и общий прототип для перехвата в стиле
C++
1
2
3
4
5
6
void name##_hook(char *format, va_list va)
{
   va_start(va,format);
   name##_original(va);
   va_end(va);
}
?
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
15.02.2013, 19:46

Использование файлов и строк
Уважаемые программисты! Нужна ваша помощь, заканчиваю написание диплома и возникла следующая проблемка: Есть программа, которая все...

Использование строк и файлов
С помощью текстового редактора создать файл, содержащий текст, длина которого не превышает 1000 символов (длина строки текста не должна...

Использование одномерного массива строк
Здравствуйте! Задание: Имеется файл 1.txt в котором содержатся строки неизвестной длины. Так же имеется файл 2.txt в котором так же...

Использование строк(как правильно?)
Здравствуйте! Была где-то реализация (условно) const char buffer; int i=0; sprintf(buffer,"message d%",i); Вопрос: можно...

Использование собственного класса строк
Здравствуйте, уважаемые господа. Есть реализация собственного класса строк: class MyString { private: size_t length; ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
24
Ответ Создать тему
Новые блоги и статьи
Алиса нашла кучу ошибок компиляции и запуска в проекте, который без проблем компилировался и запускался)))
anaschu 30.06.2026
Я пока посмеюся, но завтра проверю. А вообще интерсно. Дал алисе файл, в котором точно нет ошибок компиляции и запуска, и попросил их найти. Нашла кучу))) Критические ошибки, мешающие компиляции и. . .
сукцессия 16. Общий обзор, в основном что бы другие ии поняли
anaschu 29.06.2026
# Передаточный документ: модель микоризной сукцессии (для нового чата) Этот документ предназначен для того, чтобы новый чат Claude мог продолжить работу без необходимости заново разбираться в. . .
сукцессия 15 неявная схема
anaschu 29.06.2026
Алиса Калибровка параметров симбиотической модели: технический обзор Содержание: Введение Постановка проблемы Технические аспекты реализации Процесс внедрения изменений
сукцессия 14. Обновленная схема модели
anaschu 28.06.2026
ГЛОБАЛЬНАЯ ОПИСАТЕЛЬНАЯ СПЕЦИФИКАЦИЯ ЭКОСИСТЕМНОЙ МОДЕЛИ «SOIL CHEMISTRY & MYCORRHIZA 2. 0» https:/ / ibb. co/ NnkGpfMd Представленная интегрированная схема описывает непрерывную нелинейную. . .
сукцессия 13. Питон модель трехзонного мицелия, пока что в основном арбускулярного
anaschu 28.06.2026
## Разработка агентной модели микоризной сукцессии: от выявления артефактов к созданию комплексной системы ### Аннотация Представлено исследование по разработке агентной модели микоризной. . .
сукцессия 12. краткий список проверок модели перед запуском.
anaschu 27.06.2026
Скрытые отказы в моделях систем динамики (SD-models) экологических систем: два случая из практики Контекст Разбирался прототип модели систем динамики (SD-модели) микоризной сукцессии: пять. . .
Сукцессия 11. Проверка орудий перед войной: разработка через тестирование
anaschu 27.06.2026
Как не дать модели соврать самой себе: проверки для симуляции микоризной сукцессии Введение Когда вы строите математическую модель живой системы — грибов, растений, почвы — главная опасность. . .
10 сукцессия. Питон код войны грибов и растений
anaschu 27.06.2026
import numpy as np class PlantAgent: def __init__(self, name, strategy, initial_biomass): self. name = name self. strategy = strategy # "greedy" (широколиственные) или. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru