Главная страница
Top.Mail.Ru    Яндекс.Метрика
Текущий архив: 2009.09.20;
Скачать: CL | DM;

Вниз

Объясните тонконсти хранения чисел   Найти похожие ветки 

 
Zemen   (2008-11-21 17:08) [0]

Имеем СУБД Firebird 1.5. Вопрос касается в данном случае только этой системы.

Создаю таблицу:
CREATE TABLE CAR(SP SPEED NUMERIC(15,2));

Что значит 15 (разрядность) и точность (2). В книжке написано, разрядность общее число цифр в числе, а точность - число знаков после запятой. Точность задает формат отображения либо хранения чисел ?


 
Zemen   (2008-11-21 17:13) [1]

Вопрос возник по следующей причине. Есть таблица А, числовые колонки которой имеют тип NUMERIC(15,2). В результате расчетов в числовые колонки заносятся числовые значения, у которых количество знаков после запятой больше 2-х. При этом такое в базе они все эти знаки сохраняются!


 
Правильный$Вася   (2008-11-21 17:36) [2]

из справки по IB6
NUMERIC datatype
NUMERIC(x,y)
In the syntax above, InterBase stores exactly x digits. Of that number, exactly y digits are to the right of the decimal point. For example,
NUMERIC(5,2)
declares that a column of this type always holds numbers with exactly 5 digits, with exactly two digits to the right of the decimal point: ppp.ss


 
Правильный$Вася   (2008-11-21 17:40) [3]

The following table summarizes how InterBase stores NUMERIC and
DECIMAL datatypes based on precision and scale:
Datatype specified as… Datatype stored as…
NUMERIC     INTEGER
NUMERIC(4) SMALLINT
NUMERIC(9) INTEGER
NUMERIC(10) • DOUBLE PRECISION in dialect1
                  • INT64 in dialect 3
NUMERIC(4,2) SMALLINT
NUMERIC(9,3) INTEGER
NUMERIC(10,4) • DOUBLE PRECISION in dialect1
                    • INT64 in dialect 3


 
Anatoly Podgoretsky ©   (2008-11-21 19:04) [4]

> Zemen  (21.11.2008 17:08:00)  [0]

Есть два подхода к этому, когда точность (более правильно) это разрядность, а число знаков после запятой иногда почему то называется размер. Много путаницы внесли старые СУБД, типа dBase и FoxPro. Когда 15 означало не точность или число знаков, а число байт для хранения, а точность там называлось количесто знаков после запятой ( с точностью до Х знаков после запятой - наверно слышал такую фразу). Вот подобная запись означала в общем случае 13 знаков (из них две после запятой) + запятая + знак числа. В современных 15 это точность (precision - в твоей терминологии разрядность?) и 2 количество знаков после запятой, формат чисел BCD (не относится к обработке).


 
Anatoly Podgoretsky ©   (2008-11-21 19:05) [5]

> Zemen  (21.11.2008 17:13:01)  [1]

Для Numeric - тогда это ошибка данной СУБД


 
Anatoly Podgoretsky ©   (2008-11-21 19:06) [6]

> Правильный$Вася  (21.11.2008 17:36:02)  [2]

С ИБ надо быть поосторожнее, поскольку это зависит от уровня, на первом уровне это Float, на третьем BCD


 
Anatoly Podgoretsky ©   (2008-11-21 19:09) [7]

> Правильный$Вася  (21.11.2008 17:40:03)  [3]

Вот они эти особенности IB тот вообще 4 внутренних формата для NUMERIC.


 
Johnmen ©   (2008-11-21 19:26) [8]


> Anatoly Podgoretsky ©   (21.11.08 19:09) [7]

NUMERIC в ИБ это вообще не формат.


 
Anatoly Podgoretsky ©   (2008-11-21 20:40) [9]

> Johnmen  (21.11.2008 19:26:08)  [8]

Я могу и поправиться - это тип


 
Johnmen ©   (2008-11-21 21:07) [10]


> Anatoly Podgoretsky ©   (21.11.08 20:40) [9]

Псевдотип, если точнее...
:)


 
Anatoly Podgoretsky ©   (2008-11-21 23:45) [11]

> Johnmen  (21.11.2008 21:07:10)  [10]

Ну не буду спорить, но что бардак, так точно.


 
Loginov Dmitry ©   (2008-11-22 00:10) [12]

> Есть таблица А, числовые колонки которой имеют тип NUMERIC(15,
> 2). В результате расчетов в числовые колонки заносятся числовые
> значения, у которых количество знаков после запятой больше
> 2-х. При этом такое в базе они все эти знаки сохраняются!


