Есть у меня такая хорошая привычка. Если к вечеру готов pull request по задаче, то я его не публикую до утра. Утром на свежую голову "пробегаюсь" по сделанным изменения и, как правило, 1-2 исправления выполняю.
Постепенно добираюсь до C++20. Понравилась возможность использования template parameters в лямбдах, которая явно улучшает читаемость кода.
Пример:
#include <iostream>
#include <vector>
int main()
{
std::vector<int> intItems = { 9, 10, -7, 11 };
std::vector<float> floatItems = { 1.0f, 9.0f, 5.01f, 99.12f };
std::vector<double> doubleItems;
int counter{};
auto processItem = [&counter]<typename T>(const std::vector<T>& items) {
std::cout << "================\n";
auto it = std::max_element(items.begin(), items.end());
if (it != items.end())
{
std::cout << "Max value: " << *it << "\n";
}
else
{
std::cout << "No items\n";
}
std::cout << "================\n";
++counter;
};
processItem(intItems);
processItem(floatItems);
processItem(doubleItems);
std::cout << "Processed " << counter << " items\n";
return 0;
}
Результат исполнения:
================
Max value: 11
================
================
Max value: 99.12
================
================
No items
================
Processed 3 items
До C++20 вынужден использовать generic lambda следующим образом:
...
auto processItem = [&counter](const auto& items) {
...
}
...
Продолжаю постигать азы #vim
На этот раз копался с вставкой текста. Раньше я как-то не задумывался насчёт работы команд p и P, делал всё на автомате.
Данные команды могу вставить текст из регистра:
- ниже (команда p) или выше (команда P) текущей строки, если в регистр, из которого производится вставка, была скопирована строка целиком, например, с помощью команды yy;
- слева (команда P) или справа (команда p) от символа под курсором, если в регистр, из которого производится вставка, был скопирован набор символов, например, с помощью команды yw (копирование слова);
@entech на самом деле надо было мне чуть больше контекста добавить. Код ревью изменений в коде может выполняться разными проверяющими: от 'первый раз вообще вижу этот код' до 'вместе прорабатывали решение, очень хорошо понимаю о чем идёт речь'.
Моё сообщение было больше про второе. Как раз принимается активное участие и в исследовании, оценке и приоритизации. Абсолютно правильное утверждение насчет стилистики и т.п. - тут уже давно все автоматизировано.
С чем несогласен:
"а готов ли сам с этим кодом возиться, если надо будет разобраться с чем-то не работающим или бажным?" - если такой код попадает на ревью - я считаю что он просто не до конца подготовлен и не должен доходить до ревью. Ожидается, что в команде есть некоторые соглашения по качеству кода, стилю, подходам.
Я в последнее время больше концентрируюсь на том всё ли сделано оптимально, правильно ли выбрана структура данных, нет ли лишних аллокаций памяти, правильный ли выбран алгоритм.
#TIL Как-то всё обходил меня стороной возможный рекурсивный вызов лямбды в C++, благодаря generic lambda expression (который давно уже появился, начиная с C++14):
#include <iostream>
#include <string>
int main()
{
auto print_star = [](int n, const auto& f) -> void {
std::cout << std::string(n, '*') << std::endl;
if (n-- > 1)
{
f(n, f);
}
};
print_star(5, print_star);
return 0;
}
Вывод:
*****
****
***
**
*
Во время код ревью очень важна концентрация и погружение в задачу, выполняемую коллегами. Здорово когда получается полностью "переключиться" со своей задачи на чужую и аналитически сравнить предлагаемое решение с тем какое бы сам сделал.
Расширил набор команд, которыми я удаляю текст в #vim
Раньше в основном использовал две команды:
dd - удаление строчки, на которой находится курсор
x - удаление символа, на котором находится курсор
Сейчас набор используемых команд расширился следующими:
1) daw - удалить слово вместе с пробелами.
Строка:
One two three four
Курсор находится на одном из символов слова two или на одном из пробелов между словом One и two. Команда daw отредактирует строку следующим образом:
One three four
2) dt<символ> - удалить текст в строке, начиная от символа под курсором до первого найденного <символ> справа.
Строка:
a = b * c * (d + e)
Курсор находится на символе b. Команда dt( отредактирует строку следующим образом:
a = (d + e)
3) d<номер строки>G - удалить строки, начиная от той на которой находится курсор до <номер строки>, включая ее.
Текст:
1 0000000
2 1111111
3 2222222
4 3333333
5 4444444
6 5555555
7 6666666
8 7777777
9 8888888
10 9999999
Курсор находится на строке номер 5 с содержимым 4444444. Команда d9G отредактирует текст следующим образом:
1 0000000
2 1111111
3 2222222
4 3333333
5 9999999
По ощущениям почувствовал небольшой прирост эффективности при работе в #vim
@KarimullinArthur Linux Mint
#git tips:
Довольно часто из терминала вызываю git diff с фильтром по файлам. Например, мне нужно показать только CMakeLists.txt's, которые модифицированы:
$ git diff -- "*CMakeLists.txt"
Бывают ситуации в #vim когда нужно переключиться из Insert в Normal mode для выполнения одной команды и оставаться в Insert'е. Для выполнение этого есть следующая команда:
<C-o> [команда]
Например, я нахожусь в Insert режиме и мне нужно переместиться на строку выше и редактировать её:
<C-o> k
Снова про #vim
Начал постоянно использовать команду cit - изменение внутри значения тега и удаление до искомого сочетания. Примеры:
1. Есть следующая строка:
<a href="http://example.com">link to site</a>
курсор находится на символе `h` (в href).
Команда cit (change inside tag - изменить внутри тега) удалит `link to site` и перейдёт в режим вставки для ввода нового значения тега.
2. В следующем json файле:
{
"colors" : [ "red", "green", "blue" ],
"old_colors" : [ "yellow", "grey" ]
}
курсор находится на символе `r` (в слове red). Необходимо в "colors" получить значения "old_colors" и удалить "old_colors".
Команда d/yel и нажатие Enter преобразуют json файл в следущий вид:
{
"colors" : [ "yellow", "grey" ]
}
d - delete (удалить), / - найти, слово начинающееся на yel. Важно заметить, что сам слово которое ищется не удаляется.
Хочу рассказать про интересную конструкцию в C++, которая появилась начиная с 11-го стандарта. Называется using конструктор (встречал еще название Inheriting constructors). Лучше всего её продемонстрировать на примере:
#include <iostream>
#include <cstdint>
namespace legacy
{
struct Configuration
{
// Empty for simplicity
};
}
struct Configuration
{
// Empty for simplicity
};
class Server
{
public:
explicit Server(legacy::Configuration)
{
std::cout << "Creating server using legacy config" << std::endl;
}
explicit Server(Configuration)
{
std::cout << "Creating server using config" << std::endl;
}
explicit Server(const std::string& config)
{
std::cout << "Creating server using text config" << std::endl;
}
};
class MockServer : public Server
{
using Server::Server;
};
int main()
{
Server s1{legacy::Configuration{}};
Server s2{Configuration{}};
Server s3{"127.0.0.1:8080"};
std::cout << "----------" << std::endl;
MockServer s4{legacy::Configuration{}};
MockServer s5{Configuration{}};
MockServer s6{"127.0.0.1:8080"};
return 0;
}
Вывод в stdout будет следующий:
Creating server using legacy config
Creating server using config
Creating server using text config
----------
Creating server using legacy config
Creating server using config
Creating server using text config
Есть объект класса Server, в MockServer пишем:
using Server::Server;
В результате объекты класса MockServer можно будет создать с точно такими же конструкторами, как и объекты класса Server, при этом конструкторы явно реализовывать не требуется.
Данная возможность порой очень полезна при написании различных Mock классов в тестах.
@entech интересная новость, но я пока правда еще не дошел до NeoVim и его настройки. VSCode пробовал на большом проекте?
Сочетание команд:
f' или F'
;
ci'
для быстрого перемещения внутри строки и её редактирования впечатляет в #vim
Небольшой C++ tip:
Создавать объекты std::optional лучше всего через std::make_optional или вызов emplace(), тогда при создании объекта будет один вызов конструктора.
Написал небольшой пример:
#include <iostream>
#include <optional>
class Element
{
public:
Element()
{
std::cout << "Default CNTR" << std::endl;
}
Element(const Element&)
{
std::cout << "Copy CNTR" << std::endl;
}
Element(Element&&)
{
std::cout << "Move CNTR" << std::endl;
}
~Element()
{
std::cout << "DSTR" << std::endl;
}
void TraceData() const
{
std::cout << "[" << posX << ":" << posY << "] " << (visible ? "visible" : "hidden") << "\n";
}
private:
int posX{0};
int posY{0};
bool visible{false};
};
int main()
{
// One constructor call and one destructor call
{
auto element = std::make_optional<Element>();
element->TraceData();
}
std::cout << "-------------\n";
// One constructor call and one destructor call
{
std::optional<Element> element;
auto& elem = element.emplace();
elem.TraceData();
}
std::cout << "-------------\n";
// Two constructor calls and two destructor calls
{
auto element = std::optional<Element>(Element{});
element->TraceData();
}
return 0;
}
Вывод будет следующий:
Default CNTR
[0:0] hidden
DSTR
-------------
Default CNTR
[0:0] hidden
DSTR
-------------
Default CNTR
Move CNTR
DSTR
[0:0] hidden
DSTR
Видно что в третьем случае создался объект через default'ный конструктор, а потом вызвался конструктор переноса и далее лишний деструктор.
Впервые появилась необходимость использования std::call_once в плюсах (до этого как-то не приходилось совсем). Чтобы поэкспериментировать как работает написал небольшой сэмпл:
#include <iostream>
#include <mutex>
#include <chrono>
#include <thread>
#include <vector>
std::once_flag once_work_call;
std::mutex cout_lock;
void init()
{
std::this_thread::sleep_for(std::chrono::seconds(5));
std::scoped_lock guard(cout_lock);
std::cout << "init finished" << std::endl;
}
void work(int i)
{
std::call_once(once_work_call, init);
std::scoped_lock guard(cout_lock);
std::cout << "finish work [" << i << "]" << std::endl;
}
int main()
{
std::vector<std::thread> threads;
for(int i = 0; i < 5; ++i)
{
threads.emplace_back(work, i);
}
for(auto& t: threads)
{
t.join();
}
return 0;
}
Пока функция, переданная в std::call_once (в данном случае init), успешно не отработает на одном из потоков, остальные потоки будут ожидать, когда дойдут до вызова std::call_once, понятное дело что вызова std::call_once с одним и тем же флагом (в данном случае once_work_call).
Небольшие успехи в изучении #vim:
- постоянно передвигаюсь внутри строки с помощью
f<символ> - вперёд и далее переход на следующее совпадение символа с помощью ;
F<символ> - назад к интересующему символу.
В целом очень удобно. Раньше я в основном по строке перемещался по словам с помощью w и W - вперёд, b и B - назад.
Плюс заметил за собой, что всё чаще на новую строку перемещаюсь с помощью <номер строки>G.
Одной из ближайших целей как разработчика у меня является научиться эффективно работать в vim. Очень давно хотел приступить к этому, но не получалось выделять достаточно времени (типичная проблема разработчиков). Плюс тяжело в рабочих задачах использовать этот редактор, т.к. значительно снижалась скорость работы и чувствовалась подсознательно нехватка возможностей редактора (просто по причине низкого уровня знакомства с инструментом). Тут нужно понимать, что у vim крайне высокий порог входа и довольно нескоро ты начинаешь им проникаться. Пока вижу что без серьезных предварительных тренировок к использованию в работе не перейти.
Осознал что любимая и самая часто используемая у меня команда это ciw - удаление слова под курсором и переход во вставку. И конечно же перемещение по файлу с помощью <C-u> <C-d>.
Разработчик, гик