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

непонятки - C++

Восстановить пароль Регистрация
 
 
Рейтинг: Рейтинг темы: голосов - 37, средняя оценка - 4.65
Noname2512
4 / 4 / 1
Регистрация: 25.06.2010
Сообщений: 106
02.08.2011, 13:40     непонятки #1
у меня есть прога которая берет дабл и разделяет его на две сост. целое и дробное
C++
1
2
this->z = int(d);
this->p = ( d - int(d) )*100+0.5;
объясните почему без "+0.5" ничего не работает для чисел чья дробная часть( нечетная и меньше равна 9 )?
Similar
Эксперт
41792 / 34177 / 6122
Регистрация: 12.04.2006
Сообщений: 57,940
02.08.2011, 13:40     непонятки
Посмотрите здесь:

непонятки с матрицой C++
непонятки с указателями C++
C++ Непонятки со списком
Непонятки с синтаксисом C++
C++ Непонятки по ссылкам
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
castaway
Эксперт С++
4848 / 2987 / 368
Регистрация: 10.11.2010
Сообщений: 11,028
Записей в блоге: 10
Завершенные тесты: 1
02.08.2011, 17:24     непонятки #21
Цитата Сообщение от grizlik78 Посмотреть сообщение
Ещё раз, например нужно из суммы 123.45 выделить рубли и копейки. И то, и другое должно быть целыми числами
Так в чем проблема?
C
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
int main()
{
    double  a = 1.2;
 
    int rubli, kopeyki;
    double  a_f;
 
    rubli = floor( a );
    a_f = a - rubli;
 
    kopeyki = ceil( a_f * 100 );
 
    printf( "rubli = %d\nkopeyki = %d\n", rubli, kopeyki );
 
    return 0;
}
Добавлено через 1 минуту
Цитата Сообщение от grizlik78 Посмотреть сообщение
Вопрос был простой: зачем прибавлять 0.5
А прибавлять 0.5 я так полагаю нужно для того чтобы округление происходило в большую сторону.
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 17:24     непонятки #22
Цитата Сообщение от grizlik78 Посмотреть сообщение
Ещё раз, например нужно из суммы 123.45 выделить рубли и копейки. И то, и другое должно быть целыми числами
Ни modf ни ceil этого не дадут - тогда чтобы перебрать все знаки после запятой нужно всё время умножать на 10, пока последняя цифра не станет нулём, а затем поделить обратно, поясню

0,2563 =
25630 = 0,2563*10^5 => всего после запятой 5-1 знака, а там уже что делать c дробной частью дело ТС

А если к рублям - то преобразовать к инт int(spart*100)
grizlik78
Эксперт С++
 Аватар для grizlik78
1884 / 1416 / 102
Регистрация: 29.05.2011
Сообщений: 2,961
02.08.2011, 17:34     непонятки #23
Цитата Сообщение от lazybiz Посмотреть сообщение
Так в чем проблема?
В том, что твоя программа из 1.07 делает 1 руб 8 коп.

Цитата Сообщение от lazybiz Посмотреть сообщение
А прибавлять 0.5 я так полагаю нужно для того чтобы округление происходило в большую сторону.
К ближайшему целому

Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
А если к рублям - то преобразовать к инт int(spart*100)
К копейкам.
int(spart*100 + 0.5)

Не по теме:

Забавные вы, ребята

-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 17:36     непонятки #24
Вот за копейки, я взял код lazybiz чтобы исключить int()
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
#include <iostream.h>
#include <math.h>
 
int main()
{
    char ch;
    double val;
 
    double fpart;
    double spart;
    do
    {
        cout<<"Enter double\r\n";
        cin>>val;
 
        fpart = floor(val);
        spart = val - fpart;
        cout<<fpart<<" RU "<<int(spart*100)<<" KOP\r\n";
 
        cout<<"Celay chast'"<<fpart<<"\r\n";
        cout<<"Drobn chast'"<<spart<<"\r\n";
 
        cout<<"[Y/N] - Y - Enter new value\r\n";
        cin>>ch;
    }
    while(ch == 'Y' || ch == 'y');
    return 0;
}
Вот работа
Enter double
123.43
123 RU 43 KOP
Celay chast'123
Drobn chast'0.43
[Y/N] - Y - Enter new value


Добавлено через 1 минуту
Цитата Сообщение от grizlik78 Посмотреть сообщение
К копейкам.
int(spart*100 + 0.5)
ну может хватит, смотри пост выше!Lazybiz, юзать ceil это уже лишнее
castaway
Эксперт С++
4848 / 2987 / 368
Регистрация: 10.11.2010
Сообщений: 11,028
Записей в блоге: 10
Завершенные тесты: 1
02.08.2011, 17:37     непонятки #25
Цитата Сообщение от grizlik78 Посмотреть сообщение
В том, что твоя программа из 1.07 делает 1 руб 8 коп.
Замени ceil на floor в последнем случае тогда округление будет происходить в меньшую сторону.
А вообще мне кажется что ты уже придираешься.

Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
Lazybiz, юзать ceil это уже лишнее
Ну почему же, вовсе нет. Это смотря какой результат ты ожидаешь увидеть.
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 17:40     непонятки #26
Цитата Сообщение от lazybiz Посмотреть сообщение
Замени ceil на floor в последнем случае тогда округление будет происходить в меньшую сторону.
А вообще мне кажется что ты уже придираешься.
Согласен на 100%!
grizlik78
Эксперт С++
 Аватар для grizlik78
1884 / 1416 / 102
Регистрация: 29.05.2011
Сообщений: 2,961
02.08.2011, 17:41     непонятки #27
Цитата Сообщение от lazybiz Посмотреть сообщение
Замени ceil на floor в последнем случае тогда округление будет происходить в меньшую сторону.
Тогда ошибка будет в других числах.

Цитата Сообщение от lazybiz Посмотреть сообщение
А вообще мне кажется что ты уже придираешься.
Фигассе. Да бухгалтеры за копейку убьют!

Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
Вот работа
Ага. И вот тоже работа (твоей программы):
Enter double
1.12
1 RU 12 KOP
Celay chast'1
Drobn chast'0.12
[Y/N] - Y - Enter new value
y
Enter double
1.13
1 RU 12 KOP
Celay chast'1
Drobn chast'0.13
[Y/N] - Y - Enter new value
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 17:43     непонятки #28
grizlik78
Эксперт С++
 Аватар для grizlik78
1884 / 1416 / 102
Регистрация: 29.05.2011
Сообщений: 2,961
02.08.2011, 17:46     непонятки #29
Вы конечно и дальше можете не замечать проблему в вашем коде, это ваше дело.
Но повторю ещё раз. Код в исходном посте вполне рабочий и решает поставленную задачу. Вопрос был только в том, почему он такой. Ответ я давно дал и ТС, насколько я понимаю, его понял. А я пока помолчу.
castaway
Эксперт С++
4848 / 2987 / 368
Регистрация: 10.11.2010
Сообщений: 11,028
Записей в блоге: 10
Завершенные тесты: 1
02.08.2011, 17:54     непонятки #30
Цитата Сообщение от grizlik78 Посмотреть сообщение
Фигассе. Да бухгалтеры за копейку убьют!
Да я не про копейку, а про способ реализации. Тут как ни пиши в любом случае в какую-то сторону нужно округлять.
diagon
Higher
 Аватар для diagon
1920 / 1186 / 49
Регистрация: 02.05.2010
Сообщений: 2,925
Записей в блоге: 2
02.08.2011, 18:00     непонятки #31
Цитата Сообщение от lazybiz Посмотреть сообщение
Тут как ни пиши в любом случае в какую-то сторону нужно округлять
Есть decimal в с#, есть длинная арифметика. Обычно их в финансах используют...
grizlik78
Эксперт С++
 Аватар для grizlik78
1884 / 1416 / 102
Регистрация: 29.05.2011
Сообщений: 2,961
02.08.2011, 18:13     непонятки #32
В C99 есть функция round(), которая делает нужное округление. Но можно её и самостоятельно реализовать, например так:
C
1
2
3
4
5
6
double round(double x)
{
    if (x < 0)
        return ceil(x - 0.5);
    return floor(x + 0.5);
}
Добавлено через 1 минуту
Ну а про то, что в финансах double не используют я уже тоже говорил
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9373 / 5423 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
02.08.2011, 19:02     непонятки #33
C
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
#include <stdio.h>
#include <math.h>
 
#define PRECISION 3 /* digits after point */
 
int main(void){
    double val;
    int fract;
    
    while ( printf("Value: ") && scanf("%lf", &val) == 1 ){
        fract = (int)((val - floor(val)) * pow(10.0, PRECISION) + 0.5);
        while ( fract && ! ( fract % 10 ) )
            fract /= 10;
        printf("Int part: %d Fract part: %d\n", (int)val, fract);
    }
    
    return 0;
}
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 22:02     непонятки #34
Цитата Сообщение от grizlik78 Посмотреть сообщение
В C99 есть функция round(), которая делает нужное округление. Но можно её и самостоятельно реализовать, например так:
Код C
1
2
3
4
5
6
double round(double x)
{
* * if (x < 0)
* * * * return ceil(x - 0.5);
* * return floor(x + 0.5);
}
Добавлено через 1 минуту
Ну а про то, что в финансах double не используют я уже тоже говорил
Вот тебе код, чтобы ты удовлетворился, всё дело было в точности округления, если округляем до копейки то добавив к дробной части 10^-5 будем иметь погрешность 0,1% что считай ноль
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
#include <iostream.h>
#include <math.h>
 
int main()
{
        char ch;
        double val;
 
        double fpart;
        double spart;
        do
        {
                cout<<"Enter double\r\n";
                cin>>val;
 
                fpart = floor(val);
                spart = (val - fpart);
                cout<<fpart<<" RU "<<int(spart*100 + pow(0.1,4))<<" KOP\r\n";
 
                cout<<"Celay chast'"<<fpart<<"\r\n";
                cout<<"Drobn chast'"<<spart<<"\r\n";
 
                cout<<"[Y/N] - Y - Enter new value\r\n";
                cin>>ch;
        }
        while(ch == 'Y' || ch == 'y');
        return 0;
}
Изображения
 
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 22:05     непонятки #35
Цитата Сообщение от easybudda Посмотреть сообщение
fract = (int)((val - floor(val)) * pow(10.0, PRECISION) + 0.5);
опять это зловещее 0,5О_о,ребят ну без него же ведь работает

Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
int(spart*100 + pow(0.1,4))
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9373 / 5423 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
02.08.2011, 22:10     непонятки #36
Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
опять это зловещее 0,5,ребят ну без него же ведь работает
А что в нём зловещего?

Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
int(spart*100 + pow(0.1,4))
Это что?
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 22:14     непонятки #37
Ну просто зачем ещё 0,5 добавлять, почти во всех постах оно сквозит, достаточно добавить 10^-5 чтобы преобразование int() было корректным



Цитата Сообщение от easybudda Посмотреть сообщение
int(spart*100 + pow(0.1,4))
Это что?
ответ в коде в моём посте выше
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
double fpart;
        double spart;
        do
        {
                cout<<"Enter double\r\n";
                cin>>val;
 
                fpart = floor(val);
                spart = (val - fpart);
                cout<<fpart<<" RU "<<int(spart*100 + pow(0.1,4))<<" KOP\r\n";
 
                cout<<"Celay chast'"<<fpart<<"\r\n";
                cout<<"Drobn chast'"<<spart<<"\r\n";
вобщем pow(0.1,4) - думаю можно сказать что это погрешность округления, вобще изначально мне не понятно почему int(spart*100) не давало желаемого результата, возможно дабл для системы представлябтся как 2.0 = 1,9(9) ???
easybudda
Модератор
Эксперт С++
 Аватар для easybudda
9373 / 5423 / 914
Регистрация: 25.07.2009
Сообщений: 10,423
02.08.2011, 22:23     непонятки #38
Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
вобщем pow(0.1,4) - думаю можно сказать что это погрешность округления, вобще изначально мне не понятно почему int(spart*100) не давало желаемого результата, возможно дабл для системы представлябтся как 2.0 = 1,9(9) ???
Вот и за чем это нужно, если прибавление 0.5 к округляемой величине даёт понятный и предсказуемый результат?
castaway
Эксперт С++
4848 / 2987 / 368
Регистрация: 10.11.2010
Сообщений: 11,028
Записей в блоге: 10
Завершенные тесты: 1
02.08.2011, 22:25     непонятки #39
Цитата Сообщение от diagon Посмотреть сообщение
Есть decimal в с#, есть длинная арифметика. Обычно их в финансах используют...
Отлично. Я очень рад за C#.

Цитата Сообщение от grizlik78 Посмотреть сообщение
Но повторю ещё раз. Код в исходном посте вполне рабочий и решает поставленную задачу. Вопрос был только в том, почему он такой.
Цитата Сообщение от grizlik78 Посмотреть сообщение
Ну а про то, что в финансах double не используют я уже тоже говорил
Вот именно. Почему он такой? И при чем тут финансы-хринансы я вообще не пойму. Автор поста про финансы ни слова не сказал.
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
02.08.2011, 22:31     непонятки
Еще ссылки по теме:

C++ Непонятки с wchar_t
C++ непонятки со scanf
C++ непонятки с for

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

Или воспользуйтесь поиском по форуму:
-=ЮрА=-
Заблокирован
Автор FAQ
02.08.2011, 22:31     непонятки #40
Цитата Сообщение от easybudda Посмотреть сообщение
от и за чем это нужно, если прибавление 0.5 к округляемой величине даёт понятный и предсказуемый результат?
pow(0.1,4) - даёт тоже, ну и как мне кажется алгоритм внешне выглядит проще
Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
fpart = floor(val);
* * * * * * * * spart = (val - fpart);
* * * * * * * * cout<<fpart<<" RU "<<int(spart*100 + pow(0.1,4))<<" KOP\r\n";
Чем разбирать дробную часть на цифры

Цитата Сообщение от easybudda Посмотреть сообщение
fract = (int)((val - floor(val)) * pow(10.0, PRECISION) + 0.5);
* * * * * * * * while ( fract && ! ( fract % 10 ) )
* * * * * * * * * * * * fract /= 10;
Цитата Сообщение от easybudda Посмотреть сообщение
(int)val
- если модуль val будет превосходить INT_MAX не будет потери точности вычислений???
Yandex
Объявления
02.08.2011, 22:31     непонятки
Ответ Создать тему

Метки
округление
Опции темы

Текущее время: 19:16. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru