Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.88/40: Рейтинг темы: голосов - 40, средняя оценка - 4.88
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737

Статические методы класса (static)

02.02.2015, 23:40. Показов 8114. Ответов 27
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго времени суток!
Читаю "Решение сложных задач С++ - Саттер", у него есть такая задача: вы хотите написать шаблон класса MyClass, объект которого может быть создан только для типов, имеющих константную функцию Clone(), которая не имеет параметров и возвращает указатель на объект такого же типа, т.е. тип Т должен иметь Т* T::Clone() const

автор предоставляет решение задачи:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
template<typename T>
class HasClone
{
public:
    static void Contstraints()
    {
        T* (T::*test)() const = &T::Clone;
        test;
    }
    HasClone() { void (*p)() = Constraints; }
};
 
template<typename T>
class MyClass : public HasClone
{
    //...
}
Вопрос почему метод Contstraints() статический?
Какие есть еще назначения применения статическим методов, о которых не пишут в книгах?

Добавлено через 39 минут
???
0
Лучшие ответы (1)
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
02.02.2015, 23:40
Ответы с готовыми решениями:

Статические методы класса. (Связный список)
Добрый день, такая проблема, работаю со связным списком. Учусь. С обычными типами данных всё работает, решил с классами попробовать и с...

Статические элементы класса. Inline-методы. (Решение задачи)
Пусть класс Maket описан следующим образом: #include &lt;iostream.h&gt; class Maket { private: int m; static int k; public: ...

Статические методы
Доброго времени суток! Проблема такова: имеется класс, содержащий в себе статический компонент - список элементов класса. Имеется...

27
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
02.02.2015, 23:46
Лучший ответ Сообщение было отмечено Хулиган как решение

Решение

Требование наличия функций-членов

В приведенном ниже коде осуществляется проверка наличия у шаблонного типа функции-члена с определенной сигнатурой.

C++
1
2
3
4
5
6
7
8
9
template<typename T>
class HasClone {
public:
 static void Constraints() {
  T* (T::*test)() const = &T::Clone;
  test; // для подавления предупреждения компилятора о неиспользуемой переменной
 }
 HasClone() { void (*p)() = Constraints; } 
};
Метод Contstaints() здесь введен для обобщения: все подобные проверки помещаем в него. Сама идея проверки заключается в присваивании указателей на функции. Если сигнатуры функций не совпадают (или функция с таким именем отсутствует) код не скомпилируется. Использовать приведенный класс достаточно просто:
C++
1
2
3
class TFoo : HasClone<TFoo> {
 ...
};
Как видите, можно использовать приватное наследование, т.к. по сути мы не используем никакой функциональности этого класса в run-time. Это решение обладает теми же недостатками, что и мое, но оно обобщенное и может быть легко использовано.
(ц)
Цитата Сообщение от Хулиган Посмотреть сообщение
Вопрос почему метод Contstraints() статический?
Технически - не обязательно.
Это просто прихоть автора.

Цитата Сообщение от Хулиган Посмотреть сообщение
Какие есть еще назначения применения статическим методов, о которых не пишут в книгах?
Ну хз... в книгах по-моему уже давно описали все, что только можно.

Ну например, с помощью статической функции-члена можно запилить собственный дешовенький RTTI
1
21 / 21 / 12
Регистрация: 08.01.2015
Сообщений: 66
03.02.2015, 00:22
Цитата Сообщение от Хулиган Посмотреть сообщение
C++
1
T* (T::*test)() const = &T::Clone;
Вот смотрю я на это и думаю что мистер Саттер должен был назвать свою книгу не "Решение сложных задач на С++", а как-нибудь типа "Набор ран-тайм костылей на все случаи жизни.".

Ведь если у вас в обработку попал объект, который не должен был туда попасть- это ошибка проектирования.
Ведь куча способов предотвратить такую ситуацию - но вот... Саттер учит ловить ран-тайм то, что надо ловить на этапе компиляции, как минимум, а то и на этапе рисования на бумажке.
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 00:27
Цитата Сообщение от Robar Посмотреть сообщение
а как-нибудь типа "Набор ран-тайм костылей на все случаи жизни.".
Если вы не в состоянии понять текст на языке с++,
то хотя бы попробуйте прочитать текст на русском языке:

Если сигнатуры функций не совпадают (или функция с таким именем отсутствует) код не скомпилируется.

