template + lambda의 편리함

klyx의 이미지

대부분의 어플리케이션에 빠지지 않는 기능이 undo/redo(실행 취소/다시 실행)기능일 겁니다.
C++에서 이걸 구현하는 일반적인 방법은 실행 단위가 되는 클래스를 만들어 stack으로 관리하는 걸겁니다.

예를 들어, 다음과 같은 Application 클래스가 있다고 합시다.

#include <string>
 
class Application {
public:
	Application() {}
	void setTitle(const std::string &title) {
		m_title = title;
		updateWindowTitle();
	}
	void setCount(int count) {
		m_count = count;
	}
	void updateWindowTitle();
private:
	std::string m_title;
	int m_count = 0;
};

두가지 값을 바꾸는 기능이 있는데, 여기에 undo/redo를 도입하면 다음과 같은 형태가 될 것입니다.

#include <string>
#include <vector>
 
class UndoCommand {
public:
	UndoCommand() {}
	virtual ~UndoCommand() {}
	virtual void undo() = 0;
	virtual void redo() = 0;
};
 
class TitleCommand : public UndoCommand {
public:
	TitleCommand(std::string *title, const std::string &to, Application *app)
	: m_title(title), m_to(to), m_from(*title), m_app(app) {}
	virtual void undo() override { *m_title = m_from; m_app->updateWindowTitle(); }
	virtual void redo() override { *m_title = m_to; m_app->updateWindowTitle(); }
private:
	std::string *m_title, m_to, m_from;
	Application *m_app;
};
 
class CountCommand : public UndoCommand {
public:
	CountCommand(int *count, int to)
	: m_count(count), m_to(to), m_from(*count) {}
	virtual void undo() override { *m_count = m_from; }
	virtual void redo() override { *m_count = m_to; }
private:
	int *m_count, m_to, m_from;
};
 
class Application {
public:
	Application() {}
	~Application() {
		for (auto cmd : m_cmds)
			delete cmd;
	}
	void setTitle(const std::string &title) {
		push(new TitleCommand(&m_title, title));
	}
	void setCount(int count) {
		push(new CountCommand(&m_count, count));
	}
	void undo() {
		if (!m_cmds.empty() && m_undone != m_cmds.begin())
			(*--m_undone)->undo();
	}
	void redo() {
		if (m_undone != m_cmds.end())
			(*m_undone++)->redo();
	}
	void updateWindowTitle();
private:
	// 새 명령을 추가
	void push(UndoCommand *cmd) {
		cmd->redo(); // 일단 명령 실행해주고
		m_cmds.erase(m_undone, m_cmds.end()); // 마지막으로 undo/redo 한 거 이후로 다 지우고
		m_cmds.push_back(cmd); // 명령 추가해주고
		m_undone = m_cmds.end(); // 마지막 undo 위치 갱신
	}
	std::string m_title;
	int m_count = 0;
	std::vector<UndoCommand*> m_cmds;
	std::vector<UndoCommand*>::iterator m_undone = m_cmds.end(); // 마지막으로 undo된 위치
};

위 코드는 함수 구현이 전부 클래스 안에 들어 있어서 실제로 컴파일이 되지 않습니다.
지금은 어떤 framework같은걸 만드는게 목적이 아니기 때문에 일단 흐름만 파악해주세요.
UndoCommand 는 각각의 명령을 추상화한 기본 클래스입니다.
이경우에는 두가지 명령(TitleCommand와 CountCommand)을 클래스를 만들어 주고, 이들을 std::vector m_cmds로 관리해줍니다.
이제 명령을 실행할 때마다 UndoCommand의 객체들이 생성되어 m_cmds로 들어가고, undo/redo할때마다 m_cmds안에서 적당한 명령을 찾아서 undo/redo 해줍니다.

이것만으로도 꽤 길어졌는데, TitleCommand와 CountCommand 이 클래스들을 보면 매우 비슷하게 생겼습니다.
중복되는 부분이 너무 많아 보입니다. 게다가 어떻게 봐도 TitleCommand와 CountCommand는 재활용가능한 클래스가 아닙니다.
사실 undo/redo에서 실행되는 내용들은 보통 하나의 메뉴에 대응되기 때문에 재활용가능한 코드가 아니고 매우 구체적인 경우가 보통입니다.
손이 근질 거립니다. template을 이용하면 어떻게 하나로 묶어서 재활용 가능한 클래스로 만들 수 있을 거 같습니다.
하지만 undo()와 redo()구현에서 값을 설정한 이후에 업데이트해야 할 내용이 다르기 때문에 쉽지 않습니다.
여기에서는 lambda를 이용하면 다음과 같이 매우 간결한 구현이 만들어집니다.

template<typename T, typename Setter>
class ValueCommand : public UndoCommand {
public:
	ValueCommand(const T &to, const T &from, const Setter &setter)
	: m_to(to), m_from(from), set(setter) {}
	virtual void undo() override { set(m_from); }
	virtual void redo() override { set(m_to); }
private:
	T m_to, m_from; Setter set;
};
 
class Application {
...
	void setTitle(const std::string &title) {
		push(m_title, title, [this] (const std::string &v) {
			m_title = v;
			updateWindowTitle();
		});
	}
	void setCount(int count) {
		push(m_count, count, [this] (int v) {
			m_count = v;
		});
	}
...
private:
	template<typename T, typename Setter>
	void push(const T &to, const T &from, const Setter &setter) {
		push(new ValueCommand<T, Setter>(to, from, setter));
	}
...
};

여기에서는 명령이 2개뿐이기 때문에 별 차이가 없어보이지만 명령이 늘어나게되면 코드량에 상당한 차이가 나타납니다.
이처럼 lambda와 template을 활용하면 부분적으로 알고리즘이 다른 경우에도 하나의 클래스(또는 함수)로 통합하는게 가능합니다.

lambda에는 이보다 더 강력한 활용예도 많겠지만, 오늘 몇년만에 undo/redo를 다시 구현하다가 lambda+template의 조합을 써보고 참 편리해졌다고 느껴서 적어봤습니다.

Forums: 
익명 사용자의 이미지

좋은 강좌 감사합니다! :-)

댓글 달기

Filtered HTML

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

BBCode

  • 텍스트에 BBCode 태그를 사용할 수 있습니다. URL은 자동으로 링크 됩니다.
  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param>
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.

Textile

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • You can use Textile markup to format text.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Markdown

  • 다음 태그를 이용하여 소스 코드 구문 강조를 할 수 있습니다: <code>, <blockcode>, <apache>, <applescript>, <autoconf>, <awk>, <bash>, <c>, <cpp>, <css>, <diff>, <drupal5>, <drupal6>, <gdb>, <html>, <html5>, <java>, <javascript>, <ldif>, <lua>, <make>, <mysql>, <perl>, <perl6>, <php>, <pgsql>, <proftpd>, <python>, <reg>, <spec>, <ruby>. 지원하는 태그 형식: <foo>, [foo].
  • Quick Tips:
    • Two or more spaces at a line's end = Line break
    • Double returns = Paragraph
    • *Single asterisks* or _single underscores_ = Emphasis
    • **Double** or __double__ = Strong
    • This is [a link](http://the.link.example.com "The optional title text")
    For complete details on the Markdown syntax, see the Markdown documentation and Markdown Extra documentation for tables, footnotes, and more.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 사용할 수 있는 HTML 태그: <p><div><span><br><a><em><strong><del><ins><b><i><u><s><pre><code><cite><blockquote><ul><ol><li><dl><dt><dd><table><tr><td><th><thead><tbody><h1><h2><h3><h4><h5><h6><img><embed><object><param><hr>

Plain text

  • HTML 태그를 사용할 수 없습니다.
  • web 주소와/이메일 주소를 클릭할 수 있는 링크로 자동으로 바꿉니다.
  • 줄과 단락은 자동으로 분리됩니다.
댓글 첨부 파일
이 댓글에 이미지나 파일을 업로드 합니다.
파일 크기는 8 MB보다 작아야 합니다.
허용할 파일 형식: txt pdf doc xls gif jpg jpeg mp3 png rar zip.
CAPTCHA
이것은 자동으로 스팸을 올리는 것을 막기 위해서 제공됩니다.