Evgenii Legotckoi
Evgenii LegotckoiҚыр. 2, 2017, 7:25 Т.Ж.

C++ - Сабақ 008. Есептер

Сыныптарға қоса, C++ сандарды қолдайды. Заманауи C++ стандарты C++ тілінің бұрынғы нұсқаларында енгізілген ауқымы жоқ сандарды да, сонымен қатар C да қолдайды.

enum ColorTypes {
    Green,
    Yellow,
    Red
};

Ауқымды сандармен бірдей.

enum class ColorTypes {
    Green,
    Yellow,
    Red
};

Ауқымы бар сандарды ауқымы жоқ сандардан ажырату

Ауқымды сан айнымалы сандар бүтін айнымалы мәндерге және олардан жасырын түрлендіруге болмайтындығымен шектелмеген саннан ерекшеленеді. Мұндай түрлендіру үшін static_cast пайдалану керек ().

enum class ColorTypes { Green, Yellow, Red };

ColorTypes currentType_1 = ColorTypes::Green; // Ok
ColorTypes currentType_2 = 2; // Error, преобразование невозможно
int currentType_3 = ColorTypes::Red; // Error, преобразование невозможно
int currentType_4 = Red; // Error, Red не существует в данной области видимости
int currentType_5 = static_cast<int>(ColorTypes::Green);

Бұл аумақты басқару элементі мүшелерінің аты бірдей сыныптағы тізімдерді жариялауға мүмкіндік береді. Сондай-ақ, ол бірқатар шектеулерді енгізсе де, кодты көбірек бақылауды қамтамасыз етеді.

Сандық мәндер қалай орнатылады

Әдепкі бойынша, тізім 0-ден басталады, содан кейін тізім мүшелері ұлғаяды...

enum class ColorTypes {
    Green,      // 0
    Yellow,     // 1
    Red,        // 2
    Black,      // 3
    Blue        // 4
};

бірақ жариялау кезінде тізімдер үшін өз мәндеріңізді орнатуға болады.

enum class ColorTypes {
    Green = 15,                 // 15
    Yellow,                     // 16
    Red = 24,                   // 24
    Black = ColorTypes::Red,    // 24
    Blue                        // 25
};

Санақ үшін коммутатор регистрлерін пайдалану

Ауқымды және ауқымы жоқ сандардың екеуі де switch/case шартты және тармақталу операторларын қолдайды:

ColorTypes currentType = ColorTypes::Green;

switch (currentType)
{
    case ColorTypes::Green: std::cout << "Green"; break;
    case ColorTypes::Yellow: std::cout << "Yellow"; break;
    case ColorTypes::Black: std::cout << "Black"; break;
    case ColorTypes::Blue: std::cout << "Blue"; break;
    default: std::cout << "Unknown Type"; break;
}

Егер біз кодтың орындалуын бақылау және қамтуы бар және онсыз санауларды пайдалану кезіндегі қойылған шектеулер туралы айтатын болсақ, онда біз қандай да бір себептермен коммутатор / регистрде болған жағдайды модельдеуге болады (теру қатесі, көшіру-қою, noob, олар қайшылықтарды шешу кезінде бұрмаланды. тармақтарды біріктіру) әртүрлі типтегі тізімдер бар. Содан кейін көлемі жоқ санаулар компилятор арқылы жанама түрде қажетті түрге түрлендіріледі және код қате болса да орындалады, ал ауқымы бар санаулар жағдайында компилятор қате туралы хабарлайды және бағдарламаны құруды тоқтатады. .

Яғни, қате болған келесі код құрастырылады:

enum SideTypes {
    Top,
    Bottom,
    Right,
    Left
};

enum ColorTypes {
    Green = 8,
    Yellow,
    Red,
    Blue
};

int main(int argc, char *argv[])
{
    ColorTypes currentType = ColorTypes::Green;

    switch (currentType)
    {
        case SideTypes::Top: std::cout << "Top Side"; break;
        case ColorTypes::Green: std::cout << "Green"; break;
        case ColorTypes::Yellow: std::cout << "Yellow"; break;
        case ColorTypes::Red: std::cout << "Red"; break;
        case ColorTypes::Blue: std::cout << "Blue"; break;
        default: std::cout << "Unknown Type"; break;
    }

    return 0;
}

Ең жақсы жағдайда, компилятор Ескертуді жібереді.

warning: case value ‘0’ not in enumerated type ‘ColorTypes’ [-Wswitch]
         case SideTypes::Top: std::cout << "Top Side"; break;
         ^

Бірақ бағдарламашы «компиляторға қарағанда C++ тілін жақсы біледі және түсінеді» және ескертулерді өшіреді.

Ал келесі код жай ғана құрастырылмайды:

enum class SideTypes {
    Top,
    Bottom,
    Right,
    Left
};

enum class ColorTypes {
    Green = 8,
    Yellow,
    Red,
    Blue
};

