Openbox ошибка программирования

  • Назад
  • 1
  • 2
  • 3
  • 4
  • 5
  • Вперёд
  • Страница 4 из 5  

Рекомендованные сообщения

У меня не бизнес для своей авто пользуюсь в год один раз

Тогда ищи альтернативу или обратись раз в год к мастеру.

Рой ( автор проги ) так же тратит на то время и занимается много с прогами . Так что за работу так же хотел бы получить денег. Ты то сам не будешь бесплатно всеми помогать))) Уважать надо чужой труд!

И заполнять профиль в соответствии с правилами форума.


Изменено 4 июля 2014 пользователем Klassikovod

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

шьет 7.9.7+ по диагностке без пайки, правильно ?

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

А отломанная версия прошьет м 10.3?

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Это уже лень халяву открыть и посмотреть? Ребята! Зажрались уже….

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Я задал простой вопрос. Эксперементировать при клиенте не хочется

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

ПЫтаюсь прошить Me7.9.7 по диагностике, пишет следующее: «Условия не корректны или ошибка последовательности запроса. Ошибка программирования.» Что за нафиг? Уже за прошивку бабки отвалил. Идентификацию проходит, подключается без проблем, а заливать фиг там.

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

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

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

 Что за нафиг? Уже за прошивку бабки отвалил. Идентификацию проходит, подключается без проблем, а заливать фиг там.

Теперь придётся ещё бабок отвалить и за нормальный загрузчик :emot64: . Ну или дать тем, у кого он есть.

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

ПЫтаюсь прошить Me7.9.7 по диагностике, пишет следующее: «Условия не корректны или ошибка последовательности запроса. Ошибка программирования.» Что за нафиг? Уже за прошивку бабки отвалил. Идентификацию проходит, подключается без проблем, а заливать фиг там.

прочти хелп там все рассказано,прога работает без проблем 

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Да читал я хелп. Подключаю кабель к разъему и ноуту. Включаю прогу, выбираю блок и порт. Включаю зажигание, жду секунд 15-20 и подключаюсь. Идентификация идет, ошибки просматривает, выбираю прошивку — ошибка вышеупомянутая. Даже если сразу пытаюсь залить все равно ошибка, следом делаю идентификацию она проходит. Что не так делаю?

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Читал про различие блоков м7.9.7 и ме7.9.7 ? Почитай хелп по чиплодырю .

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Вчера Киа спектра Бош м7.9.7 по диагностике иденты отдал, флеш потер, а запись завершилось ошибкой, блин не помню какой))), разобрал в бсл записал без проблем.

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Причём тут отличия м и ме? Если заливаться тупо через кабель должно без всяких разрешений.

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

F

Ребят подскажите пожалуйста смогу ли я прошить этой программой м73 автел закрытый

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

F

Ребят подскажите пожалуйста смогу ли я прошить этой программой м73 автел закрытый

Да!

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах

За всех- пжалыста!

м797 взьмёт-да!

ме 797 возьмёт-да!

м797+                 -да!

ме797+                 -да!

м797-                    -да!

ме797-                   -да!

я7.2+                   -да!

я7.2-                     -да!

м73+                      -да! нет такого                -да!

и всем колллллллективное спасибо — да!

парни вы чё?????

рулетка        — да!!!!

Вы извените,но достало!!!!


Изменено 9 октября 2015 пользователем vladgl


  • Like


    1

Поделиться сообщением


Ссылка на сообщение
Поделиться на других сайтах
  • Назад
  • 1
  • 2
  • 3
  • 4
  • 5
  • Вперёд
  • Страница 4 из 5  

Создайте аккаунт или войдите в него для комментирования

Вы должны быть пользователем, чтобы оставить комментарий

Cегодня вальнул A373DA02,версия 3.2.1 началось с записи в блок чистого еепрома,

считывание прошло успешно,через 81-пин разъём блока.Затем запись чистого еепрома

питание от АКБ, блок успешно вышел на связь,потом появилась версия булоадера 2.05

потом строка записи….дальше Ошибка программирования.

Попробовал снова еепром считать,чтение прошло нормально,стал опять записывать,

определяется версия бутлоадера,запись идёт..и опять ошибка программирования.

При этом блок ещё живой,выдаёт идентификаторы.Попробовал записать флешь,запись прошла нормально,

но заметил как, то долго после успешной записи флешь, долго не появляется строка Завершено..

После длительного ожидания всё таки появилась. Блок Чек не зажигает стартера нет, повтор записи

флешь,и, чек горит,но двигатель не пускается,….Третий раз блок на связь не выходит..

Думаю если прошивал бы только флешь,проблем бы не было,наверно засада кроется в записи Епрома.

  • Назад
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • Вперёд
  • Страница 5 из 21  

Рекомендованные сообщения

napoxoq

    • Поделиться

#121

Ну так пусть человек попробует, вдруг у него звезды счастливые :)

Никак он не хочет понять что нет бесплатных решений. И платных кот наплакал.

Ссылка на сообщение

waleko

    • Поделиться

#122

А с ChipLoaderом SMS-ский адаптер дружит?

Нет.

По моему автор Энигмы Эммибокс (Emmibox).

И её можно найти свободно на других сайтах.

Ссылка на сообщение

Саша

napoxoq

    • Поделиться

#124

И нам подскажи, мы толпой ломанемся :pleasantry:.

жаль токма нету нигде :search:


Изменено 28 октября 2010 пользователем napoxoq

Ссылка на сообщение

AleXandr

    • Поделиться

#125

Есть,только не продают….и продавать не собираются.

Ссылка на сообщение

valkaz

    • Поделиться

#126

Правильно, но требование к адаптеру одно — нормальная работа на скорости 57600. У меня прекрасно работают с опеном адаптеры от Menen’a и SMS-software

значит флешер- адакт возмет

Ссылка на сообщение

Горький70

    • Поделиться

#127

Спектра 2007 через дилерский разъём прошил легко.

Ссылка на сообщение

autovic

    • Поделиться

#128

Там разъём КИА?Подскажите, к каким контактам подключались?

Ссылка на сообщение

msa96

int2146655

    • Поделиться

#130

Вышла версия 3.1.8, обновляемся через прогу.

Ссылка на сообщение

Scaner

    • Поделиться

#131

каждую новыю версию нужно заного регестрировать?

Ссылка на сообщение

Саша

    • Поделиться

#132

нет. Там есть файлик редми. Там написано как делать при обновлении.

Ссылка на сообщение

Vasiliy Armeev

  • Автор
    • Поделиться

#133

нет, не надо, переименуйте файл OpenBox317.ini в OpenBox.ini

и все заработает. В будущем операцию проделывать будет не надо.

Ссылка на сообщение

GENNA

    • Поделиться

#134

А как с обновлениями, я например пишу письмо с просьбой выслать, ну и высылают. Это так и должно быть? Или я в бронепоезде.

Ссылка на сообщение

Саша

    • Поделиться

#135

выходишь в сеть с компа где установлена прога. Запускаешь опенбокс, там есть вкладка — обновление on-line. Все.

Ссылка на сообщение

uri

    • Поделиться

#136

 А меня такой вкладки нету……………  

Ага теперь есть  :rolleyes:


Изменено 8 ноября 2010 пользователем uri

Ссылка на сообщение

msa96

    • Поделиться

#137

Значит версия старая, начиная с 3,16,20 есть.

Ссылка на сообщение

GENNA

    • Поделиться

#138

О. Заработало. Надо все таки внимательнее быть.

Ссылка на сообщение

kutun

    • Поделиться

#139

Парни, я вот думку думаю… В КТС от БОШа с начала 2010 года поддерживается ПАСС ТРУ, — — — — — КТСы имею, нет никого рядом со мной с ОПЕНом, а то попробовали б, отписались, отчитались. А так ОПЕН недельки через две возьму, проверю, как БОШ свои заявки оправдывает.

а простите это о чем речь?можно поподробней для танкистов? вроде ктс пользую от и до со всеми обнавлениями а о чем речь не понял!


Изменено 8 ноября 2010 пользователем kutun

Ссылка на сообщение

yiri

    • Поделиться

#140

на 797 да, на М73 иди найди.

И как выходить из положения?Прошивки, которые в продаже, ведь в формате комбика или по просьбе трудяшихся будет рассылка не криптованных?

Пока OpenBox-ом выходит залить только из современных 7.9.7+,а М73 нечем раскриптовать: подскажите парни кто в теме.

Ссылка на сообщение

GENNA

    • Поделиться

#141

Ну аверное Вы не правы. В магазине Адакта декрипт, т.е. боксом можно грузить и поделками от СМС тоже можно.

Ссылка на сообщение

napoxoq

    • Поделиться

#142

Прошивки, которые в продаже, ведь в формате комбика или по просьбе трудяшихся будет рассылка не криптованных?

@yiri, По диагонали читаем?


Изменено 14 ноября 2010 пользователем napoxoq

Ссылка на сообщение

GENNA

    • Поделиться

#143

Перестал работать Опенбокс. Обновление до 3.1.8 сделал, вроде бы открывался но проверить не на чем было. И вот сегодня сообщение—- приложение просрочено, пож. посетите наш веб сайт—-. Ну и не открывается ест-но. Ну и че с ним делать то?

Может конечно бокс и крут, но что то благодаря ему у меня уже минус 6 к.руб. из за потери клиентов, причем не по моей вине а по косякам с ним.

Ссылка на сообщение

gory

    • Поделиться

#144

Или туплю, или что говорит ROY ?

P.S. Протупил, но в свете последних веяний.


Изменено 20 ноября 2010 пользователем gory

Ссылка на сообщение

GENNA

    • Поделиться

#145

приложение просрочено, пож. посетите наш веб сайт — это говорит роевская прога.

А вот сам Радион пока не отписался.

Ссылка на сообщение

GENNA

    • Поделиться

#146

Ну вот и сам отписался.

Здравствуйте.

или переустановить систему.

или ждать след. версию

или пользоваться предыдущей

Переустановить как то весьма проблемно.

Откатил к старой версии, но тут совсем проблемы….

Файл: A327SL08_V4.5.bin

Ошибка КС.

Хотя это совсем странно, именно этот файл записан лодырем. Файлик наш с размером все ОК. Должен писаться, однако нет.

Ссылка на сообщение

waleko

    • Поделиться

#147

На окте по этому поводу написали так:

«Сравните в файле OpenBox.ini строку key. В 3.16 ,3.17 и 3.18 — значение должно быть одинаковое.«

Я сравнил, у меня одинаковые.

Ссылка на сообщение

Горький70

    • Поделиться

#148

Я сравнивал у меня разные исправил пока не проверял.

Ссылка на сообщение

int2146655

    • Поделиться

#149

А я в пятницу вечером, при помощи опенбокса успешно завалил М73 A317DB04 :pleasantry:

дело было так:

1. подключил опенбокс,

2. считал прошивку,

3. загрузил A317KA07_V4.5_E3_RON_95, пошла очистка флеш и после ошибка программирования

вот и все :angel:

PS где проц купить? яндексом пошарил, дорого что то везде


Изменено 21 ноября 2010 пользователем int2146655

Ссылка на сообщение

Саша

    • Поделиться

#150

А я так и не смог его запустить на трех компах.

Иденты читает и все. А дальше — общая ошибка программирования.

На нескольких адаптерах пробовал, которые работают везде без проблем.

СМС, Чипсофт, галетто и опенпорт2.0.

Ладно, до января оставил, там в Тагиле парни замучают его думаю до конца.

Ссылка на сообщение

  • Назад
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • Вперёд
  • Страница 5 из 21  

auto-mat [ТопикСтартер]

Возраст: 46 лет

Перейти

Перейти к сообщениям

ТопикCтартер

|
Сообщения автора
Награда за ответ
|по убыванию
|Режим чтения


Для просмотра нужна авторизация!

Для просмотра Вам необходимо авторизироваться.
Если Вы еще не зарегистрированы, перейдите по ссылке: Регистрация.

x

Вчера не смог без доработки Опендиагфлешером прошить Бош 7.9.7+. Доходит то до половины, то в начале то в конце и обрывается связь и выходит «ОШИБКА ПРОГРАММИРОВАНИЯ». Поменял 3 к-лайн адаптера на железный ком-порт и такая же ерунда. Затем подумал а не в ком-порте ли дело? Взял USB ВАГ-КОМ ккл-адаптер и 7 раз подряд прошил уже им. Результат ни одного сбоя. Затем снова подключаю от ком-порта, доходит примерно до середины и вновь «ОШИБКА ПРОГРАММИРОВАНИЯ». А я считал ошибочно физический ком-порт более стабильным. Скорее всего причиной аналогичного сбоя в соседней теме по М74К  https://avtomastera.net/forum.php?mod=viewthread&tid=951 является то-же самое

Последние посетители

  • СергейMV

    2023-06-01

  • sergey_tesak

    2023-05-25

ИзбранноеИзбранное
ТрансляцииТрансляции
ЗакладкиЗакладки
ЗакладкиКоллекции
++3
-


Ответ

Реквизит
Жалоба

Mat

Возраст: 41 лет

Второй


Опубликовано 20-8-2015 08:56:27
|
Сообщения автора

Все от проводов зависит: экранирование, длина.

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

auto-mat [ТопикСтартер]

Возраст: 46 лет

Третий


Опубликовано 20-8-2015 10:11:05
|
Сообщения автора

Подключил вот такой адаптер на чипе FTDI работает стабильно…

USB-OBD2-LITE-409-II-VAG-KKL.jpg

Войдите/Зарегистрируйтесь, чтобы увеличить

Прокомментировать
Свернуть


Ответ
+ 1
0

Реквизит
Жалоба

TNT

Четвертый


Опубликовано 26-10-2015 08:12:42
|
Сообщения автора

Тоже столкнулся с проблемой прошивки Bosch 7.9.7+. OpenBox не видел вообще никак этот блок (подключал «разрешение программирования» на 23 ногу вместо 43, а потом и вместе, химичил по-разному). К стати, при выборе прошивки от Ледокола B104CR(LR)02_v7.4.bin программа ругалась: «Ошибка открытия файла»  Пришлось перепаивать резистор и шить ChipLoader 1.97.7. Блок сразу вышел на связь. Записал это B104CR(LR)02_v7.4.bin.  ChipLoader не ругался на файл. Поставил блок, авто запустилось. Только в OpenDiag почему-то пишет ту же прошивку B103CU03, но красными буквами- «Не родное программное обеспечение». Теперь есть сомнения, точно ли прошивка тюн записалась. K-line такой как выше Vag KKL. До этого прошил блок М73 и М74. А с этим Bosch 7.9.7+ помучался.

Больше фотографий
Миниатюра
Полный размер

Фотографии открывыются, пожалуйста, подождите…

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

auto-mat [ТопикСтартер]

Возраст: 46 лет

5#


Опубликовано 26-10-2015 09:44:23
|
Сообщения автора

добавил TNT в 26-10-2015 11:12
Тоже столкнулся с проблемой прошивки Bosch 7.9.7+. OpenBox не видел вообще никак этот блок (подключа

Чтобы Опенбокс работал с этими блоками без доработки нужно сначала подать 23-й пин на массу, затем подать все питания на эбу, ВНИМАНИЕ И ТОЛЬКО ПОСЛЕ ЭТОГО открывать программу и выбирать номер ком-порта, прошивку и т.д. После такого порядка Опенбокс как и Опендиаг-флешер шьет 7.9.7+ без доработки.

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

demon8225

Возраст: 41 лет

6#


Опубликовано 31-10-2015 10:25:28
|
Сообщения автора

добавил TNT в 26-10-2015 08:12
Тоже столкнулся с проблемой прошивки Bosch 7.9.7+. OpenBox не видел вообще никак этот блок (подключа

Пытался писать декрипт скорее всего. Опен бокс их не пишет, поэтому и ругался. Чип лодырь берет.

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

sanich

Возраст: 38 лет

7#


Опубликовано 19-12-2015 14:47:47
|
Сообщения автора

сегодня бош 797+ не пошел через бсл с 43 ногой ни OpenDiagFlasher_ом ни OpenBox_ом Bosh 7.9.7+ Ошибка программирования.???

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

Ruslan

Возраст: 39 лет

8#


Опубликовано 24-12-2015 20:04:59
С мобилы
|
Сообщения автора

Здравствуйте!!! А есть ли какой нибудь небольшой софт для бош 797+ чтобы тупо менять только температуру вкл.вентилятора?

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

auto-mat [ТопикСтартер]

Возраст: 46 лет

9#


Опубликовано 25-12-2015 07:16:03
|
Сообщения автора

добавил Ruslan в 24-12-2015 23:04
Здравствуйте!!! А есть ли какой нибудь небольшой софт для бош 797+ чтобы тупо менять только температ

Вопрос не совсем в тему, но есть редактор калибровок от Мовyх который это делает. Проект пока обновляется, но старая версия программы это делала именно с Бош-7.9.7+

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

Mat

Возраст: 41 лет

10#


Опубликовано 25-12-2015 09:01:42
|
Сообщения автора

добавил Ruslan в 24-12-2015 20:04
Здравствуйте!!! А есть ли какой нибудь небольшой софт для бош 797+ чтобы тупо менять только температ

Модуль М73 для редактора ECU.
https://avtomastera.net/forum.ph … =1368&fromuid=2
(Источник: АвтоМастера.нет)

Прокомментировать
Свернуть


Ответ
+

Реквизит
Жалоба

Существует две фундаментальные стратегии: обработка исправимых ошибок (исключения, коды возврата по ошибке, функции-обработчики) и неисправимых (assert(), abort()). В каких случаях какую стратегию лучше использовать?

Виды ошибок

Ошибки возникают по разным причинам: пользователь ввёл странные данные, ОС не может дать вам обработчика файла или код разыменовывает (dereferences) nullptr. Каждая из описанных ошибок требует к себе отдельного подхода. По причинам ошибки делятся на три основные категории:

  • Пользовательские ошибки: здесь под пользователем подразумевается человек, сидящий перед компьютером и действительно «использующий» программу, а не какой-то программист, дёргающий ваш API. Такие ошибки возникают тогда, когда пользователь делает что-то неправильно.
  • Системные ошибки появляются, когда ОС не может выполнить ваш запрос. Иными словами, причина системных ошибок — сбой вызова системного API. Некоторые возникают потому, что программист передал системному вызову плохие параметры, так что это скорее программистская ошибка, а не системная.
  • Программистские ошибки случаются, когда программист не учитывает предварительные условия API или языка программирования. Если API требует, чтобы вы не вызывали foo() с 0 в качестве первого параметра, а вы это сделали, — виноват программист. Если пользователь ввёл 0, который был передан foo(), а программист не написал проверку вводимых данных, то это опять же его вина.

Каждая из описанных категорий ошибок требует особого подхода к их обработке.

Пользовательские ошибки

Сделаю очень громкое заявление: такие ошибки — на самом деле не ошибки.

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

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

Конечно, такое не всегда возможно. Иногда проверять вводимые данные слишком дорого, иногда это не позволяет сделать архитектура кода или разделение ответственности. Но в таких случаях ошибки должны обрабатываться однозначно как исправимые. Иначе, допустим, ваша офисная программа будет падать из-за того, что вы нажали backspace в пустом документе, или ваша игра станет вылетать при попытке выстрелить из разряженного оружия.

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

Системные ошибки

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

Но как их обрабатывать, как исправимые или неисправимые?

Это зависит от обстоятельств.

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

Но падение программы из-за того, что ОС не может выделить сокет, — это не слишком дружелюбное поведение. Так что лучше бросить исключение и позволить catch аккуратно закрыть программу.

Но бросание исключения — не всегда правильный выбор.

Кто-то даже скажет, что он всегда неправильный.

Если вы хотите повторить операцию после её сбоя, то обёртывание функции в try-catch в цикле — медленное решение. Правильный выбор — возврат кода ошибки и цикличное исполнение, пока не будет возвращено правильное значение.

Если вы создаёте вызов API только для себя, то просто выберите подходящий для своей ситуации путь и следуйте ему. Но если вы пишете библиотеку, то не знаете, чего хотят пользователи. Дальше мы разберём подходящую стратегию для этого случая. Для потенциально неисправимых ошибок подойдёт «обработчик ошибок», а при других ошибках необходимо предоставить два варианта развития событий.

Обратите внимание, что не следует использовать подтверждения (assertions), включающиеся только в режиме отладки. Ведь системные ошибки могут возникать и в релизной сборке!

Программистские ошибки

Это худший вид ошибок. Для их обработки я стараюсь сделать так, чтобы мои ошибки были связаны только с вызовами функций, то есть с плохими параметрами. Прочие типы программистских ошибок могут быть пойманы только в runtime, с помощью отладочных макросов (assertion macros), раскиданных по коду.

При работе с плохими параметрами есть две стратегии: дать им определённое или неопределённое поведение.

Если исходное требование для функции — запрет на передачу ей плохих параметров, то, если их передать, это считается неопределённым поведением и должно проверяться не самой функцией, а оператором вызова (caller). Функция должна делать только отладочное подтверждение (debug assertion).

С другой стороны, если отсутствие плохих параметров не является частью исходных требований, а документация определяет, что функция будет бросать bad_parameter_exception при передаче ей плохого параметра, то передача — это хорошо определённое поведение (бросание исключения или любая другая стратегия обработки исправимых ошибок), и функция всегда должна это проверять.

В качестве примера рассмотрим получающие функции (accessor functions) std::vector<T>: в спецификации на operator[] говорится, что индекс должен быть в пределах валидного диапазона, при этом at() сообщает нам, что функция кинет исключение, если индекс не попадает в диапазон. Более того, большинство реализаций стандартных библиотек обеспечивают режим отладки, в котором проверяется индекс operator[], но технически это неопределённое поведение, оно не обязано проверяться.

Примечание: необязательно бросать исключение, чтобы получилось определённое поведение. Пока это не упомянуто в исходных условиях для функции, это считается определённым. Всё, что прописано в исходных условиях, не должно проверяться функцией, это неопределённое поведение.

Когда нужно проверять только с помощью отладочных подтверждений, а когда — постоянно?

К сожалению, однозначного рецепта нет, решение зависит от конкретной ситуации. У меня есть лишь одно проверенное правило, которому я следую при разработке API. Оно основано на наблюдении, что проверять исходные условия должен вызывающий, а не вызываемый. А значит, условие должно быть «проверяемым» для вызывающего. Также условие «проверяемое», если можно легко выполнить операцию, при которой значение параметра всегда будет правильным. Если для параметра это возможно, то это получается исходное условие, а значит, проверяется только посредством отладочного подтверждения (а если слишком дорого, то вообще не проверяется).

Но конечное решение зависит от многих других факторов, так что очень трудно дать какой-то общий совет. По умолчанию я стараюсь свести к неопределённому поведению и использованию только подтверждений. Иногда бывает целесообразно обеспечить оба варианта, как это делает стандартная библиотека с operator[] и at().

Хотя в ряде случаев это может быть ошибкой.

Об иерархии std::exception

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

Я предлагаю наследовать только от одного из этих четырёх классов:

  • std::bad_alloc: для сбоев выделения памяти.
  • std::runtime_error: для общих runtime-ошибок.
  • std::system_error (производное от std::runtime_error): для системных ошибок с кодами ошибок.
  • std::logic_error: для программистских ошибок с определённым поведением.

Обратите внимание, что в стандартной библиотеке разделяются логические (то есть программистские) и runtime-ошибки. Runtime-ошибки — более широкое определение, чем «системные». Оно описывает «ошибки, обнаруживаемые только при выполнении программы». Такая формулировка не слишком информативна. Лично я использую её для плохих параметров, которые не являются исключительно программистскими ошибками, а могут возникнуть и по вине пользователей. Но это можно определить лишь глубоко в стеке вызовов. Например, плохое форматирование комментариев в standardese приводит к исключению при парсинге, проистекающему из std::runtime_error. Позднее оно ловится на соответствующем уровне и фиксируется в логе. Но я не стал бы использовать этот класс иначе, как и std::logic_error.

Подведём итоги

Есть два пути обработки ошибок:

  • как исправимые: используются исключения или возвращаемые значения (в зависимости от ситуации/религии);
  • как неисправимые: ошибки журналируются, а программа прерывается.

Подтверждения — это особый вид стратегии обработки неисправимых ошибок, только в режиме отладки.

Есть три основных источника ошибок, каждый требует особого подхода:

  • Пользовательские ошибки не должны обрабатываться как ошибки на верхних уровнях программы. Всё, что вводит пользователь, должно проверяться соответствующим образом. Это может обрабатываться как ошибки только на нижних уровнях, которые не взаимодействуют с пользователями напрямую. Применяется стратегия обработки исправимых ошибок.
  • Системные ошибки могут обрабатываться в рамках любой из двух стратегий, в зависимости от типа и тяжести. Библиотеки должны работать как можно гибче.
  • Программистские ошибки, то есть плохие параметры, могут быть запрещены исходными условиями. В этом случае функция должна использовать только проверку с помощью отладочных подтверждений. Если же речь идёт о полностью определённом поведении, то функции следует предписанным образом сообщать об ошибке. Я стараюсь по умолчанию следовать сценарию с неопределённым поведением и определяю для функции проверку параметров лишь тогда, когда это слишком трудно сделать на стороне вызывающего.

Гибкие методики обработки ошибок в C++

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

Это относительно легко сделать в высокоуровневых функциях. Вы точно знаете, почему что-то пошло не так, и можете обработать это соответствующим образом. Но в случае с низкоуровневыми функциями всё не так просто. Они не знают, что пошло не так, они знают лишь о самом факте сбоя и должны сообщить об этом тому, кто их вызвал.

В C++ есть два основных подхода: коды возврата ошибок и исключения. Сегодня широко распространено использование исключений. Но некоторые не могут / думают, что не могут / не хотят их использовать — по разным причинам.

Я не буду принимать чью-либо сторону. Вместо этого я опишу методики, которые удовлетворят сторонников обоих подходов. Особенно методики пригодятся разработчикам библиотек.

Проблема

Я работаю над проектом foonathan/memory. Это решение предоставляет различные классы выделения памяти (allocator classes), так что в качестве примера рассмотрим структуру функции выделения.

Для простоты возьмём malloc(). Она возвращает указатель на выделяемую память. Если выделить память не получается, то возвращается nullptr, то есть NULL, то есть ошибочное значение.

У этого решения есть недостатки: вам нужно проверять каждый вызов malloc(). Если вы забудете это сделать, то выделите несуществующую память. Кроме того, по своей натуре коды ошибок транзитивны: если вызвать функцию, которая может вернуть код ошибки, и вы не можете его проигнорировать или обработать, то вы тоже должны вернуть код ошибки.

Это приводит нас к ситуации, когда чередуются нормальные и ошибочные ветви кода. Исключения в таком случае выглядят более подходящим решением. Благодаря им вы сможете обрабатывать ошибки только тогда, когда вам это нужно, а в противном случае — достаточно тихо передать их обратно вызывающему.

Это можно расценить как недостаток.

Но в подобных ситуациях исключения имеют также очень большое преимущество: функция выделения памяти либо возвращает валидную память, либо вообще ничего не возвращает. Это функция «всё или ничего», возвращаемое значение всегда будет валидным. Это полезное следствие согласно принципу Скотта Майера «Make interfaces hard to use incorrectly and easy to use correctly».

Учитывая вышесказанное, можно утверждать, что вам следует использовать исключения в качестве механизма обработки ошибок. Этого мнения придерживается большинство разработчиков на С++, включая и меня. Но проект, которым я занимаюсь, — это библиотека, предоставляющая средства выделения памяти, и предназначена она для приложений, работающих в реальном времени. Для большинства разработчиков подобных приложений (особенно для игроделов) само использование исключений — исключение.

Каламбур детектед.

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

Так что же делать?

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

К счастью, я могу определить, что вы делаете, когда обнаруживаете ошибку нехватки памяти: чаще всего вы журналируете это событие и прерываете программу, поскольку она не может корректно работать без памяти. В таких ситуациях исключения — просто способ передачи контроля другой части кода, которая журналирует и прерывает программу. Но есть старый и эффективный способ передачи контроля: указатель функции (function pointer), то есть функция-обработчик (handler function).

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

Я называю такой подход обработкой исключений и придерживаюсь его при работе с памятью.

Решение 1: обработчик исключений

Если вам нужно обработать ошибку в условиях, когда наиболее распространённым поведением будет «журналировать и прервать», то можно использовать обработчика исключений. Это такая функция-обработчик, которая вызывается вместо бросания объекта-исключения. Её довольно легко реализовать даже в уже существующем коде. Для этого нужно поместить управление обработкой в класс исключений и обернуть в макрос выражение throw.

Сначала дополним класс и добавим функции для настройки и, возможно, запрашивания функции-обработчика. Я предлагаю делать это так же, как стандартная библиотека обрабатывает std::new_handler:

class my_fatal_error
{
public:
    // тип обработчика, он должен брать те же параметры, что и конструктор,
    // чтобы у них была одинаковая информация
    using handler = void(*)( ... );

    // меняет функцию-обработчика
    handler set_handler(handler h);

    // возвращает текущего обработчика
    handler get_handler();

    ... // нормальное исключение
};

Поскольку это входит в область видимости класса исключений, вам не нужно именовать каким-то особым образом. Отлично, нам же легче.

Если исключения включены, то для удаления обработчика можно использовать условное компилирование (conditional compilation). Если хотите, то также напишите обычный подмешанный класс (mixin class), дающий требуемую функциональность.

Конструктор исключений элегантен: он вызывает текущую функцию-обработчика, передавая ей требуемые аргументы из своих параметров. А затем комбинирует с последующим макросом throw:

If```cpp #if EXCEPTIONS #define THROW(Ex) throw (Ex) #else #define THROW(Ex) (Ex), std::abort() #endif
> Такой макрос throw также предоставляется [foonathan/compatiblity](https://github.com/foonathan/compatibility).

Можно использовать его и так:

```cpp
THROW(my_fatal_error(...))

Если у вас включена поддержка исключений, то будет создан и брошен объект-исключение, всё как обычно. Но если поддержка выключена, то объект-исключение всё равно будет создан, и — это важно — только после этого произойдёт вызов std::abort(). А поскольку конструктор вызывает функцию-обработчика, то он и работает, как требуется: вы получаете точку настройки для журналирования ошибки. Благодаря же вызову std::abort() после конструктора пользователь не может нарушить постусловие.

Когда я работаю с памятью, то при включённых исключениях у меня также включён и обработчик, который вызывается при бросании исключения.

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

А если я хочу продолжить работу после бросания исключения?

Методика с обработчиком исключений не позволяет этого сделать в связи с постусловием кода. Как же тогда продолжить работу?

Ответ прост — никак. По крайней мере, это нельзя сделать так же просто, как в других случаях. Нельзя просто так вернуть код ошибки вместо исключения, если функция на это не рассчитана.
Есть только одно решение: сделать две функции. Одна возвращает код ошибки, а вторая бросает исключения. Клиенты, которым нужны исключения, будут использовать второй вариант, остальные — первый.

Извините, что говорю такие очевидные вещи, но ради полноты изложения я должен был об этом сказать.

Для примера снова возьмём функцию выделения памяти. В этом случае я использую такие функции:

void* try_malloc(..., int &error_code) noexcept;

void* malloc(...);

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

void* malloc(...)
{
    auto error_code = 0;
    auto res = try_malloc(..., error_code);
    if (!res)
        throw malloc_error(error_code);
    return res;
}

Не делайте этого в обратной последовательности, иначе вам придётся ловить исключение, а это дорого. Также это не даст нам скомпилировать код без включённой поддержки исключений. Если сделаете, как показано, то можете просто стереть другую перегрузку (overload) с помощью условного компилирования.

Но даже если у вас включена поддержка исключений, клиенту всё равно может понадобиться вторая версия. Например, когда нужно выделить наибольший возможный объём памяти, как в нашем примере. Будет проще и быстрее вызывать в цикле и проверять по условию, чем ловить исключение.

Решение 2: предоставить две перегрузки

Если недостаточно обработчика исключений, то нужно предоставить две перегрузки. Одна использует код возврата, а вторая бросает исключение.

Если рассматриваемая функция не имеет возвращаемого значения, то можете её использовать для кода ошибки. В противном случае вам придётся возвращать недопустимое значение для сигнализирования об ошибке — как nullptr в вышеприведённом примере, — а также установить выходной параметр для кода ошибки, если хотите предоставить вызывающему дополнительную информацию.

Пожалуйста, не используйте глобальную переменную errno или что-то типа GetLastError()!

Если возвращаемое значение не содержит недопустимое значение для обозначения сбоя, то по мере возможности используйте std::optional или что-то похожее.

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

std::system_error

Подобная система идеально подходит для работы с кодами ошибок в С++ 11.

Она возвращает непортируемый (non-portable) код ошибки std::error_code, то есть возвращаемый функцией операционной системы. С помощью сложной системы библиотечных средств и категорий ошибок вы можете добавить собственные коды ошибок, или портируемые std::error_condition. Для начала почитайте об этом здесь. Если нужно, то можете использовать в функции кода ошибки std::error_code. А для функции исключения есть подходящий класс исключения: std::system_error. Он берёт std::error_code и применяется для передачи этих ошибок в виде исключений.

Эту или подобную систему должны использовать все низкоуровневые функции, являющиеся закрытыми обёртками ОС-функций. Это хорошая — хотя и сложная — альтернатива службе кодов ошибок, предоставляемой операционной системой.

Да, и мне ещё нужно добавить подобное в функции виртуальной памяти. На сегодняшний день они не предоставляют коды ошибок.

std::expected

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

А глобальные переменные вообще не вариант!

В № 4109 предложено решение: std::expected. Это шаблон класса, который также хранит возвращаемое значение или код ошибки. В вышеприведённом примере он мог бы использоваться так:

std::expected<void*, std::error_code> try_malloc(...);

В случае успеха std::expected будет хранить не-null указатель памяти, а при сбое — std::error_code. Сейчас эта методика работает при любых возвращаемых значениях. Комбинация std::expected и функции исключения определённо допускает любые варианты использования.

Заключение

Если вы создаёте библиотеки, то иногда приходится обеспечивать максимальную гибкость использования. Под этим подразумевается и разнообразие средств обработки ошибок: иногда требуются коды возврата, иногда — исключения.

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

Более гибкий подход — просто предоставить две перегрузки, одну с исключениями, а вторую без. Это даст пользователям максимальную свободу, они смогут выбирать ту версию, что лучше подходит в их ситуации. Недостаток этого подхода: вам придётся больше потрудиться при создании библиотеки.

Понравилась статья? Поделить с друзьями:
  • Openall32 dll ошибка windows 10
  • Openall32 dll ошибка grid
  • Openal32 dll ошибка stalker
  • Openal32 dll отсутствует как исправить ошибку
  • Openal32 dll код ошибки