Tools: Remove whitespaces at the end of lines in Visual Studio

Известный поисковый запрос, вынесенный в заголовок — одна из самых навязчивых проблем Visu­al Stu­dio.

Опытные пользователи знают, что в поиске-замене есть свой Reg­Exp. Но работает он не так, как у людей.

Что делать? Искать готовые решения.

Например:

:Zs#$ - все пробелы на конце строк
^:d+ — все цифры в начале строк (например, из какого-нибудь блога код скопировал)

Список будет пополняться.

Book: Достоевский — Игрок

Та самая книга, которую Достоевский надиктовал за 25 дней. Короче, чем “Село Степанчиково”, “Неточка Незванова” и “Двойник”, но длиннее, чем “Бедные люди”. Чтобы усилить жанровую чересполосицу — это считается романом (а иногда и “Белые ночи” считают романом), в то время как “Вечный муж” (такого же размера) — рассказ.

Сложно сказать, насколько каторжной была работа: Достоевский диктовал с 12 до 4, т.е. 4 часа в день.

fb2-версия оказалась на купленном в воскресенье букридере. Вот и прочитал.

Очень достойно и захватывающе. Радует la baboulin­ka и впечатляет раздолбайство главного героя, который может выиграть деньги, но тратит их на парижское житьё пресловутой made­moi­selle Blanche. Нету безысходного мрака и сырости, которым шибает нас из школьной программы, где “Белые ночи” и “Преступление и наказание”. Просто срыв резьбы и падение, падение, падение.

Словом, Достоевский, конечно, велик. Несмотря на школьную программу.

Script: скомпилировать все *.sln файлы

Ещё один CMD-скрипт. Делает Switch для tor­toise, после чего перекомпилирует все *.sln в проекте.

@echo off
set MSBUILD=%WinDir%Microsoft.NETFrameworkv3.5msbuild
set PROJECT=d:ProjectsCSharpMinesweeperProj
TortoiseProc /command:switch /path:"%PROJECT%"
cd %PROJECT%
for %%f in (*.sln) do "%MSBUILD%" %%f
cd %~dp0.

К сожалению, switch с заменой ветки в текущей стабильной версии делать нельзя. В night­ly версии у switch добавился параметр url, но бета-версии Tor­toise на работе обычно не ставят.

Script: Скриптовая эволюция

Дано — написать скрипт, который кидает определённые файлы из локальной папки в папку на подключенном сетевом диске. Например, веб-проект, в котором мы изменили 2–3 dll. OS Win­dows.

На каком языке писать скрипт?

ActivePerl под Win­dows — иногда не принимает переданных параметров
Cyg­win — нету
Pow­er­Shell — просто так даже Hel­lo World не запустишь. Система будет ругаться, что он… не подписан (маразм? маразм!)

Похоже, технологии ушли слишком далеко. Нам остаётся только старый формат CMD файлов.

copydll.cmd:

@echo off
set FROM=d:proj1bin
set TO=W:stableproj1bin
FOR %%A IN (%*) DO IF EXIST "%FROM%%%A.dll" COPY /Y "%FROM%%%A.dll" "%TO%%%A.dll"

Добавляем папку в Path и вызываем

copydll Proj.Module1 Proj.Module2

А этот cmd можно кинуть на панель в Total Com­man­der и просто перетаскивать туда нужный файл:

@echo off
set TO=W:stableproj1bin
FOR %%A IN (%*) DO IF EXIST %%A COPY /Y %%A "%TO%%%~nxA"

Конечно, можно просто перетащить на панель Total Commander’а саму папку, но тогда придётся дважды щёлкать мышкой (ОК в диалоге копирования и потом Replace All)

Да, ещё. Если написать в CMD-шке PAUSE, она в этом месте попросит Press any key to con­tin­ue.…

Book: Программист-прагматик

Книга довольно старая (эпохи, кажется, Win­dows 2000) и очень плохо переведена (есть глава “Мой исходный текст съел кот Мурзик”).

Набор инструментов типичен для того времени — C++, CORBA, Perl и т.п.

Внимательно читать не стоит, а вот просмотреть — надо. В книге есть мысли, которые просто в голову не приходили.

Полезное:

1. Предлагать варианты решения проблемы, а не отговорки.

2. “Закон разбитых окон” выполняется и для программ. Если есть ошибки и нестабильности — очень скоро их станет намного больше.

3. Качество должно быть одним из требований.

4. Читайте по одной технической книге ежеквартально. И другие книги — не относящиеся к технической литературе.

5. Экспериментируйте с различными операционными системами.

6. Оставайтесь в курсе событий — интернет и т.п.
(Советует журналы и news-группы — в духе времени. Сейчас закрылись и основной журнал по C++, и The Perl Jour­nal. Статьи из последнего можно найти в Интернете по метке TPJ).
и критически осмысляйте всё, что нашли и услышали.

7. “Лучше быть проигнорированным, чем недооценённым”. Беседуйте с гуру и с потребителями.

8. Принцип для выступлений WISDOM:

- Чему вы хотите научить?

- Какова их заинтересованность?

- Насколько искушена аудитория?

- Насколько детальным должно быть выступление?

- Кто должен обладать информацией?
 
- Как мотивировать слушателей?

9. DRY — Don’t repeat your­self. Важно писать код так, чтобы его было легко использовать повторно. Тогда отпадает большинство эпизодов с дублированием.

10. Ортогональность — следует исключать взаимодействие между объектами, не относящимися друг к другу. Искать баги становится легче, код — более удобным для повторного использования, и немного возрастает производительность.

11. Из 10 — подобные функции могут быть объеденены в Strat­e­gy, а глобальные данные — в Sin­gle­tone

12. Обратимость. Вытекает из предыдущего. Помним, что не существует окончательных решений. Если чтобы заменить MS SQL на Ora­cle надо переписывать всё — это не порядок.

13. Чтобы стрелять ночью прицельно, используют трассирующие пули, которые светятся в темноте. Куда попадёт трассирующая — попадёт и обычная. Используйте “трассирующие” предложения, чтобы понять, что от вас хотят.

14. Прототипы (не работает, но показать можно):
- Архитектуры (классы-заглушки)

— Новых функциональных возможностей для системы (интерфейс, который не работает)

— Структуры или содержания внешних данных

— Компонентов

— Рабочих характеристик

— Интерфейса

15. “Границы моего языка есть границы моего мира”.

(Неспроста Хайдеггер искал истину в языке, иногда получая слова в духе Хлебникова)

Программируйте как можно ближе к предметной области вашей задаче. Если команды можно упаковать в классы-“пакеты” — пакуйте! Т. ж. хорошо писать мини-языки, там, где они могут пригодиться.

Если язык парсится, его лучше описать в BNF, которая годится для всех контекстно-свободных грамматик (Perl, например,ей не опишешь)

Т.к. срок службы большинства прикладных программ превышает ожидаемый, лучше, чтобы язык был гибче и удобочитаемей. Ближе к настройкам rc, чем к настройкам send­mail.

16. Проводите оценки по следующим градациям:

1–15 дней — дни

1 — 8 недель — недели

8 — 30 недель — месяцы

> 30 недель — надо хорошенько подумать

Если вы уверены, что проект займёт 125 рабочих дней (= 25 недель) — говорите “примерно 6 месяцев).

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

17. Сохраняйте данные в виде простого текста. Так проще править руками

18. Графические интерфейсы:

преимущество — WYSIWYG — получаешь то, что видишь

недостаток — WYSIAWYG — получаешь только то, что видишь

Поэтому любите командную строку, а в ней — cyg­win или даже UWin.

19. Используйте 1 редактор, но по максимуму. Требования: настраиваемость, расширяемость, программируемость

20. Отлаживать без паники. Не обвинять, а исправлять. Не предполагайте — доказывайте.

21. Добиться воспроизводимости, и только потом чинить. Метод “резинового утёнка”

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

23. Совершенных программ не бывает. Бывают программы, которые легко поддерживать.

24. Пользуйтесь исключениями только в исключительных случаях.

25. Если запутались — применяйте закон Деметера. Но без фанатизма.

26. Настройка лучше интеграции. Текст программы — абстракция, метаданные — подробности.

27. Проектируйте, не забывая о службах — некоторые процессы так и просятся, чтобы их распараллелили. Для распределённых приложений — концепция доски объявлений.

28. Реорганизуйте чаще, раньше, регулярней.

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

30. Работайте с пользователем, чтобы говорить на его языке.

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

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

33. Вопросы, если упёрлись:

Можно ли сделать проще?
Мы решаем проблему, или увязли в деталях?
Почему это проблема?
Что делает её сложной?
Нужно ли это?

34. Начинайте, когда знаете, что будете делать. Иначе дров наломаете.

35. Автоматизируйте всё.

Генерация текстов — Perl
Системные дела — Cmd, Cyg­win (теперь и Pow­er­Shell)
Тестирование интерфейса — есть и для этого средства (в наше время — iMacro, AutoIt)

36. Тестируйте беспощадно. Тестируйте в т.ч. сами тесты — специально поставьте неправильный вывод и т.п. и проверьте, найдёт ли его текст.

Cygwin: Ловим ошибки

Не забываем добавлять в начало bash-скрипта (сразу после адреса):

set -euo pipefail

чтобы увидеть ошибки в коде, переменных и пайпах.

А вот ошибки в функциях нам так и не покажут.

C#: Number Range в Combobox

Например, нам нужно положить в telerik-оподобный con­trol список из вот таких элементов:

<telerik:radcomboboxitem runat="server" text="2010" value="10" >

Такие штуки надо генерировать в одну строку:

private const int CC_VALID_YEARS_RANGE = 9;
....................................................................................
ddlYear.DataSource = Enumerable.Range(DateTime.Today.Year, CC_VALID_YEARS_RANGE)
                               .ToDictionary(item => item.ToString(),
                                             item => (item % 1000).ToString());
ddlYear.DataTextField = "Key";
ddlYear.DataValueField = "Value";
ddlYear.DataBind();

А если надо добавить ещё одну строку Year с пустым значением — нужно предварительно сбросить Dic­tio­nary в ToList() (получится List < Key­Val­ue­Pair < string, string > >) и сделать insert в 0-ой индекс. Дело в том, что Dic­tio­nary<> сам по себе не сортируется — соответственно, Insert-а в нём нет и fore­ach перебирает его в том же порядке, в каком элементы добавились.

private const int CC_VALID_YEARS_RANGE = 9;
....................................................................................
List <KeyValuePair<string, string>> yearsList =
              Enumerable.Range(DateTime.Today.Year, CC_VALID_YEARS_RANGE)
                        .ToDictionary(item => item.ToString(),
                                      item => (item % 1000).ToString()).ToList();
yearsList.Insert(0, new KeyValuePair < string,string > ("Year", String.Empty));
ddlYear.DataSource = yearsList;
ddlYear.DataTextField = "Key";
ddlYear.DataValueField = "Value";
ddlYear.DataBind();

Cygwin: Лучше, чем у Лекса Кравецкого

Во время оно lex-kravet­s­ki был не только коммунистом, но ещё и программистом. Например, написал большой пост о том, как конвертировать wav в mp3 консольным конвертером и 2 BAT-файлами.
При всем нашем уважении к пакетным файлам, то же самое на cyg­win выглядит не в пример лучше, лаконичней и помещается в 1 скрипт:

#!/bin/sh
LameDir="C:/lame"
if ([ $1 ]) then
 Artist="$1"
else
 echo -n "Artist="
 read Artist
fi
if ([ $2 ]) then
 Genre="$2"
else
 echo -n "Genre="
 read Genre
fi
CurrentDir=`dirs`
ParamAlbum=`echo $CurrentDir | sed 's!^(/?.*)*/(([0-9]{4})[ -]+)?(.*)$!--ty "3" --tl "4"!g'` #RegExp: ^(/.*)*/(([0-9]{4})[ -]+)?(.*)$
for filename in *.wav
do
  FileShortName=`echo $filename | sed 's!^(.*).wav$!1!g'` #RegExp: ^(.*).wav$
  ParamTrack=`echo $FileShortName | sed 's!^([0-9]{2})[ -.]+(.*)!--tn "1" --tt "2"!g'` #RegExp: ^([0-9]{2})[ -.]+(.*)
  echo "$LameDir/lame.exe -V2 "$CurrentDir/$FileShortName.wav" "$CurrentDir/$FileShortName.mp3" --pad-id3v2 --ta="$Artist" $ParamAlbum $ParamTrack --tg="$Genre""
done

Как использовать?

  1. Сохраняем скрипт в нашу директорию со скриптами под именем tomp3 (или ещё каким-нибудь). Во второй строке указываем директорию, куда мы распаковали Lame.
  2. Копируем файлы с CD в wav в таком виде:2009 — Album / 01 — Song.wavИли в другом:Album/01 — Song with a long-ling name.wav(год может быть и пустым)
  3.  Открываем cyg­win И переходим в папку с Wav-ками
  4. Набираем SH ~/tomp3 “Artist” “Genre”. Если без них — скрипт сам спросит.
  5. После завершения кодирования: rm *.wav
  6. И quit

К сожалению, скрипта с подсветкой синтаксиса для bash-скриптов у меня нет. Поэтому я подсветил как brush:perl. Получилось ярко и доступно.

BAT: Неизвестные факты из жизни пакетных файлов

Многие слышали про bat-файлы. Но не все знают что:

  1. Помимо BAT, были ещё CMD, WSH (Win­dows Script Host с поддержкой аж двух языков — JScript и VBScript), а чуть позже появился Pow­er Shell. JScript и VBScript интересны тем, что были намного сложнее и непонятнее BAT. Но и на них писали.
  2. BAT работали ещё в DOS-е, поэтому его выполняет command.com. А вот CMD — это более новый формат, его выполняет cmd.exe. Поэтому лучше использовать расширение CMD.
  3. BAT и CMD не понимают вложенный if (то есть if может быть только на 1 уровне). Поэтому чтобы сделать условия, приходится рисовать блок-схему и расставлять везде GO TO. Да-да, ту самую блок-схему, которую ещё во времена “Мифического человеко-месяца” рисовали уже после завершения проекта.
  4. @echo off нужно, чтобы bat-ник не писал вызовы на экран.
  5. rem — это строка комментария.
  6. Если написать в файле script1.bat строку script2 (причём script2.bat существует — т.е. мы вызываем его как команду), то script2.bat запустится, а script1.bat — прекратит выполнение. Чтобы не прекращал, надо писать EXEC script2.
  7. Переход в директорию, где лежит скрипт — cd %~dp0.
  8. Можно поставить cyg­win и наслаждаться *nix-овой командной строкой под Win­dows. Кстати, там можно вкладывать сколько угодно If-ов. А ещё можно поставить ActivePerl. К сожалению, писать скрипт прямо в командной строке у Perl под Win­dows не получится — не такой стандарт для кавычек.
  9. Параметры командной строки лежат в переменных  %1..%9. %0 — имя скрипта (без расширения). %10 и больше — нет.
  10. Включать-выключать сервисы из пакетных файлов — это просто:  Проверить:
    SC QUERY "XService" | find "RUNNING"
    IF not "%ERRORLEVEL%" == "0" GOTO StartService
    ................
    :StartService

    Запустить:

    NET START "XService"

    Остановить:

    NET STOP "XService"
  11. Откомпилировать и запустить пример для wxWid­get:
    • Cоздаём в каталоге, прописанном в AUTOEXEC, вот такой makesample.cmd
      @echo off
      make -f makefile.bcc
      for %%f in (*.exe) do start %%f
      exit
    • Заходим в каталог с файлами
    • Запускаем (нужно, чтобы был установлен компилятор make)

 

C++: Вопросы на собеседовании

Вопросы на собеседовании по С++” — один из самых популярных запросов, по которым находят этот блог.

А значит, настало время расширить этот постинг.

Т.к. такие списки периодически пропадают из Сети, я буду делать свой — с приложенными ответами.

Итак, понеслось:

Напоследок — картинка от jia3ep-а.