(ц)
1
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737
03.02.2015, 01:04  [ТС]
hoggy, спасибо за хорошее пояснение, время глупых вопросов, раз на то пошло
C++
1
basic_string<..., char_traits<char>, ...>
использует char_traits<char> для указания свойств и операций выполняемых над символами строки, этот класс содержит такой статический метод сравнения 2 символов:
C++
1
static bool eq (const char_type& c, const char_type& d);
с какой целью этот метод объявлен как статический? из-за того что объект char_traits<char> не создается а при сравнении символов вызывается что то вроде такого:
C++
1
char_traits<char>::eq(c, d)
так ли это, или я это придумал?
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 01:27
Цитата Сообщение от Хулиган Посмотреть сообщение
так ли это
Ага. Тут суть такая:

В зависимости от типа символов разные операции над ними могут выполняться по разному.

C++
1
char_traits<конкретный тип символов>
Действительно определяет "свойства" конкретного типа символов.

Собственно он нужен только для того, что бы определить:
как именно те или иные операции будут выполняться над символами?
Он "выбирает" реализацию.

Но самим этим операциям объект "свойств" не нужен.
Он никак в них не участвует и не фигурирует.

Его задача была - только определить какую нужно использовать реализацию, но в самой этой реализации он уже никак не участвует.

А раз функции не нужен объект для работы, то и не зачем привязывать её к этому объекту.
Поэтому она и сделана статической.

C++
1
2
char_traits<char>::eq(c, d); //<--- примерно так она и используется
   //объект char_traits<char> её для работы просто не нужен
1
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737
03.02.2015, 18:54  [ТС]
hoggy, еще один нубовский вопросик созрел
Задача: проверить является ли класс В производным от А, вот решение:
C++
1
2
3
4
5
6
7
8
9
10
template<typename A, typename B>
class MyClass
{
    class No  {};
    class Yes { No no[2]; };
    static Yes test(A*);
    static No  test(...);
public:
    enum {Is = sizeof(test(static_cast<B*>(0))) == sizeof(Yes) };
};
если метод test() сделать не статическим, то возникнет ошибка - о невозможности вызвать метод без объекта, но ведь когда мы вызываем не статические методы в конструкторе, то по сути объекта то еще не существует? или все же там где то первоначально создается "this"? что то последнее время дурные мысли стали посещать. И почему то компилятор не жалуется на отсутствие реализации
0
30 / 47 / 19
Регистрация: 23.10.2014
Сообщений: 1,001
03.02.2015, 18:56
я лично тоже это не очень понимаю, в чем вообще смысл статичных методов? я еще понимаю статичные переменные, но методы? как это вообще понять?
0
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737
03.02.2015, 19:02  [ТС]
Dark Byte, статические методы не привязываются в объектам, поэтому им в качестве первого неявного параметра не передается указатель на объект который вызывает метод, в следствии чего они не могут использовать не статические члены, не статические члены-функции класса
0
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
03.02.2015, 19:12
Цитата Сообщение от Хулиган Посмотреть сообщение
но ведь когда мы вызываем не статические методы в конструкторе, то по сути объекта то еще не существует? или все же там где то первоначально создается "this"?
this есть, значит и объект есть, к которому будет применена вызываемая функция. Другое дело, что он ещё не полностью сконструирован.
Цитата Сообщение от Хулиган Посмотреть сообщение
И почему то компилятор не жалуется на отсутствие реализации
Потому что вызова функций нет. То, что стоит внутри sizeof не приводит к вызову функции test.
1
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 19:13
Цитата Сообщение от Хулиган Посмотреть сообщение
но ведь когда мы вызываем не статические методы в конструкторе, то по сути объекта то еще не существует?
существует.

Он может быть ещё не до конца настроен, но все его данные-члены уже инициализированны. this - реально существует.
1
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
03.02.2015, 19:14
Цитата Сообщение от Dark Byte Посмотреть сообщение
в чем вообще смысл статичных методов?
в том, чтобы показать связь функции и класса (а не объекта) в котором она объявлена.
Подобную связь можно было бы показать и созданием соответствующего пространства имен.
Но namespace'ы всё же более объемлющая сущность нежели классы.
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 19:16
Цитата Сообщение от Хулиган Посмотреть сообщение
И почему то компилятор не жалуется на отсутствие реализации
потому что она никак не используется.

В метапрограммировании это называется "техника макетов".

C++
1
 sizeof(test(static_cast<B*>(0)))
это sizeof(типа, который является результатом функции test )

Для того, что бы узнать, что это за тип данных, реализация компилятору не нужна.
Достаточно посмотреть, что за тип возвращается в прототипе функции.
0
30 / 47 / 19
Регистрация: 23.10.2014
Сообщений: 1,001
03.02.2015, 19:22
Цитата Сообщение от Хулиган Посмотреть сообщение
Dark Byte, статические методы не привязываются в объектам, поэтому им в качестве первого неявного параметра не передается указатель на объект который вызывает метод, в следствии чего они не могут использовать не статические члены, не статические члены-функции класса
Цитата Сообщение от Tulosba Посмотреть сообщение
в том, чтобы показать связь функции и класса (а не объекта) в котором она объявлена.
Подобную связь можно было бы показать и созданием соответствующего пространства имен.
Но namespace'ы всё же более объемлющая сущность нежели классы.
что то я ничего не понял... методы это методы, они так или иначе связаны с объектом класса
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 19:27
Цитата Сообщение от Dark Byte Посмотреть сообщение
что то я ничего не понял... методы это методы, они так или иначе связаны с объектом класса

C++
1
2
3
4
struct sample
{
   static void hello() { std::cout<<"hello!\n"; } //<--- как он связан с объектом?
};
0
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737
03.02.2015, 19:36  [ТС]
Цитата Сообщение от Dark Byte Посмотреть сообщение
что то я ничего не понял... методы это методы, они так или иначе связаны с объектом класса
static методы как эквивалент пространству имен
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
namespace A
{
    void SomeFunc() {}
}
 
struct B
{
    static void SomeFunc() {}
};
 
int main()
{
    A::SomeFunc();
    B::SomeFunc();
}
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 19:39
Цитата Сообщение от Хулиган Посмотреть сообщение
static методы как эквивалент пространству имен
не эквивалент, поскольку статические методы умеют больше, а значит могут быть использованы там, где не хватает возможностей обычных спейсов.

Принципиальное отличие заключается в модификаторах доступа public/protected/private
Класс может запретить использование извне.

Свободные функции в спейсах можно рассматривать,
как статические методы чей модификатор доступа априори public.
0
 Аватар для Хулиган
88 / 83 / 21
Регистрация: 08.08.2012
Сообщений: 737
03.02.2015, 19:51  [ТС]
Dark Byte, статический метод возвращающий количество созданных экземпляров данного класса
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
class A
{
    static int n;
 
public:
    A() { ++n; }
    static int count() { return n; }
};
int A::n = 0;
 
 
 
int main()
{
    A* tmp;
 
    //...
 
    if(A::count() < 5)
        tmp = new A; 
    else
        std::cout << "error";       
}
0
30 / 47 / 19
Регистрация: 23.10.2014
Сообщений: 1,001
03.02.2015, 19:53
Хулиган, ну офигеть блин, я вообще не знал что так можно делать -_- только я все равно не понимаю зачем нужна такая шняга если есть неймспейсы? они ведь для этого и предназначены...
0
Эксперт С++
 Аватар для hoggy
8973 / 4319 / 960
Регистрация: 15.11.2014
Сообщений: 9,760
03.02.2015, 19:55
Цитата Сообщение от Dark Byte Посмотреть сообщение
Хулиган, ну офигеть блин, я вообще не знал что так можно делать -_- только я все равно не понимаю зачем нужна такая шняга если есть неймспейсы? они ведь для этого и предназначены...
представьте себе "свободные функции", вызывать которые имеют право только экземпляры конкретного класса.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
03.02.2015, 19:55
Помогаю со студенческими работами здесь

Статические поля и методы
Ни как не могу понят что же такое статические поля и методы, и какое их принципиальное отличие от простых методов полей.

Статические методы (сложение полиномов)
Всем доброго дня! Собственно, хочу попросить помощи.. написал код, статический метод для сложения 2х полиномов. Полиномы представляются...

Статические члены класса
Доброго времени суток. Задача состоит в том, чтобы создать список обыектов класса Test, с возможностью последующего добавления элементов...

Статические поля класса
class mi { int a,b; public: mi(int a, int b) : a(a),b(b) {} void sw() { std::cout&lt;&lt;a&lt;&lt;&quot; - &quot;&lt;&lt;b&lt;&lt;std::endl; } ...

Статические поля класса
Не подскажете как инициализировать статический массив? Но чтобы я сам мог ввести этот массив.


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

Или воспользуйтесь поиском по форуму:
20
Ответ Создать тему
Новые блоги и статьи
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Access
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
Создание Single Page Application на фреймах
krapotkin 16.11.2025
Статья исключительно для начинающих. Подходы оригинальностью не блещут. В век Веб все очень привыкли к дизайну Single-Page-Application . Быстренько разберем подход "на фреймах". Мы делаем одну. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru