реклама на сайте
подробности

 
 
2 страниц V   1 2 >  
Reply to this topicStart new topic
> inline методы классов
koluna
сообщение May 28 2009, 12:15
Сообщение #1


Профессионал
*****

Группа: Участник
Сообщений: 1 040
Регистрация: 3-01-07
Пользователь №: 24 061



Здравствуйте!

WinAVR.
Пишем класс. Большинство его методов очень простые и маленькие. Хотим сделать эти методы inline.
Причём класс ДОЛЖЕН БЫТЬ описан в .h файле, а реализация методов ДОЛЖНА БЫТЬ ВЫПОЛНЕНА в .cpp файле. Не спрашивайте почему wink.gif
Т. е., трудность - разделение описания класса и определения его inline методов на два файла.

Код ниже - правильная реализация inline методов класса?
Может быть они должны быть реализованы или строго в теле класса (по-умолчанию уже inline) или вне класса, но с указанием спецификатора доступа и обязательно в .h файле?

qwe.h
Код
class qwe
{
   public:
      inline void f(void);
};


qwe.cpp
Код
void qwe::f(void)
{
// что-то
};


Благодарю заранее!


--------------------
Благодарю заранее!
Go to the top of the page
 
+Quote Post
IgorKossak
сообщение May 28 2009, 12:24
Сообщение #2


Шаман
******

Группа: Модераторы
Сообщений: 3 064
Регистрация: 30-06-04
Из: Киев, Украина
Пользователь №: 221



Цитата(n_bogoyavlensky @ May 28 2009, 15:15) *
Может быть они должны быть реализованы или строго в теле класса (по-умолчанию уже inline) или вне класса, но с указанием спецификатора доступа и обязательно в .h файле?

Именно так, в одном файле.
Go to the top of the page
 
+Quote Post
koluna
сообщение May 28 2009, 12:26
Сообщение #3


Профессионал
*****

Группа: Участник
Сообщений: 1 040
Регистрация: 3-01-07
Пользователь №: 24 061



Цитата(IgorKossak @ May 28 2009, 15:24) *
Именно так, в одном файле.


Почему?


--------------------
Благодарю заранее!
Go to the top of the page
 
+Quote Post
IgorKossak
сообщение May 28 2009, 13:05
Сообщение #4


Шаман
******

Группа: Модераторы
Сообщений: 3 064
Регистрация: 30-06-04
Из: Киев, Украина
Пользователь №: 221



Цитата(n_bogoyavlensky @ May 28 2009, 15:26) *
Почему?

Потому, что при компиляции любого *.c\*.cpp файла, в который включен Ваш *.h файл, должна быть видна реализация inline функции. Ведь inline подразумевает вставку в код именно тела функции, а не её вызов.
Go to the top of the page
 
+Quote Post
Harbour
сообщение May 28 2009, 19:37
Сообщение #5


Местами Гуру
*****

Группа: Validating
Сообщений: 1 103
Регистрация: 5-12-04
Пользователь №: 1 323



Страуструп: "Чтобы реализация вызова подстановкой стала возможна даже для не слишком развитых систем программирования, нужно, чтобы не только определение, но и описание функции-подстановки находилось в текущей области видимости. В остальном спецификация inline не влияет на семантику вызова."

inline - это только hint компилятору. в случае определения функции где-то вовне, он ее не видит - следовательно не знает ее размер и не может сгенерить ее как inline, посему вставляет стандартный call
Go to the top of the page
 
+Quote Post
HARMHARM
сообщение May 28 2009, 22:17
Сообщение #6


читатель даташитов
****

Группа: Свой
Сообщений: 853
Регистрация: 5-11-06
Из: Днепропетровск
Пользователь №: 21 999



В принципе, простой класс вообще может быть целиком описан в .h файле. Впрочем, практика на любителя.
Еще имейте в виду - компилятор обычно инлайнит только очень короткие функции, вроде такой:
Код
    inline unsigned int getUsedCount( void )
    {
        return qty;
    }

Функции подлиннее делает вызываемыми, причем уже в конкретном объекте компиляции, если объектов несколько, функция будет в каждом...
Тут, конечно, много зависит от компилятора и его настроек. Возможно, придется пользоваться директивами вроде
Код
#pragma inline=forced
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение May 28 2009, 22:31
Сообщение #7


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(HARMHARM @ May 29 2009, 01:17) *
если объектов несколько, функция будет в каждом...
на этапе компиляции. На этапе линковки останется только один экземпляр - ибо все копии этой функции имеют одно и то же имя. Аналогичный (а может и этот же) механизм действует при инстанцировании шаблонов.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
HARMHARM
сообщение May 29 2009, 03:54
Сообщение #8


читатель даташитов
****

Группа: Свой
Сообщений: 853
Регистрация: 5-11-06
Из: Днепропетровск
Пользователь №: 21 999



Цитата(Сергей Борщ @ May 29 2009, 01:31) *
на этапе компиляции. На этапе линковки останется только один экземпляр - ибо все копии этой функции имеют одно и то же имя. Аналогичный (а может и этот же) механизм действует при инстанцировании шаблонов.

Точно! Не заметил, спасибо! rolleyes.gif
Не совсем понятно, почему линкер не ругается на одинаковые символы в этом случае? Подозреваю, что играют роль ? или ?? в начале названия метки, как и в ?cstartup... (IAR)
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение May 29 2009, 11:09
Сообщение #9


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(HARMHARM @ May 29 2009, 06:54) *
Не совсем понятно, почему линкер не ругается на одинаковые символы в этом случае?
Потому что их имена экспортируются не директивой PUBLIC, а директивой PUBWEAK (можно почитать в описани на ассемблер), которая как раз для таких случаев и придумана. Ну или компилятор генерит аналогичные признаки в объектном коде. Еще одно применение этой директивы, которым вы наверняка пользуетесь - putchar(). Библиотечный экспортируется с PUBWEAK (это видно, если посмотреть список символов библиотеки при помощи xlib), а ваш - с PUBLIC.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
alx2
сообщение May 29 2009, 11:10
Сообщение #10


Местный
***

Группа: Участник
Сообщений: 340
Регистрация: 25-10-05
Из: Пермь, Россия
Пользователь №: 10 091



Цитата(n_bogoyavlensky @ May 28 2009, 17:15) *
Причём класс ДОЛЖЕН БЫТЬ описан в .h файле, а реализация методов ДОЛЖНА БЫТЬ ВЫПОЛНЕНА в .cpp файле.
Что подразумевается под словом "описан"? Forward declaration под это понятие подпадает? Если да, то в .h файле можно написать "class qwe;", а в .cpp файле - его полное определение:
Код
class qwe {
public:
  void f(void) { /* что-то */ }
};


--------------------
Всего наилучшего,
Alex Mogilnikov
Go to the top of the page
 
+Quote Post
sergeeff
сообщение May 29 2009, 15:17
Сообщение #11


Профессионал
*****

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Про inline уже неоднократно писалось на форуме.
1. Даже если написано inline - это не значит, что компилятор сделает функцию как inline.
2. Если желаете, чтобы функция, объявленная как inline, с большей вероятностью стала такой - она должна быть описана в .h файле.
3. Если функция в .h декларирована как inline, а описана в .cpp (даже если и inline) она везде будет вызываться как обычная функция.
Go to the top of the page
 
+Quote Post
Сергей Борщ
сообщение May 29 2009, 16:58
Сообщение #12


Гуру
******

Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095



Цитата(sergeeff @ May 29 2009, 18:17) *
3. Если функция в .h декларирована как inline, а описана в .cpp (даже если и inline) она везде будет вызываться как обычная функция.
как интересно.
Код
file.h
class a_t
{
public:
    inline void test();
}
#ifdef IN_HEADER
void a_t::test()
{
}
#endif

file.cpp:

#include "file.h"
#ifndef IN_HEADER
void a_t::test()
{
}
#endif
a_t a;
void Test()
{
    a.test();
}
компилятор для file.cpp не заметит отличий.


--------------------
На любой вопрос даю любой ответ
"Write code that is guaranteed to work, not code that doesn’t seem to break" (C++ FAQ)
Go to the top of the page
 
+Quote Post
sergeeff
сообщение May 29 2009, 17:20
Сообщение #13


Профессионал
*****

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Ну да, правильнее сказать, что везде, кроме самого модуля (file.cpp, например), где inline функция описана.
Go to the top of the page
 
+Quote Post
HARMHARM
сообщение May 30 2009, 09:58
Сообщение #14


читатель даташитов
****

Группа: Свой
Сообщений: 853
Регистрация: 5-11-06
Из: Днепропетровск
Пользователь №: 21 999



Немножечко офф.
Цитата(Сергей Борщ @ May 29 2009, 14:09) *
Еще одно применение этой директивы, которым вы наверняка пользуетесь - putchar().

Я не использую putchar, вместо этого сделал свой sprintf с промежуточным буфером. Вопрос такой - будет ли эффективнее вариант с printf/putchar с той точки зрения что нет расхода памяти на буфер?
Go to the top of the page
 
+Quote Post
sergeeff
сообщение May 30 2009, 11:22
Сообщение #15


Профессионал
*****

Группа: Свой
Сообщений: 1 481
Регистрация: 10-04-05
Пользователь №: 4 007



Цитата(HARMHARM @ May 30 2009, 12:58) *
Немножечко офф.

Я не использую putchar, вместо этого сделал свой sprintf с промежуточным буфером. Вопрос такой - будет ли эффективнее вариант с printf/putchar с той точки зрения что нет расхода памяти на буфер?


Вопрос в том, что для вас эффективность. Плюс к тому, при выводе в буфер есть проблема его потенциального переполнения.
Go to the top of the page
 
+Quote Post

2 страниц V   1 2 >
Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 20th July 2025 - 07:39
Рейтинг@Mail.ru


Страница сгенерированна за 0.0206 секунд с 7
ELECTRONIX ©2004-2016