Posts

마젠토 콜렉션(Collection) 사용하기

마젠토 콜렉션(Collection)은 한번에 여러개의 객체(product, customer, category 등등)를 불러오는데 사용됩니다.
예를들어 카테고리 페이지에있는 모든 제품을 불러온다던지, 모든 커스토머(customer) 정보를 불러올때 유용하게 쓰입니다.

아래와 같은 php코드를 루트에 올려놓고 시험해 보세요.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
require_once "app/Mage.php";
Mage::app();

// 모든 제품 불러오기
$collection = Mage::getResourceModel('catalog/product_collection'); // 첫번째 방법
$collection = Mage::getModel('catalog/product')->getCollection(); // 두번째 방법

// 모든 제품이 필요하지는 않으므로 5개만 로드합니다.
$collection->getSelect()->limit(5);
$collection->load(); // 콜렉션을 불러옵니다.
foreach($collection AS $product) {
    // 제품의 모든 데이터 출력
    print_r($product->getData());
}

위의 코드를 실행해 보시면 제품의 모든 데이터가 출력이 됩니다.
콜렉션 객체를 불러올때 두가지 방법을 보여드렸는데 첫번째 방법과 두번째 방법의 결과가 약간은 틀린걸 확인할수 있습니다.

첫번째 방법은 최소한의 데이터만 로드함으로써 메모리의 사용을 최대한 줄이는 방법이고,
$collection = Mage::getResourceModel(‘catalog/product_collection’);

두번째 방법은 제품의 모든 데이터를 불러옴으로써 메모리 사용에 상관없이 콜렉션을 사용하는 방법입니다.
$collection = Mage::getModel(‘catalog/product’)->getCollection();

개인적으로 메모리 사용이 적은 첫번째 방법을 선호합니다.
그렇다고 필요한 제품의 데이터를 불러오지 못하는것은 아닙니다.
아래와 같이 addAttributeToSelect()을 사용해서 필요한 Attribute을 불러올수 있습니다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
require_once "app/Mage.php";
Mage::app();

$collection = Mage::getResourceModel('catalog/product_collection');

// 아래처럼 원하는 attribute을 콜렉션에 포함시킬수 있습니다.
$collection->addAttributeToSelect('price');
$collection->addAttributeToSelect('weight');

// 모든 제품이 필요하지는 않으므로 5개만 로드합니다.
$collection->getSelect()->limit(5);
$collection->load(); // 콜렉션을 불러옵니다.
foreach($collection AS $product) {
    // 제품의 모든 데이터 출력
    print_r($product->getData());
}

Attribute으로 불러올 데이터를 addAttributeToFilter()를 사용해서 걸러낼수도 있습니다.
예를들어 id가 10보다 커야 한다던지 sku가 ‘abcd’를 포함해야 하는경우 아래와 같이 콜렉션을 설정할수 있습니다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
require_once "app/Mage.php";
Mage::app();

$collection = Mage::getResourceModel('catalog/product_collection');

// 아래처럼 원하는 attribute을 콜렉션에 포함시킬수 있습니다.
// product entity_id가 10보다 커야하는경우
$collection->addAttributeToFilter('entity_id', array('gt' => 10));

// price가 $100 이하여야 하는 경우
$collection->addAttributeToFilter('price', array('lt' => 100));

// sku가 'abcd'를 포함해야 하는 경우
$collection->addAttributeToFilter('sku', array('like' => '%abcd%'));

// sku가 'zzz1'이 아니어야 하는 경우
$collection->addAttributeToFilter('sku', array('neq' => 'zzz1'));

// description이 아무것도 없는 제품을 포함하는 경우
$collection->addAttributeToFilter('description', 'null');

$collection->load(); // 콜렉션을 불러옵니다.
foreach($collection AS $product) {
    // 제품의 모든 데이터 출력
    print_r($product->getData());
}

이런식으로 addAttributeToSelect()나 addAttributeToFilter()를 사용해서 원하는 attribute을 콜렉션에 포함시키거나 필터를 적용해서 걸러낼수 있습니다.
콜렉션의 사용은 한번에 많은 객체를 불러와야 할때, 또 한번에 여러 객체의 정보를 수정해야할때 유용하게 쓰입니다.

마젠토의 싱글턴 패턴

마젠토의 싱글턴 패턴
getSingleton()과 getModel()의 차이점과 사용법

싱글턴 패턴은 하나의 클래스에서 하나의 인스턴스만을 생성하도록 하는 소프트웨어 디자인 패턴중의 하나입니다.
마젠토에서 이런 싱글턴 패턴의 사용은 객체의 인스턴스 생성 남용을 막으려는 방법이고 결과적으로 메모리의 사용을 최소화하기 위한 방책입니다.

예를 들자면 데이터베이스 인스턴스가 가장 대표적인 예입니다.
페이지가 로딩되면서 여러번의 데이터베이스 사용이 필요할수 있지만 데이터베이스 연결 인스턴스는 하나면 족합니다.
데이터베이스를 사용할때마다 똑같은 데이터베이스에 연결시키는 인스턴스를 여러번 생성하게되면 불필요한 리소스를 낭비하게 됩니다.
이럴때 이미 생성된 데이터베이스 인스턴스를 재활용하고자 하는것이 기본적인 싱글턴 패턴의 컨셉트 입니다.

마젠토에서 객체 인스턴스를 생성하는 방법은 getSingleton(‘모델/클래스’); 또는 getModel(‘모델/클래스’); 이 사용되는데 getSingleton(‘모델/클래스’)의 경우 인스턴스가 이미 생성되어있지 않은경우 getModel(‘모델/클래스’) method를 호출하게 됩니다.
아래는 app/Mage.php의 getSingleton() 와 getMethod() method입니다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// getSingleton
public static function getSingleton($modelClass='', array $arguments=array())
{
    $registryKey = '_singleton/'.$modelClass;
    if (!self::registry($registryKey)) {
        self::register($registryKey, self::getModel($modelClass, $arguments));
    }
    return self::registry($registryKey);
}

// getModel
public static function getModel($modelClass = '', $arguments = array())
{
    return self::getConfig()->getModelInstance($modelClass, $arguments);
}

getModel()의 사용예

1
2
3
4
// $productId를 이용한 product 객체의 인스턴스 생성,
$product = Mage::getModel('catalog/product')->load($productId);
// product의 이름 가져오기
$name = $product->getName();

getSingleton()의 사용예

1
2
3
4
// customer session 불러오기
$customerSession = Mage::getSingleton('customer/session');
// customer 인스턴스 불러오기
$customer = $customerSession->getCustomer();

getModel()과 getSingleton()의 비교

1
2
3
4
5
6
7
8
9
10
11
12
// getSingleton()을 사용하여 product 객체를 product_id 34로 로딩
$product = Mage::getSingleton('catalog/product')->load(34);
echo $product->getId(); // 34 가 출력됨

// getModel()을 사용하여 product 객체를 로딩
$product = Mage::getModel('catalog/product');
echo $product->getId(); // 아무것도 출력되지 않음
// getModel()은 새로운 객체를 생성하기 때문에 아무것도 출력되지 않음

// getSingleton()을 사용하여 이미 존재하는 product 객체를 로딩
$product = Mage::getSingleton('catalog/product');
echo $product->getId(); // 34 가 출력됨

line 11의 경우, getSingleton()은 이미 생성된 객체를 불러오기 때문에 굳이 product_id 를 로딩하지 않아도 위의 line 2에서 생성된 객체를 자동으로 로딩하게 됩니다.
그래서 line 12에서는 기존에 로딩되어있던 product 객체의 id를 출력하게 됩니다.