Нет, в базе сохраняются именно 2 знака запятой. Более того, все число физически хранится в целочисленном формате (для 3-го (актуального сейчас) диалекта - см [3]). "Благодаря" этому, не так-то просто увеличить число знаков после запятой в процессе эксплуатации, так что если есть сомнения, лучше заранее брать с запасом ))


 
Zemen   (2008-11-24 12:51) [13]

Проше прощения за отсутствие, коллеги, был в отъезде.

Обсуждения ушли в сторону. То, что по разному хранит, это одно. Но такое поведение вообще нормально ? Причем обнаружил такую вещь. При с нуля таблицы по схеме: CREATE TABLE, INSERT, UPDATE такая ситуация не наблюдается. Если делаю UPDATE SET FFF = 0.12344124, то потом вижу значение FFF = 0.12. Получается, что все-таки СУБД округлила при обновлении под тип поля. Но в тоже самое время есть старая таблица с точно таким же типом поля. Попытка выполнить такой же запрос обновления, и в таблице сидят неокругленные значения с 8 знаками после запятой!


 
Правильный$Вася   (2008-11-24 12:57) [14]

ты так и не уточнил диалект
а это важно


 
Zemen   (2008-11-24 12:59) [15]

Диалект изначально третий, но перевод БД на третий диалект ситуацию не изменил.


 
Zemen   (2008-11-24 13:00) [16]

Сейчас попробую сделать демо-базу с двумя таблицами в качестве примера.


 
Виталий Панасенко   (2008-11-24 14:08) [17]


> Zemen   (24.11.08 12:59) [15]
>
> Диалект изначально третий, но перевод БД на третий диалект
> ситуацию не изменил.

А в старой - 1-й...


 
Zemen   (2008-11-24 14:29) [18]

Эврика! Диагноз поставлен, но адекватного решения похоже нет.

Никакой демо-базы делать не нужно. Рассматриваем только разрядность от 10 чисел. В диалекте 1 тип Numeric(15,2) хранится как DOUBLE PRECISION. При обновлении никакого округления не возникает, значение 0.12345678 хранится как есть. Диалект 3 тот же самый тип хранится как INT64. Значение 0.12345678 при обновлении округляется до 0.12.

Однако при переводе базы с первого диалекта на третий поведение не меняется. Это, конечно, удивило. Поля, созданные в первом диалекте, после перевода на третий ведут себя по старому, т.е. как DOUBLE PRECISION.  Смена типа через ALTER TYPE дает ошибку, т.к. СУБД не может конвертировать значения из DOUBLE PRECISION в INT64. Новые же поля ведут себя ожидаемо. Это - диагноз :).

Теперь решение. Переход на третий диалект без полной перетряски базы не поможет. По сути надо писать отдельную программу, которая для каждого числового поля будет переписывать данные. В настоящее время это исключено. Тогда придется, учитывая данную особенность, вносить изменения в программый код, т.е. плодить заплатки или боротся со следствием, а не причиной, что требует гораздо больше времени и сил. Такие, вот, пироги.

Не ошибся ли я в своих изыскания и есть ли другие пути решения ?


 
Правильный$Вася   (2008-11-24 15:23) [19]

выгрузить в скрипт, залить его в правильную?


 
Zemen   (2008-11-24 17:25) [20]


> выгрузить в скрипт, залить его в правильную?

Сейчас это фантастика. Дешевле будет заплатки лепить :)


 
Правильный$Вася   (2008-11-24 18:21) [21]

"никогда не хватает времени, чтобы что-то сделать правильно
зато чтобы переделать, причем N раз, время находится"
вольный перефраз из законов Мерфи


 
Johnmen ©   (2008-11-24 20:54) [22]


> Обсуждения ушли в сторону.

Чего тут обсуждать-то? Документацию лень открыть?



Страницы: 1 вся ветка

Текущий архив: 2009.09.20;
Скачать: CL | DM;

Наверх




Память: 0.52 MB
Время: 0.012 c
15-1248325667
Andy BitOff
2009-07-23 09:07
2009.09.20
Наконец-то слетел хелп...


2-1247822256
Tneduts
2009-07-17 13:17
2009.09.20
Удаленное подключение к базе


15-1248080977
ford
2009-07-20 13:09
2009.09.20
trackBar с несколькими бегунками


2-1248180547
artemm
2009-07-21 16:49
2009.09.20
подсчет одинаковых строк


1-1215268868
AlexNe
2008-07-05 18:41
2009.09.20
Тихо умирающее приложение