Форум программистов, компьютерный форум CyberForum.ru
Наши страницы

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
Рейтинг: Рейтинг темы: голосов - 34, средняя оценка - 4.82
TimoshkaXXX
0 / 0 / 0
Регистрация: 15.04.2009
Сообщений: 49
#1

битовые поля!! - C++

07.07.2009, 00:28. Просмотров 4359. Ответов 7
Метки нет (Все метки)

добрый вечер!
имеется класс представления битовых полей:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#ifndef HTBITFIELDS
#define HTBITFIELDS
 
#include <iostream>
using namespace std;
 
#include "Cmystring.h"
 
class TBitFields
{
  private:  
      unsigned* bfields;    // массив размещения битовых полей
      int size;         // размер bfields 
      int fCount;       // к-во полей
      Cmystring* fNames;    // имена полей
      char*  fSizes;        // размеры полей
      char*  fIndex;        // индексы полей в bfields
      char*  fPos;      // позиции полей в bfields
  public:
      // Конструкторы ---------------------------------
      TBitFields (int _fCount, Cmystring* _fNames, char* _fSizes);                  
      // Деструктор -----------------------------------
      ~TBitFields (void);     
      // установить значение поля
      void SetField(const Cmystring _fName, unsigned fVals);    
      // получить значение поля
      unsigned GetField(const Cmystring _fName) const;  
//    unsigned GetField(const Cmystring _fName) const;  
      // прибавить значение к полю
      void AddField(const Cmystring _fName, unsigned fVals);    
      unsigned& operator[] (const Cmystring _fName);
//    unsigned operator[] (const Cmystring _fName);
      TBitFields& operator= (unsigned num);
      TBitFields& operator[] (const Cmystring _fName);
      friend ostream& operator<<(ostream& stream, const TBitFields& _bf);
      //friend istream& operator>>(istream& stream,       TBitFields& _bf);
  
private:
    int _findField(const Cmystring _fName) const;
 
};
 
