반응형

[C++11] shared_ptr 정리 및 간단한 사용 예제

 

shared_ptr은  C++11부터 사용 가능한 smart pointer입니다.

 

std::shared_ptr 란

포인터를 통해 객체의 공유 소유권을 유지하는 스마트 포인터입니다. 힙 메모리에 객체를 위한 메모리와 참조 카운터를 위한 메모리가 할당됩니다. 여러 shared_ptr 객체가 동일한 객체를 소유할 수 있습니다. 다음 중 하나가 발생하면 개체가 소멸되고 메모리 할당이 해제됩니다.
- 객체를 소유하고 있는 마지막 남은 shared_ptr이 파괴됩니다.
- 객체를 소유하고 있는 마지막 남은 shared_ptr에는 operator= 또는 reset()을 통해 다른 포인터가 할당됩니다.
- 객체는 생성 중에 shared_ptr에 제공되는 delete-expression 또는 사용자 지정 삭제자를 사용하여 소멸됩니다.

 

std::shared_ptr 사용법

1. object를 가르키는 shared_ptr 생성자 호출할 수 있습니다. => std::shared_ptr<T> ptr(T *);

2.  새로운 shard_ptr 객체를 생성하기 위해 std::make_shard를 사용할 수 있습니다.
=> std::shared_ptr<T> ptr = std::make_shared<T>(); 

3. 동일한 포인터를 공유할 수 있습니다. 공유하는 변수가 늘어날수록 참조 카운터가 증가하며, use_count() 함수를 사용하여 참조 카운터를 확인할 수 있습니다. reset() 함수는 nullptr처럼 포인터를 해제하고 참조 카운터를 감소시킵니다.

=> std::shared_ptr<T> ptr1 = std::make_shared<T>(); 

=> std::shared_ptr<T> ptr2 = ptr2;

 

#include <iostream>
#include <memory>

class Student
{
public:
    Student() { std::cout << __FUNCTION__ << std::endl; }
    ~Student() { std::cout << __FUNCTION__ << std::endl; }

    std::shared_ptr<Student> getPtr()
    {
        return std::shared_ptr<Student>(this);
    }
};

int main()
{
    Student* p = new Student;

    // 1. p object를 가르키는 shared_ptr 생성자 호출합니다.
    std::shared_ptr<Student> ptr1(p); 
    std::cout << "ptr1.use_count(): " << ptr1.use_count() << std::endl;

    // 2. 새로운 shard_ptr 객체를 생성하기 위해 std::make_shard를 사용합니다.
    //  std::make_shard는 객체와 참조 카운터를 위한 메모리를 할당합니다.
    std::shared_ptr<Student> ptr2 = std::make_shared<Student>(); 
    std::cout << "ptr2.use_count(): " << ptr2.use_count() << std::endl;

    // 3. 동일한 포인터를 공유합니다. (ptr3과 ptr4)

    std::shared_ptr<int> ptr3(new int(5));
    std::cout << "ptr3.use_count(): " << ptr3.use_count() << std::endl;
    std::cout << "ptr3 value : " << *ptr3 << std::endl;

    std::shared_ptr<int> ptr4 = ptr3;
    std::cout << "ptr4.use_count(): " << ptr4.use_count() << std::endl;
    std::cout << "ptr4 value : " << *ptr4 << std::endl;

    std::cout << "ptr4 reset! " <<  std::endl;
    ptr4.reset();
    std::cout << "ptr3.use_count(): " << ptr3.use_count() << std::endl;

    return 0;
}

std::shared_ptr를 사용할 때 주의할 점

개체의 소유권은 다른 shared_ptr에 값을 할당하거나 복사 구성을 통해서만 다른 shared_ptr과 공유할 수 있습니다. 다른 shared_ptr이 소유한 원시 기본 포인터를 사용하여 새 shared_ptr을 구성하면 정의되지 않은 동작이 발생합니다.

 

아래 예제는 ptr2에 ptr1에서 객체의 주소를 return 받아 할당하도록 하였는데, 실행파일이 죽게됩니다.

#include <iostream>
#include <memory>

class Student
{
public:
    Student() { std::cout << __FUNCTION__ << std::endl; }
    ~Student() { std::cout << __FUNCTION__ << std::endl; }

    std::shared_ptr<Student> getPtr()
    {
        return std::shared_ptr<Student>(this);
    }
};

