Разбор float по символьно

Apocalyps
Offline
Зарегистрирован: 26.07.2015

Есть переменная с типом float, в которой например хранится значение 3.03. Как можно разбить это значение, чтобы вывести его потом на 7-ми сегментный индикатор? Т.е на 3, 0, 3 в отдельном массиве или переменных

qwone
qwone аватар
Offline
Зарегистрирован: 03.07.2016
dimax
dimax аватар
Offline
Зарегистрирован: 25.12.2013

Ещё как вариант разбить через modf() на целую и дробную части

Apocalyps
Offline
Зарегистрирован: 26.07.2015

Первый вариант имхо для меня идеальный. А с modf() мне кажется потом будет еще геморрой дробную часть разбирать под удобный мне формат

UPD: вариант с dtostrf отлетает, скетч потом в МК не помещается(пишу под tiny13)

dimax
dimax аватар
Offline
Зарегистрирован: 25.12.2013

Apocalyps, для tiny13 вообще нет смысла использовать float. Всегда есть варианты обхода с целочисленной арифметикой.

Apocalyps
Offline
Зарегистрирован: 26.07.2015

dimax пишет:

Apocalyps, для tiny13 вообще нет смысла использовать float. Всегда есть варианты обхода с целочисленной арифметикой.

подскажите пожалуйста, а то не могу сообразить

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

float - число с плавающей точкой, состоит из мантиссы и порядка. AVR аппаратно не поддерживается.

dimax предлагает вместно них использовать числа с фиксированной точкой. Реализуются посредством целочисленной арифметики, поддерживаемой аппаратно.

nik182
Offline
Зарегистрирован: 04.05.2015

А мне всегда казалось, что целые это всегда без точки. Если заранее использовать целые числа, скажем в 100 раз больше реальных, и помнить об этом. Потом легче разбить целое на цифры и вывести на индикатор. Поставив точку в нужной позиции увидим float. И память подпрограммами плавающей арифметики не будет забиваться.

типа 3.03*5.55 = 303*555/100 -  быстрее посчитает и памяти меньше займёт.

DetSimen
DetSimen аватар
Offline
Зарегистрирован: 25.01.2017

Только на 10000 делить надо в посследнем случае. Ну ты понел.

nik182
Offline
Зарегистрирован: 04.05.2015

Чтобы два десятичных знака осталось надо делить на 100. Если разделить на 10000 то десятичные знаки будут нули. Еще на 100 поделится, когда десятичной точкой два знака отделим.

DetSimen
DetSimen аватар
Offline
Зарегистрирован: 25.01.2017

Да

Logik
Offline
Зарегистрирован: 05.08.2014

Ну можна и на 100 делить, но вощето - 

Числа с фиксированной точкой.

Форма записи числа с фиксированной точкой использовалась в основном на ранних этапах развития вычислительной техники. Запись числа с фиксированной точкой обычно имеет знаковый и цифровой разряды. Фиксированная точка означает, что на этапе конструирования ЭВМ было определено, сколько и какие разряды машинного слова отведены под изображение целой и дробной частей числа. Запятая в разрядной сетке может быть зафиксирована, в принципе, после любого разряда.

Пример.
Ячейка с целой и дробной частью.

ris3_1.gif

http://www.edu.sbor.net/mars/info/prof/m_mater/mat/teoria2/3.html

Только положение точки Вы сами себе выбираете как удобней а не при конструировании ЭВМ.  например ваше 3,03 в формате как на картинке будет 00000011.00001000 точней это будет 3,03125. Ошибка округления. Как персчитывать? - Ну с целой понятно думаю, а дробную часть будем считать как целое умноженое на 2^-n. В примере дробная 00001000 в десятичной форме просто целое 8 и имеем 8*1/256=0,03125. 

Зачем все эти сложности? А за тем, что матоперации с такой формой записи числа ничем не отличаются от операций с целыми. Работают быстро. От програмиста требуется только помнить где заканчивается целая часть числа и начинается дробная.

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

nik182 пишет:

А мне всегда казалось, что целые это всегда без точки. Если заранее использовать целые числа, скажем в 100 раз больше реальных, и помнить об этом. Потом легче разбить целое на цифры и вывести на индикатор. Поставив точку в нужной позиции увидим float. 

Будет не float, а fixed.

nik182
Offline
Зарегистрирован: 04.05.2015

В языке  СИ нет такого типа fixed.  На дисплее будет int, который будет выглядеть как float.

qwone
qwone аватар
Offline
Зарегистрирован: 03.07.2016

nik182 пишет:

В языке  СИ нет такого типа fixed.  На дисплее будет int, который будет выглядеть как float.

Похоже до "ардуинщиков" не дошла весть о появлении Си++ Ведь в Си++ можно изобретать разные переменные. Комплексных тоже вроде встроеных нет. Но нормально работают.

/**/
//-----------------------------------------
// класс рубль
class rub_t {
  protected:
    unsigned int rub;// целая часть
    unsigned int kop;// дробная часть
  public:
    rub_t(): rub(0), kop(0) {}
    rub_t(int rub_, int kop_): rub(rub_), kop(kop_) {}
    rub_t(float num): rub(0), kop(0) {
      rub = num;
      kop = (long)(num * 100) % 100;
    }
    void viev() {
      Serial.print(rub);
      Serial.print('.');
      Serial.print(kop);
    }
};
//--------Компановка---------------------------
rub_t AAA1(10, 99);
rub_t AAA2(2.45);
//------main()-------------------------------
void setup() {
  Serial.begin(9600);
  AAA1.viev();
  Serial.println();
  AAA2.viev();
}
void loop() {
}

 

nik182
Offline
Зарегистрирован: 04.05.2015

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

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

nik182, при чем здесь вообще Ардуино и Си? Речь идет о способах представления чисел, и от языков программирования это никак не зависит. Хотя бы потому, что языки программирования вторичны по отношению к способам представления чисел. Так что не нужно уводить разговор в сторону, деже если Вам очень не хочется признавать, что Вы написали о float, не подумав.

И, кстати, если уж на то пошло, на дисплее будет строка, являющаяся символьным представления числа с фиксированной точкой.

nik182
Offline
Зарегистрирован: 04.05.2015

Да да, конечно. Именно строка символов, представляющаяя число с десятичной дробной частью. Эквивалентная представлению типа float с фиксированным количеством знаков после десятичного разделителя, который хотел видеть на экране ТС. Получить строку для представления можно разными способами. Один из них это использование арифметики fixed point. Но даже сделав все вычисления целочисленными fixed point в конце вы получае не целое а действительное число с фиксированным количеством десятичных знаков, которому соответствует тип float. Если вы выведете полученное значение fixed point без дополнительных телодвижений вы получите число без десятичной точки. Потому что оно целое и десятичные знаки существуют только в вашем воображении в количестве, заданном перед началом вычислений. Т.е в конце из целого fixed надо представить float, которое целый fixed подразумевает. Тут конечно можно долго спорить есть точка в fixed или нет. С точки зрения целочисленности или с т.з. виртуального представления. 

magicwolf
Offline
Зарегистрирован: 03.04.2014

Если правильно понял, то надо преобразовать float  в строчку. С этим действительно есть проблема. Вот когда-то написал такую функцию, вроде работала:

#include <string.h>

typedef unsigned char byte_t;

float calc_max_flt ( char max_digit )
{
  float max_flt = 1.;

  if ( max_digit<=0 )
    return 1.;

  while (1) {
    if (max_digit <= 0)
      break;
    max_flt *= 10;
    max_digit--;
  }
  return max_flt;
}

char fill_str_digit (float * flt, char * str, byte_t * i, char max_len_str, float max_flt, char null_char )
{
  char n;
  max_flt /= 10.;
  while (1) {
    if (*i >= max_len_str-1)
      break;
    if (max_flt<1.)
      break;
    n = (char)(*flt / max_flt);
    if (n>9) {
      str[0] = 0;
      return -1;
    }
    if ( n==0 && *flt>=1. )
      str[*i]=null_char;
    else
      str[*i] = '0' + n;
    *i=*i+1;
    if (n>0)
      *flt = *flt - max_flt*(float)n;
    max_flt /= 10.;
  }
  return 0;
}