int main(int argc, char *argv[])
{
    ColorTypes currentType = ColorTypes::Green;

    switch (currentType)
    {
        case SideTypes::Top: std::cout << "Top Side"; break;
        case ColorTypes::Green: std::cout << "Green"; break;
        case ColorTypes::Yellow: std::cout << "Yellow"; break;
        case ColorTypes::Red: std::cout << "Red"; break;
        case ColorTypes::Blue: std::cout << "Blue"; break;
        default: std::cout << "Unknown Type"; break;
    }
    return 0;
}

Компилятор компиляция қатесін береді:

error: could not convert ‘Top’ from ‘SideTypes’ to ‘ColorTypes’
         case SideTypes::Top: std::cout << "Top Side"; break;
                         ^

Санақ үшін нақты бүтін сан түрін орнату жолы

Enums сонымен қатар анықталған бүтін түрі болуы керек нақтырақ түрі болуы мүмкін:

  • таңбасыз таңба;
  • таңба;
  • int;
  • long int;
  • және т.б.

Бұл санау мәндері бар айнымалылар үшін жадтың белгілі бір көлемін бөлуге мүмкіндік береді. Мүмкін, бұл енгізілген әзірлеуге қатысты болуы мүмкін. Мақсатты платформаға байланысты жадтың белгілі бір көлемі бөлінеді.

enum class SideTypes : short int {
    Top,
    Bottom,
    Right,
    Left
};

Санау үшін итератор

Соңында, ауқымға негізделген for циклін пайдалануға болатын сандар үшін итератор жасайық.

#include <iostream>

enum class ColorTypes
{
    Blue,
    Red,
    Green,
    Purple,
    First=ColorTypes::Blue,   // участник перечисления для первого элемента
    Last=ColorTypes::Purple // участник перечисления для последнего элемента
};

ColorTypes operator++(ColorTypes& x)
{
    // std::underlying_type преобразовывает тип ColorTypes в целочисленный тип, под которым данный enum был объявлен
    return x = static_cast<ColorTypes>(std::underlying_type<ColorTypes>::type(x) + 1);
}

ColorTypes operator*(ColorTypes c)
{
    return c;
}

ColorTypes begin(ColorTypes r)
{
    return ColorTypes::First;
}

ColorTypes end(ColorTypes r)
{
    ColorTypes l=ColorTypes::Last;
    return ++l;
}

int main(int argc, char *argv[])
{
    // Используем круглые скобки для инстанцирования перечисления
    for(const auto& c : ColorTypes())
    {
        std::cout << static_cast<int>(c) << std::endl;
    }

    return 0;
}
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

ДК
  • Қар. 26, 2019, 9:38 Т.Ж.
  • (өңделген)

классная статья! Большое спасибо. Хотел бы добавить для тех, кто будет использовать это в другом классе- перед операторами и методами begin(), end() нужно поставить friend.

Пікірлер

Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
Кіріңіз немесе Тіркеліңіз
Г

C++ - Тест 001. Первая программа и типы данных

  • Нәтиже:66ұпай,
  • Бағалау ұпайлары-1
t

C++ - Тест 001. Первая программа и типы данных

  • Нәтиже:33ұпай,
  • Бағалау ұпайлары-10
t

Qt - Тест 001. Сигналы и слоты

  • Нәтиже:52ұпай,
  • Бағалау ұпайлары-4
Соңғы пікірлер
G
GoattRockҚыр. 3, 2024, 1:50 Т.Қ.
Linux жүйесінде файлдарды қалай көшіруге болады Задумывались когда-нибудь о том, как мы привыкли доверять свои вещи службам грузоперевозок? Сейчас такие услуги стали неотъемлемой частью нашей жизни, особенно когда речь идет о переездах между …
d
dblas5Шілде 5, 2024, 11:02 Т.Ж.
QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
k
kmssrАқп. 8, 2024, 6:43 Т.Қ.
Qt Linux - Сабақ 001. Linux астында Autorun Qt қолданбасы как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Анатолий КононенкоАқп. 5, 2024, 1:50 Т.Ж.
Qt WinAPI - Сабақ 007. Qt ішінде ICMP Ping арқылы жұмыс істеу Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
Енді форумда талқылаңыз
Evgenii Legotckoi
Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
F
FynjyШілде 22, 2024, 4:15 Т.Ж.
при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
BlinCT
BlinCTМаусым 25, 2024, 1 Т.Ж.
Нарисовать кривую в qml Всем привет. Имеется Лист листов с тосками, точки получаны интерполяцией Лагранжа. Вопрос, как этими точками нарисовать кривую? ChartView отпадает сразу, в qt6.7 появился новый элемент…
BlinCT
BlinCTМамыр 5, 2024, 5:46 Т.Ж.
Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
Evgenii Legotckoi
Evgenii LegotckoiМамыр 2, 2024, 2:07 Т.Қ.
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.

Бізді әлеуметтік желілерде бақылаңыз