int main()
{
    Student* p = new Student;

    //p object를 가르키는 shared_ptr 생성자 호출합니다.
    std::shared_ptr<Student> ptr1(p); 
    std::cout << "ptr1.use_count(): " << ptr1.use_count() << std::endl;

    // ptr2에 ptr1에서 객체의 주소를 return 받아 할당하도록 해봅니다.
    std::shared_ptr<Student> ptr2 = ptr1->getPtr(); // 
    std::cout << "ptr2.use_count(): " << ptr2.use_count() << std::endl;

    return 0;
}

Student라는 클래스를 하나 동적선언하였고, 이 객체를 두 개의 shared_ptr ptr1, ptr2에 할당하였습니다.  ptr2를 생성하는 시점에 ptr1에 의해서 이미 Student 객체의 파괴자는 호출되어 객체가 소멸됩니다. 객체가 없는 상태에서 ptr2에서 다시 파괴자를 호출하는 구조이기 때문에 실행파일이 죽게됩니다. 

#include <iostream>
#include <memory>

class Student
{
public:
    Student() { std::cout << __FUNCTION__ << std::endl; }
    ~Student() { std::cout << __FUNCTION__ << std::endl; }

    std::shared_ptr<Student> getPtr()
    {
        return std::shared_ptr<Student>(this);
    }
};

int main()
{
    Student* p = new Student;

    //p object를 가르키는 shared_ptr 생성자 호출합니다.
    std::shared_ptr<Student> ptr1(p); 
    std::cout << "ptr1.use_count(): " << ptr1.use_count() << std::endl;

    //p object를 가르키는 shared_ptr 생성자 호출합니다.
    std::shared_ptr<Student> ptr2(p); 
    std::cout << "ptr2.use_count(): " << ptr2.use_count() << std::endl;

    return 0;
}

참조

https://en.cppreference.com/w/cpp/memory/shared_ptr

 

std::shared_ptr - cppreference.com

template< class T > class shared_ptr; (since C++11) std::shared_ptr is a smart pointer that retains shared ownership of an object through a pointer. Several shared_ptr objects may own the same object. The object is destroyed and its memory deallocated when

en.cppreference.com

https://en.cppreference.com/w/cpp/memory/unique_ptr

 

std::unique_ptr - cppreference.com

(1) (since C++11) template <     class T,     class Deleter > class unique_ptr ; (2) (since C++11) std::unique_ptr is a smart pointer that owns and manages another object through a pointer and disposes of that object when the unique_ptr goes out of sco

en.cppreference.com

 

반응형
반응형

[googletest] ASSERT_EQ와 같은 ASSERT_xx 단헌 함수 간단 사용 예제

 

ASSERT_XX 사용법

기대하는 값과 실제 값을 비교하기 위한 테스트 함수입니다. 

ASSERT_XX (기대값, 실제값) 형식으로 사용합니다. 

EQ(==), NE(!=), LT(<), GT(>), LE(<=), GE(>=), STREQ(문자열 비교), STRCASEEQ(문자열 대소문자 무시 비교), DOUBLE_EQ(부동소수점 비교)를 사용합니다.

 

사용 예제

#include <gtest/gtest.h>
#include <string>

TEST(SampleTest, Test1){
	int value = 4;
	ASSERT_EQ(4, value);
}

TEST(SampleTest, Test2){
	int value = 4;
	ASSERT_NE(5, value);
}

TEST(SampleTest, Test3){
	int value = 4;
	ASSERT_LT(3, value);
}

TEST(SampleTest, Test4){
	int value = 4;
	ASSERT_GT(3, value);
}

TEST(SampleTest, Test5){
	int value = 4;
	ASSERT_LE(4, value);
}

TEST(SampleTest, Test6){
	int value = 4;
	ASSERT_GE(4, value);
}

TEST(SampleTest, Test7){
	std::string s1 = "hello";
	std::string s2 = "hello";
	ASSERT_EQ(s1, s2);
}

TEST(SampleTest, Test8){
	const char* s3 = "hello";
	const char* s4 = "hell2";
	ASSERT_STREQ(s3,s4);
}

테스트 결과는 아래와 같이 출력됩니다.

 

 

반응형
반응형

[googletest] gtest_main.cc를 라이브러리에 포함시키기

 

아래 링크에서 main 함수를 넣었는데, main 함수에서 동작이 고정되어 있다면, googletest에서 제공하는 gtest_main.cc의 main 함수를 이용할 수 있습니다. 이 gtest_main.cc의 main 함수를 library에 포함시켜서, 유닛 테스트 코드에서의 main은 사용하지 않도록 해보겠습니다. 

 

https://zidarn87.tistory.com/617

 

[googletest] googletest 다운로드 및 간단한 유닛 테스트 케이스 수행하기 - 리눅스 용

[googletest] googletest 설치 및 간단한 테스트케이스 수행하기 리눅스 우분투 20.04 버전에서 googletest 코드를 다운로드하고, googletest 코드로 정적 library를 만든 다음, 이 library를 가지고 간단한 테스트

zidarn87.tistory.com

 

기존 코드 - 유닛 테스트 코드에 main 함수를 구현함

아래 main 함수를 보면 gtest 라이브러리를 초기화하는 부분과 모든 단위 테스트를 수행하는 명령이 있습니다. 

이 부분은 항상 고정으로 사용한다면 gtest_main.cc의 main 함수를 이용하면 됩니다.

#include <gtest/gtest.h>

TEST(Sample, Test1){

}

TEST(Sample, Test2){
  FAIL();
}

int main(int argc, char** argv){

    testing::InitGoogleTest(&argc, argv);
    return RUN_ALL_TESTS();
}

gtest_main.cc를 라이브러리화

우선 아래 명령어를 이용하여 gtest_main.cc를 빌드합니다. 

g++ googletest-1.13.0/googletest/src/gtest_main.cc -c -I ./googletest-1.13.0/googletest/include/

빌드하면 gteste_main.o 파일이 생성되는 것을 볼 수 있습니다.

gtest-all.o와 gtest_main.o 를 library에 포함시킵니다. 

ar rcv libgtest.a gtest-all.o gtest_main.o

 

유닛 테스크 코드의 main 함수를 제거하고 빌드하기

아래와 같이 main 함수를 제거하고, 테스트 케이스 코드만 작성합니다.

#include <gtest/gtest.h>

TEST(Sample, Test1){

}

TEST(Sample, Test2){
  FAIL();
}

빌드하고, 실행해봅니다. 

빌드도 잘 수행되었고, 테스트 케이스도 잘 수행된 것을 볼 수 있습니다.

반응형
반응형

[googletest] googletest 설치 및 간단한 테스트케이스 수행하기

 

리눅스 우분투 20.04 버전에서 googletest 코드를 다운로드하고, googletest 코드로 정적 library를 만든 다음, 이 library를 가지고 간단한 테스트케이스를 수행하여 보겠습니다. 

 

googletest 설치하기

우선 아래 경로에 접속합니다.

https://github.com/google/googletest

 

GitHub - google/googletest: GoogleTest - Google Testing and Mocking Framework

GoogleTest - Google Testing and Mocking Framework. Contribute to google/googletest development by creating an account on GitHub.

github.com

최신 release 버전을 다운받기 위해 아래 빨간색 부분을 클릭합니다.

아래 링크의 주소를 복사합니다.

https://github.com/google/googletest/archive/refs/tags/v1.13.0.tar.gz

아래 명령어로 파일을 다운로드하고, 압축을 해제합니다.

wget https://github.com/google/googletest/archive/refs/tags/v1.13.0.tar.gz
tar xvf v1.13.0.tar.gz

압축해제하면 googletest-1.13.0 이라는 폴더가 생성되는 것을 볼 수 있습니다.

 

gtest-all 빌드 및 라이브러리로 만들기

모든 src를 가지고 있는 gtest-all.cc 파일을 빌드합니다.

g++ googletest-1.13.0/googletest/src/gtest-all.cc -c -I ./googletest-1.13.0/googletest/include/ -I ./googletest-1.13.0/googletest/

빌드가 완료되면 gtest-all.o 파일이 생성되는 것을 볼 수 있습니다.

이 파일을 가지고 libgtest.a 이라는 라이브러리를 만들어 봅니다.

ar rcv libgtest.a gtest-all.o

 

간단한 테스트 케이스 수행하기

아래의 코드 내용으로  simple.cpp를 생성합니다.

#include <gtest/gtest.h>

// 테스트케이스
TEST(Sample, Test1){

}

TEST(Sample, Test2){
	FAIL();
}

int main(int argc, char** argv){
    testing::InitGoogleTest(&argc, argv);
    return RUN_ALL_TESTS();
}

 

아래 명령어로 simple.cpp를 빌드하는데, 위에서 만든 libgtest.a 라이브러리를 로드합니다.

g++ simple.cpp -lgtest -L. -I./googletest-1.13.0/googletest/include/ -pthread

빌드하면 a.out 파일이 생성되는데, 이를 실행하면 테스트 수행 결과가 출력됩니다.

반응형

+ Recent posts