char float_to_str(float flt, char * str, char max_len_str, char digit_int, char digit_frac, bool sign, char null_char)
{
  byte_t i;
  float max_flt;
  char rc;

  if (str == 0)
    return -1;
  memset(str,0,max_len_str);

  if ( digit_int==0 ) {
    max_flt=flt;
    digit_int=1;
    while ( 1 ) {
      max_flt/=10.;
      if ( max_flt<1. )
        break;
      digit_int++;
    }
  }

  max_flt=calc_max_flt(digit_int);

  if (flt >= max_flt || flt <= (-1.)*max_flt)
    return -1;

  i = 0;
  if (sign == true) {
    if (flt<0) {
      str[i] = '-'; flt *= -1.;
    }
    else
      str[i] = '+';
    i++;
  }
  else {
    if (flt<0) {
      str[i] = '-';  flt *= -1.; i++;
    }
  }

  rc=fill_str_digit(&flt,str,&i,max_len_str,max_flt,null_char);
  if ( rc!=0 )
    return -1;

  if (digit_frac == 0) {
    str[i] = 0;
    return 0;
  }

  str[i] = '.';
  i++;

  max_flt=calc_max_flt(digit_frac);
  flt *= max_flt;

  rc=fill_str_digit(&flt,str,&i,max_len_str,max_flt,null_char);
  if ( rc!=0 )
    return -1;

  str[i] = 0;
  return 0;
}

 

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

nik182 пишет:

Но даже сделав все вычисления целочисленными fixed point в конце вы получае не целое а действительное число с фиксированным количеством десятичных знаков, которому соответствует тип float.

Отнюдь.

Fixed тем и отличается от float, что у первого фиксированное количество дробных знаков, а у второго - фиксированное общее количество знаков (как правило, двоичных, а не десятичных).

Цитата:

Если вы выведете полученное значение fixed point без дополнительных телодвижений вы получите число без десятичной точки.

Мне как-то даже неудобно напоминать, что если Вы используете свой тип даннх, то и для его вывода на печать следует писать свой код. Неужели это кому-то не очевидно?

Цитата:

Т.е в конце из целого fixed надо представить float, которое целый fixed подразумевает.

Еще раз: с чего Вы предположили, что fixed подразумевает float?

Цитата:

Тут конечно можно долго спорить есть точка в fixed или нет.

У Вас есть сомнения по поводу того, есть ли в типе данных с названием "fixed point" точка?

 

Похоже, мы с Вами существуем в различных Вселенных...

nik182
Offline
Зарегистрирован: 04.05.2015

Действительно в разных вселенных. В моей примером fixed point можно привести подсчёты рублей выражая их через копейки и считая копейки, всегда иметь рубли с точностью до двух десячичных знаков после запятой. Но числа которыми мы будем при этом оперировать всегда целые. 

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

Ну, вариант с рублями и копейками, он нагляден, но все-таки это не совсем fixed point, т.к. fixed point подразумевает дробные числа, а копейки - всегда целые (рубли - те же копейки, но с масштабным множителем).

Кроме того, fixed point обычно использует двоичную систему счисления, а число знаков после точки определяется максимально эффективным использованием разрядной сетки. Ну, например, если нам нужно измерять длины до 10 м, то разумно 4 разряда отвести под целые, а оставшиеся 4(12) в 8(16)-разрядном числе - под дробную часть. Тогда единица младшего разряда составит 62.5(0.24)мм.

И еще: вопрос вывода чисел fixed point в строку (на экран, на принтер...) к самому представлению отношения не имеет, а при необходимости рассматривается последним - когда все прочие характеристики уже определены.

nik182
Offline
Зарегистрирован: 04.05.2015

Да какая разница. Двоичные,  десятичные, важен принцип. Обработка одного числа, без деления на мантиссу и порядок.

andriano
andriano аватар
Offline
Зарегистрирован: 20.06.2015

nik182 пишет:

Да какая разница. Двоичные,  десятичные, важен принцип. Обработка одного числа, без деления на мантиссу и порядок.

Основной принцип отражен в названии - число с фиксированной точкой. Т.е. целая и дробная части имеют фиксированную длину. Как обрабатывать - вопрос отдельный. Десятичные обрабатывать на двоичной машине невозможно - только на двоично-десятичной (или десятичной, если такая экзотика где встретится) И, кстати, мантисса и порядок - это тоже одно число. И, соответственно, обрабатывать тоже можно по-разному.

voha888
Offline
Зарегистрирован: 10.03.2020

dimax пишет:

Ещё как вариант разбить через modf() на целую и дробную части

Спасибо!

Правда сперва не понял как работает, но потом сколхозил такое для вывода температуры с точностью до сотых:

(uint8_t)(modff(value,NULL)*100)
DetSimen
DetSimen аватар
Offline
Зарегистрирован: 25.01.2017

voha888 пишет:

потом сколхозил такое для вывода температуры с точностью до сотых:

(uint8_t)(modff(value,NULL)*100)

Ты действительно можешь шкурой отличить +5 от +5.15°С ?