#endif
который внутри каждого элемента массива unsigned'ов может хранить заданное число битовых полей заданной длины.
все объявленные здесь функции у меня работают, но нужно еще перегрузить операции присваивания и индексации так чтобы выполнялся следующий код:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
int main()
{
    setlocale(LC_CTYPE, "Russian");
 
    Cmystring fnames[] = { "a", "b", "c", "d", "e", "f", "g", "h" };
    char      fsize[]  = {   8,  16,   8,  16,  16,   8,   8,   8 };
    int fcount = 8;
 
    TBitFields bf(fcount, fnames, fsize);
 
    bf["a"] = 1;    cout << "a = " << bf["a"] << endl;
    bf["b"] = 2;    cout << "b = " << bf["b"] << endl;
    bf["c"] = 3;    cout << "c = " << bf["c"] << endl;
    bf["d"] = 4;    cout << "d = " << bf["d"] << endl;
    bf["e"] = 5;    cout << "e = " << bf["e"] << endl;
    bf["f"] = 6;    cout << "f = " << bf["f"] << endl;
    bf["g"] = 7;    cout << "g = " << bf["g"] << endl;
    bf["h"] = 8;    cout << "h = " << bf["h"] << endl;
подскажите пожалуйста как это сделать??

зы описание методов класса:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include "TBitFields.h"
 
TBitFields::TBitFields (int _fCount, Cmystring* _fNames, char* _fSizes)
{
    fCount = _fCount;//количество полей
    fSizes = new char[fCount]; //размеры полей
    fNames = new Cmystring[fCount]; //имена полей
    int s = 0;
    fIndex =  new char[fCount];//смещение от начала страницы 
    fPos = new char[fCount]; //номер элемента массива bfields в котором надоится поле
    for (int i = 0; i < fCount; i++)
    {
        fNames[i] = _fNames[i]; //копируем имена полей и их размеры
        fSizes[i] = _fSizes[i];
        s += fSizes[i]; //сумма размеров всех полей
    }
    fIndex[0] = 0;//для первого поля индекс равен 0
 
    for (int i = 1; i < fCount; i++)
    {
        fIndex[i] = fIndex[i - 1] + fSizes[i - 1]; //для каждого последующего поля индекс равен индексу предыдущего + размер ..                                                         //предыдущего поля
    }
    int k = 0;
    for (int i = 0; i < fCount; )
    {
        while (fIndex[i] < sizeof(unsigned) * (k + 1))//пока индекс поля меньше смещения текущего элемента bfields относительно начала, 
            fPos[i++] = k;                             //позиция данного поля равна индексу текущего элемента bfileds
        k++;
    }
    size = s / sizeof(unsigned) + 1;
    bfields = new unsigned[size];
    
}
TBitFields::~TBitFields()
{
    delete [] bfields;
    delete [] fSizes;
    delete [] fNames;
    delete [] fIndex;
    delete [] fPos;
}
void TBitFields::SetField(const Cmystring _fName, unsigned fVals)//установить значение поля
{
    for (int i = 0; i < fCount; i++)
    {
        if (fNames[i] == _fName)
        {
            unsigned tmp =  ~0;
            int left = fIndex[i] - fPos[i] * sizeof(unsigned);
            int right = left + fSizes[i] - 1;
            tmp = ~(((tmp << 32 - left) >> 32 - left + right) << right);
            bfields[fPos[i]] &= tmp;
            bfields[fPos[i]] |= fVals << right;
            break;
        }
    }
}
unsigned TBitFields::GetField(const Cmystring _fName) const
{
    unsigned res = 0;
    for (int i = 0; i < fCount; i++)
    {
        if (fNames[i] == _fName)
        {
            unsigned tmp =  ~0;
            int left = fIndex[i] - fPos[i] * sizeof(unsigned);
            int right = left + fSizes[i] - 1;
            tmp = (((tmp << 32 - left) >> 32 - left + right) << right);
            res = bfields[fPos[i]] & tmp >> right;
            break;
        }
    }
    return res;
}
void TBitFields::AddField(const Cmystring _fName, unsigned fVals)
{
    unsigned tmp = GetField(_fName);
    SetField(_fName, tmp + fVals);
}
ostream& operator<<(ostream& stream, const TBitFields& _bf)
{
    for (int i = 0; i < _bf.fCount; i++)
    {
        stream << _bf.fNames[i] << " = " << _bf.GetField(_bf.fNames[i]) << endl;
    }
    return stream;
}
/*unsigned TBitFields::operator[] (const Cmystring _fName)
{
    
}*/
unsigned& TBitFields::operator [](const Cmystring _fName)
{
    unsigned result = GetField(_fName);
    return result;
}
TBitFields& TBitFields::operator= (unsigned num)
{
    return *this;   
}
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
07.07.2009, 00:28
Здравствуйте! Я подобрал для вас темы с ответами на вопрос битовые поля!! (C++):

Битовые операции, битовые поля. - C++
Здравствуйте! Еслть 4 диапазона чисел: 0-100, 0-100, 0-6000, 0-3. Сделать в виде битовых операций. Записать в unsigned int. Использовать...

Битовые поля. Поля без типа и имени - C++
Вчера сделал очень интересную и болезненную ошибку (с точки зрения времени, так как искал я ее долго)... представьте структуру (битовое...

Битовые поля - C++
struct TKeyFlags { int Old :1; int :6; int ...

Битовые поля - C++
Народ, напишите программу для включения 3-го разряда однобайтового числа и переключения 7-го. Никак не понимаю что и как...Напишите пож..

С++ битовые поля - C++
Написал код в котором используется характеристический вектор, все компилиться и вроде бы все замечательно, но мне нужен 20битный тип, а не...

Битовые поля - C++
Подскажите пожалуйста, как мне сохранить две цифры с помощью битовых полей?

7
accept
4822 / 3243 / 165
Регистрация: 10.12.2008
Сообщений: 10,682
07.07.2009, 10:52 #2
при индексации наверное достаточно просто значения, а при присваивании нужен изменяемый объект, тут вроде для первого случая GetField ещё можно применить и для второго - нужно объект получить в который можно записать данные, то есть как бы второй GetField только возвращающий ссылку (короче что-то, через что можно получить доступ)
0
TimoshkaXXX
0 / 0 / 0
Регистрация: 15.04.2009
Сообщений: 49
07.07.2009, 11:39  [ТС] #3
аргументом l-value индексации является строка (имя поля), а вот что она возвращать должна - в этом и проблема.
если она будет возвращать индекс элемента массива unsigned'ов, то непонятно как будет работать присваивание
и само присваивание тоже непонятно как должно работать, т.к. аргументом операции присваивания является не объект этого класса а число.
помогите плз!!
0
Humanitis
172 / 164 / 6
Регистрация: 12.01.2009
Сообщений: 430
07.07.2009, 12:18 #4
Может использовать прокси-класс,в котором хранить ссылку на объект TBitFields и имя поля.
А в самом прокси классе перегрузить оператор присваивания и оператор приведения к unsigned.
Тогда
C++
1
2
3
4
unsigned TBitFields::operator[] (const Cmystring _fName)
{
        
}
будет выглядеть
C++
1
2
3
4
ProxyClassForTBitFields TBitFields::operator[] (const Cmystring _fName)
{
      return ProxyClassForTBitFields(*this,_fName);  
}
0
TimoshkaXXX
0 / 0 / 0
Регистрация: 15.04.2009
Сообщений: 49
07.07.2009, 12:21  [ТС] #5
хм интересно!! а как тогда будет выглядеть сам прокси-класс???
0
Humanitis
172 / 164 / 6
Регистрация: 12.01.2009
Сообщений: 430
07.07.2009, 12:28 #6
Цитата Сообщение от TimoshkaXXX Посмотреть сообщение
хм интересно!! а как тогда будет выглядеть сам прокси-класс???
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class ProxyClassForTBitFields
{
   TBitFields& bf;
   Cmystring fName;
public:
   ProxyClassForTBitFields(TBitFields& _bf,Cmystring _fName)
{
bf=_bf;
fName=_fName;
}
   ProxyClassForTBitFields& operator=(const unsigned val)
{
    bf.SetField(fName,val);
    return *this;
}
 
};
0
TimoshkaXXX
0 / 0 / 0
Регистрация: 15.04.2009
Сообщений: 49
07.07.2009, 12:31  [ТС] #7
может его тогда лучше унаследовать от TBitFields?
0
Humanitis
172 / 164 / 6
Регистрация: 12.01.2009
Сообщений: 430
07.07.2009, 13:03 #8
Цитата Сообщение от TimoshkaXXX Посмотреть сообщение
может его тогда лучше унаследовать от TBitFields?
Нет
1
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
07.07.2009, 13:03
Привет! Вот еще темы с ответами:

Битовые поля - C++
Добрый вечер! Помогите, пожалуйста, найти ошибку в программе. Компиляцию проходит безошибочно, но данных не выдает при запуске. ...

Битовые поля - C++
Здравствуйте, расскажите мне пожалуйста как происходят операции с битами. Мне необходимо получить 1 байт, разделить его на 2 битовых поля...

битовые поля - C++
пользователь вводит с клавиатуры 2 беззнаковых длинных целых числа a, b. Вывести на консоль множества A, B, которые представляют битовые...

Битовые поля - C++
Классы для хранения байта (в виде битовых полей)1. Создать такой метод, чтобы ноль перемещался от разряда к разряду слева направо. Помогите...


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

Или воспользуйтесь поиском по форуму:
8
Yandex
Объявления
07.07.2009, 13:03
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru