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

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

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 37, средняя оценка - 4.65
Noname2512
4 / 4 / 1
Регистрация: 25.06.2010
Сообщений: 106
#1

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

02.08.2011, 13:40. Просмотров 4573. Ответов 81

у меня есть прога которая берет дабл и разделяет его на две сост. целое и дробное
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++
Народ, уже всё перерыл, впервые с проблемой столкнулся :( Создаю консольное приложение: файл class.h #pragma once class Order{ ...

Непонятки с bind2nd - C++
Привет. Вот к примеру есть такой код: #include <algorithm> #include <functional> #include <iostream> #include <array> ...

Непонятки со scanf - C++
char str_check; //строка для функции "check", которая проверяет введенные данные { scanf("%s",&str_check); } выскакивает...

Непонятки с wchar_t - C++
Прошу помощи: что-то не так с wchar_t. Не копирует имена файлов. Содержание program_config.txt config.cfg C:\\Projects\\abx.txt...

Непонятки с итераторами - C++
Здравствуйте, господа программисты. Пытался написать вот эту задачку, но получается что-то нехорошее. Помогите, пожалуйста! #include...

Непонятки по ссылкам - C++
Доброго времени суток! Помогите, пожалуйста разобраться. Изучаю ссылки и мне кое-что не понятно. Допустим мы объявили функцию: ...

Непонятки с синтаксисом - C++
В заголовочном файле мне встретилось такое описание класса: class source_mcast_reciever { uint16_t m_port; ...

После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
castaway
Эксперт С++
4880 / 3016 / 370
Регистрация: 10.11.2010
Сообщений: 11,076
Записей в блоге: 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
Эксперт С++
1903 / 1435 / 109
Регистрация: 29.05.2011
Сообщений: 2,990
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
Эксперт С++
4880 / 3016 / 370
Регистрация: 10.11.2010
Сообщений: 11,076
Записей в блоге: 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
Эксперт С++
1903 / 1435 / 109
Регистрация: 29.05.2011
Сообщений: 2,990
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
Эксперт С++
1903 / 1435 / 109
Регистрация: 29.05.2011
Сообщений: 2,990
02.08.2011, 17:46     непонятки #29
Вы конечно и дальше можете не замечать проблему в вашем коде, это ваше дело.
Но повторю ещё раз. Код в исходном посте вполне рабочий и решает поставленную задачу. Вопрос был только в том, почему он такой. Ответ я давно дал и ТС, насколько я понимаю, его понял. А я пока помолчу.
castaway
Эксперт С++
4880 / 3016 / 370
Регистрация: 10.11.2010
Сообщений: 11,076
Записей в блоге: 10
Завершенные тесты: 1
02.08.2011, 17:54     непонятки #30
Цитата Сообщение от grizlik78 Посмотреть сообщение
Фигассе. Да бухгалтеры за копейку убьют!
Да я не про копейку, а про способ реализации. Тут как ни пиши в любом случае в какую-то сторону нужно округлять.
diagon
Higher
1926 / 1192 / 49
Регистрация: 02.05.2010
Сообщений: 2,925
Записей в блоге: 2
02.08.2011, 18:00     непонятки #31
Цитата Сообщение от lazybiz Посмотреть сообщение
Тут как ни пиши в любом случае в какую-то сторону нужно округлять
Есть decimal в с#, есть длинная арифметика. Обычно их в финансах используют...
grizlik78
Эксперт С++
1903 / 1435 / 109
Регистрация: 29.05.2011
Сообщений: 2,990
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
Эксперт С++
9456 / 5469 / 927
Регистрация: 25.07.2009
Сообщений: 10,495
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
Эксперт С++
9456 / 5469 / 927
Регистрация: 25.07.2009
Сообщений: 10,495
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
Эксперт С++
9456 / 5469 / 927
Регистрация: 25.07.2009
Сообщений: 10,495
02.08.2011, 22:23     непонятки #38
Цитата Сообщение от -=ЮрА=- Посмотреть сообщение
вобщем pow(0.1,4) - думаю можно сказать что это погрешность округления, вобще изначально мне не понятно почему int(spart*100) не давало желаемого результата, возможно дабл для системы представлябтся как 2.0 = 1,9(9) ???
Вот и за чем это нужно, если прибавление 0.5 к округляемой величине даёт понятный и предсказуемый результат?
castaway
Эксперт С++
4880 / 3016 / 370
Регистрация: 10.11.2010
Сообщений: 11,076
Записей в блоге: 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++
Нашёл код в интернете и немного не понимаю его в некоторых местах. Объясните мне пожалуйста, зачем в коде вот тут: ...

непонятки с указателями - C++
Добрый день! изучаем плюсы, наткнулся на одну странную ситуацию, не могу сообразить, в чём дело... :( Есть структура - Point...

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

Непонятки с выводом - C++
Задание гласит: Вычислить приближенное значение конечной суммы с точностью e:0.0005 ...

Непонятки с рандомом - C++
Не могу понять из-за чего рандом в классе повторяется. Есть такой код генерации пароля... class Pas { string symbols; ...


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

Или воспользуйтесь поиском по форуму:
-=ЮрА=-
Заблокирован
Автор 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     непонятки
Ответ Создать тему
Опции темы

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