페이지

2014년 5월 6일 화요일

Substance 무료강연

2013년 8월 9일 금요일

C++ 개발자를 위한 Visual Studio 2013 - #1

이 글은 아래 글을 번역한 글임을 밝혀둡니다. 번역이 매끄럽지 않을 수도 있습니다. 잘못된 부분이 있다면 댓글로 남겨주세요.

원문 링크

2013년 7월 18일 목요일

내가 알아둬야 할 것들

이제 중급 레벨의 프로그래머 정도 되면 프로그래밍 언어만 알아서는 안된다는걸 자동적으로 느끼게 된다. 일단 나보다 실력이 좋은 사람들이 마구마구 아는 척을 해대면 무슨말인지 모르고 멍때리고 있을 순 없잖아!

내가 아직 제대로 이해하지 못하는 있는 것 중 하나가 CPU cache 에 관련 된 것이다.
뭐 대충 access memory 시간을 줄이기 위한 것이란건 알겠는데, 그래서 이게 어떻게 동작을 하는지 또 이게 문제가 생길 수 있는 여지가 뭔지 또는 이걸 이해하므로써 어떤 이득을 얻을 수 있는지 알고 싶어졌다.. 마구마구

그러던 중 오늘 아주 좋은 동영상을 발견해서 소개하려고 한다. 영어라는 점이 아쉽지만 내용은 정말 좋다.

[동영상 링크]



2013년 7월 16일 화요일

Unity Shader 의 Time 값


유니티에서 셰이더 코드를 짜는데 Time값을 _Time을 사용한다. 근데 이 time 값이 float4 타입인데 각 element가 아래 보는거와 같이 x = t/20, y = t, z = t*2, w = t*3 이따위다...

float4 _Time : Time (t/20, t, t*2, t*3)

왜 저렇게 값을 넣어놨는지 모르겠지만 처음에 저것 때문에 계산이 제대로 안되서 해멨음.. 신발

[참고] http://docs.unity3d.com/Documentation/Components/SL-BuiltinValues.html

2013년 7월 9일 화요일

C# 에서 readonly 와 const 의 차이

readonly 는 선언만 해도 된다. (생성자 함수내에서만 초기화 가능)
const 는 선언과 동시에 값을 설정해야 한다.

class Foo
{
    public readonly float foo1;
    public const float foo2 = 0.25f;

    Foo()
    {
             foo1 = 0.1f;
    }
}

2013년 6월 27일 목요일

AOT 컴파일러 와 Generic type 사용

사용 언어: C#

오늘 문제가 됐던 버그가 iOS버전으로 빌드를 하니 제네릭 클래스 타입을 사용하는 부분에서 crash가 생겼다. 안드로이드 버전에서는 문제가 안됐었는데 iOS버전에서만 crash가 나는 것이었다.

처음엔 원인을 몰라 당황했는데 유니티 트러블 슈팅 관련 문서를 읽어보니. 눈에 띄는 것이 있었다. "AOT": 현재 XCode에서 사용하는 컴파일 방식인데 이 방식은 JIT랑은 다른 형식의 컴파일 형태이다. 두 컴파일 방식에 대한 차이는 이곳 에서 확인 할 수 있다.

AOT방식에 사용제한이 있는데 generic parameter을 사용할때 explicit 한 type만 사용가능하다. 그래서 현재 프로젝트에서 generic type 을 generic container에 인자로 사용하자 크래쉬가 발생한 것이다.

해결한 방법으로는 현재 레퍼런스 타입을 받으므로 Systme.Object 를 인자로 받게 하고 다시 값을 얻어올때 generic type으로 캐스팅하도록 하였더니 이 문제를 해결 할 수 있었다.

문제가 됐던 코드 방식은 이것:

public class Test<T> : MonoBehaviour where T : Test<T>
{
        private Dictionary<int, T> m_test = new Dictionary<int, T>();

        public void foo()
       {
               Test<T> test = m_test[0];
       }    
}


해결 코드는 이것:

public class Test<T> : MonoBehaviour where T : Test<T>
{
        private Dictionary<int, System.Object> m_test = new Dictionary<int, System.Object>();

        public void foo()
       {
               Test<T> test = (Test<T>)m_test[0];
       }    
}


2013년 6월 22일 토요일

C++ 생성자 함수에서 virtual 함수를 호출 할 수 있을까?

이 글은 c++에서 왜 생성자 함수내에서 가상함수 호출을 하면 안되는가 대한 이유를 설명하는 글입니다. 아래 링크의 내용을 참고하였으나 제 마음대로 해석한 내용이므로 원본의 문장과는 다를 수 있습니다.

http://www.stroustrup.com/bs_faq2.html#vcall


답은 호출 할 수 있다. 하지만 사용하는데 조심해야 한다.
예상치 못한 일이 발생 할 수 있기 때문이다. 생성자 함수 내에서 가상함수를 호출하는 메카니즘은 사실 무의미하다. 왜냐면 파생클래스로부터의 오버라이딩이 아직 발생하지 않았기 때문이다. 모든 객체들은 "base before derived" 라는 기본 개념을 기반으로 설계 되어진다.

아래 예제 코드를 보자.

 #include<string>
 #include<iostream>
 using namespace std;

 class B {
 public:
  B(const string& ss) { cout << "B constructor\n"; f(ss); }
  virtual void f(const string&) { cout << "B::f\n";}
 };

 class D : public B {
 public:
  D(const string & ss) :B(ss) { cout << "D constructor\n";}
  void f(const string& ss) { cout << "D::f\n"; s = ss; }
 private:
  string s;
 };

 int main()
 {
  D d("Hello");
 }
 
출력 결과:
 B constructor
 B::f
 D constructor

D::f 가 아니다!! 어찌보면 너무 당연해보이지만.. 만약 B::B()로부터 D::f()가 호출되기 위해서 그 규칙이 다르다면 어떨지 생각해봐라. D::D() 생성자는 아직 실행되기도 전이기 때문에 D::f() 함수는 초기화 되지도 않은 string s를 할당하려고 할 것이다. 그 결과는 거의 곧 바로 크래쉬가 날 것이 뻔하다.

소멸자는 "derived class before base class" 을 기반으로 한다. 그래서 가상함수들은 생성자함수 내에서 처럼 동작한다. 오직 로컬 정의들만 사용되어진다. 그리고 현재 지워진 파생된 클래스를 건드리지 않기 위해 호출을 하지 않는다.

그동안 이것은 구현상의 제약사항으로 제안되어져 왔다. 하지만 그렇지 않다. 실제론 일반 함수내에서 가상함수를 호출하는 것처럼 생성자에서 호출하는 것은 안전하지 않은 규칙으로 여기는 것이 더 쉬울것이다.