Views: 2357
Какие задачи нам позволяют решать структуры и объединения?
Для разработчика встроенных систем эффективность и компактность кода всегда на первом месте. Если программировании на Ассемблере ты сам определяешь как и где располагаются данные, то при программировании на Си надо позаботиться, что бы объяснить компилятору как ты хочешь, что бы данные были расположены. Для чего это надо, в первую очередь, для удобства обработки и обращения к данным.
Например, мне необходимо, чтобы данные были расположены последовательно в памяти. Для этого я опишу структуру, например:
//
struct
{
int16_t s_CLt; // данные в формате 16 байт со знаком
int16_t s_tNd; //
int16_t s_rEL; //
uint8_t s_SEc; // данные в формате 8 байт без знака (только положит значение)
uint8_t s_Nin; //
unsigned Accident :2; // данные в формате 2 байта без знака (только положит значение)
unsigned Freezing :1; // данные в формате 1 байта
}EE; // данные подлежащие хранению в еепром
//
Это будет гарантировано, что данные в памяти будут расположены последовательно и займут 9 байт (если система процессора микроконтроллера 8 битная) или 6 слов (если 16 битная). Один дополнительный байт будут занимать две переменные описанные как Accident и Freezing, они займут соответственно 0 – 1 байт (Accident ) и 2 байт (Freezing).
Обратиться т.е записывать данные и читать можно таким образом, например:
//
// Для записи
EE.s_CLt = 4562;
EE.Accident = 2;
// Для чтения
temp = EE.s_CLt;
temp1 = EE.s_Nin;
//
Со структурами struct все довольно понятно, это расположение данных последовательно в памяти и удобный доступ к ним, особенно, если надо писать какие-то флаги управления и потом данные “скопом” передавать через какой либо интерфейс на другое устройство. Но часто возникает необходимость например иметь представление одних и тех же данных и в виде байта (или слова) и в виде бит. Как это сделать, для этого в Си есть гибкий механизм объединения union.
Например, для передачи данных через последовательный порт нам необходимо иметь доступ к данным ка к байту, а для эффективности управления флагами управления содержащимся в этом байте, и меть доступ как к биту. Вот такой фокус и позволяют делать объединения. Еще раз структуры последовательно располагать данные в памяти, объединения описывать одни и те же данные разными именами и при этом разными типа данными.
Например, мы имеем структуру данных:
//
struct
{
int16_t s_CLt; // данные в формате 16 байт со знаком
uint8_t s_SEc; // данные в формате 8 байт без знака (только положит значение)
unsigned Accident :2; // данные в формате 2 байта без знака (только положит значение)
unsigned Freezing :1; // данные в формате 1 байта
}EE; //
//
Визуально это выглядит так:
Вся структура занимает 4 байта. EE.s_CLt занимает 2 байта, EE.s_SEc занимает 1 байта, переменные Accident, Freezing (два и и один байт) будут размещены в 4 байте.
Теперь нам, например, необходимо работать с битами переменной s_CLt, можно конечно использовать операциями с битами, например, нам надо контролировать состояние бита 0 в этой переменной мы, можем вычислить так, выполняем побитовое “&” с переменной и в зависимости от состояния операции выполняет если true или falce:
//
if(EE.s_CLt & 0b0000000000000001) ******;
else *******;
//
Но можно каждому биту присвоить свое имя, это улучшает понимание программы и не рисвоать, что нарисовано выше например писать просто так:
//
if(EE.FLED1) ******;
else *******;
//
Где EE.FLED1 мы дали имя биту 0 переменной EE.s_CLt, как это сделать? В нашу структуру надо внедрить объединение. Структуры и объединения можно как угодно комбинировать для всевозможного описания данных в памяти для удобной последующей обработки. В нашем варианте это будет выглядеть так:
//
struct
{
//-
union
{
int16_t s_CLt; // данные в формате 16 байт со знаком
struct
{
unsigned FLED1 :1;// название бита 0
unsigned FLED2 :1;// название бита 1
unsigned FLED3 :1;// название бита 2
//*****
};
};
uint8_t s_SEc; // данные в формате 8 байт без знака (только положит значение)
unsigned Accident :2; // данные в формате 2 байта без знака (только положит значение)
unsigned Freezing :1; // данные в формате 1 байта
}EE; //
//
Переменную s_CLt мы помещаем, в обеднение в котором находиться эта переменная и новая внутренняя структура. Название ни объединению, ни структуре мы не даем. В этом варианте мы сможет обращаясь, например, к EE.FLED1 контролируя или изменяя состояние бита 0 переменной EE.s_CLt.
Как это выглядит визуально.
Еще раз к пониманию структур, это возможность “объяснения” компилятору, что данные надо расположить в памяти последовательно. А к пониманию объединений, что данные одни и те же могут иметь разное название. Надеюсь я смог “на пальцах” объяснить эти гибкие особенности Си.
И для окончания, например, мне необходимо обработать эти 4 байка как одно 32 битное слово, как это сделать? Это сделать просто если нашу структур поместить в объединение и добавит в ней нашу 32 битную переменную:
union
{
struct
{
//-
union
{
int16_t s_CLt; // данные в формате 16 байт со знаком
struct
{
unsigned FLED1 :1;// название бита 0
unsigned FLED2 :1;// название бита 1
unsigned FLED3 :1;// название бита 2
//*****
};
};
uint8_t s_SEc; // данные в формате 8 байт без знака (только положит значение)
unsigned Accident :2; // данные в формате 2 байта без знака (только положит значение)
unsigned Freezing :1; // данные в формате 1 байта
};
uint16_t s_32bit; // в ней все наши биты
}EE; //
Теперь при необходимости можно обратиться к переменной EE.s_32bit и получить все данные или изменить одной операцией.
Визуально это можно представить так:
Файлы для загрузки
Просто о структурах и объединениях в Си 343.32 KB 165 downloads
Проект с примером организации данных ...Это может быть интересно
Altium Designer – создание рисунков на печатной платеViews: 3551 Для создание рисунков на печатной платы в Altium Designer можно использовать возможность использовать в Altium Designer сторонних скриптов. Мне возможность эта очень понравилась и я решил её расшарить …
Цифровой спидометр для автомобиляViews: 10311 Универсальность печатной платы ch-c0030pcb позволяет создавать на её основе разнообразные устройства. Одним из таких устройств является электронный спидометр для автомобиля, в котором можно задать два компаратора скорости, например, для …
AD9833 – Programmable Waveform Generator – part twoViews: 1936 Прошло время и появилась тема, что-бы закончить проект AD9833 – Programmable Waveform Generator. Приехали печатные платы. В этот раз я печатные платы заказывал в https://jlcpcb.com/ делал это в …
Altium Designer my setup system and project structureViews: 1056 Используйте только последнее обновление!!! Updates https://catcatcat.d-lan.dp.ua/altium-designer-my-libraries-project-templates-system-settings-by-catcatcat-v23-09/ Тут хочу поделиться как я настраиваю Altium Designer и как я использую файлы DXPPreferences.DXPPrf для быстрой конфигурации и получения …
УКВ – радиоприем, часть 2Views: 6433 Пришло свободное время решил вторую часть проекта реализовать (правда есть мысль и третью с использование цветного OLED и функцией ch-светомузыки, но это только задумка… Для понимания функций интегрального …
Индикатор температурыViews: 2833 Проект для начинающих, на демо плате BB-2T3D-01. Простой индикатор температуры. Проект никак не задумывался, просто на витрину магазин Ворон нужна была демонстрационная модель на макетной плате, чего нибудь работающего. Остановились на индикаторе …
MCC PIC24 – модуль OUTPUT COMPARE – в режиме генератора звуковых сигналовViews: 812 При проектировании простых устройств автоматики, часто необходимо иметь механизм звукового оповещения. Самый верхний уровень, это формирование голосовых сообщений, но об этом, как то по позже… В самом примитивном …
Сенсорный выключатель светаViews: 16533 Хотя в настоящий момент актуальны системы управления освещением с передачей данных по электросети, но я думаю, что проекты такого рода тоже имеют право на жизнь. Анонс Три вида …
MAX7219/21 и 8х8 LED дисплеиViews: 1070 MAX7219, MAX7221 предназначены для вывода информации на 8 разрядов семисегментного индикатора, но на нем легко организовать вывод на светодиодные индикаторы 8х8. продолжение следует…. Это может быть интересно
Система AT команд версии V2.0 для ESP8266 и ESP32Views: 13722 Появление нового модуля на базе ESP32 заставило систематизировать систему AT команд, а так же систему обновления и для модулей на базе ESP8266. Начиная с версии v2.0 в ESP8266 …


