深入理解MAGENTO – 第九章 – 数据集合瓦瑞恩
本来,作为一个PHP程序员,如果你想攒一组变量的相关你有一个选择,古老的 阵列 。 尽管共享一个地址的名称与C存储器的阵列,一个PHP数组是一种通用的字典可变数组索引像数值对象结合行为的影响。
在其他语言的选择不是那么简单。 您有 多个数据结构 语义学的选择,每组速度和提供特定的优势,存储。 PHP的理念是消除客户程序员这个选择,让他们从一个有用的数据结构,这是”足够好”。
所有这一切难堪的是一个特定类型的软件开发人员,和PHP 5着手改变通过提供内置的地位利国的类和接口,允许你创建自己的数据结构。
Magento充分利用了这一点,在使用Varien_Data_Collection来做数据收集的时候,它实现了php内置IteratorAggregate对象迭代器和Countable两个接口。
下面是用php内置类ArrayObject的一个例子。
$array = new ArrayObject();
class MyCollection extends ArrayObject{}
$collection = new MyCollection();
$collection[] = 'bar';
在接下来的文章中,我认为你已经了解ArrayObject、IteratorAggregate、Countable。如果还是很陌生,我建议你先阅读这篇文章PHP5对象迭代(Object Iteration)。当然你不必了解很底层的东西,你只需要知道如何用就可以了。
在Magento代码中,其实每个Model都有个Collection。了解这些数据收集器是如何工作的是你成为一个真正Magento开发人员的关键点。
下面让我们开始吧,前面我们创建过一个Helloworld模块,现在我们继续用他开始我们接下来的学习。
创建一个数据收集器
首先,我们创造一些新的对象。
$thing_1 = new Varien_Object();
$thing_1->setName(‘Richard’);
$thing_1->setAge(24);
$thing_2 = new Varien_Object();
$thing_2->setName(‘Jane’);
$thing_2->setAge(12);
$thing_3 = new Varien_Object();
$thing_3->setName(‘Spot’);
$thing_3->setLastName(‘The Dog’);
$thing_3->setAge(7);
Magento中所有的Model都继承Varien_Object,在面向对象编程中,这样做的好处是当你想往多个Model中添加方法的时候,你只需要简单地修改一个文件即可。
在继承Varien_Object的类中,有两个魔术方法,get/set,你可以很方便的向对象中加入一个属性(值),让我们看个例子。
var_dump($thing_1->getName());
如果你忘记了属性的名字,你可以将所有数据都获取到:
var_dump($thing_3->getData());
你将看到以下结果:
array
‘name’ => string ‘Spot’ (length=4)
‘last_name’ => string ‘The Dog’ (length=7)
‘age’ => int 7
注意last_name属性,是用下滑线分隔的,如果你想用get和set魔术方法,那么需要使用驼峰命名法。
$thing_1->setLastName(‘Smith’);
在新版本的magento中你可以用array关联数组的方式获取数据。
var_dump($thing_3[“last_name”]);
T这个归功于php5的新特性,ArrayAccess接口。也是 “Object Oriented Programming”.
现在然我们把这些对象加到数据收集器Varien_Data_Collection中。很多程序员将Collection看成是数组,当然我不反对。
$collection_of_things = new Varien_Data_Collection();
$collection_of_things
->addItem($thing_1)
->addItem($thing_2)
->addItem($thing_3);
大多数Magento data Collections继承于Varien_Data_Collection,你可以使用里面的任何一个方法。
那么我们可以做些什么呢?接下来我们使用foreach去循环它。
foreach($collection_of_things as $thing)
{
var_dump($thing->getData());
}
这里还有方法取出第一个数据和最后一个数据。
var_dump($collection_of_things->getFirstItem());
var_dump($collection_of_things->getLastItem()->getData());
将你的数据转成xml
var_dump( $collection_of_things->toXml() );
只像取某一个字段
var_dump($collection_of_things->getColumnValues(‘name’));
Magneto还给我们提供了一些基本的过滤功能
var_dump($collection_of_things->getItemsByColumnValue(‘name’,’Spot’));
模型数据收集器(Model Collections)
前面我们有提到,所有Magento的模型数据收集器都继承Varien_Data_Collectionm,所以理论上我们可以使用之前的所有方法。下面让我们以product模型实战下。
public function testAction()
{
$collection_of_products = Mage::getModel(‘catalog/product’)->getCollection();
var_dump($collection_of_products->getFirstItem()->getData());
}
基本所有的Magento模型都有个方法叫getCollection默认情况下,它会返回系统中所有的数据。
Magento的数据收集器Collection包含很多复杂的逻辑来处理数据,无论是否使用索引或缓存、EAV表等。
上面的产品数据收集器,它里面还有Varien_Data_Collection_Db类。这个类给你很多有用的方法,例如如果你向看sql的select语句。
public function testAction()
{
$collection_of_products = Mage::getModel(‘catalog/product’)->getCollection();
var_dump($collection_of_products->getSelect()); //might cause a segmentation fault
}
上面的方法将输出
object(Varien_Db_Select)[94]
protected ‘_bind’ =>
array
empty
protected ‘_adapter’ =>
…
从上面可以看出,Magento使用的是ZendFramework的数据库链接层。接下来让我们看看更有意义的东西
public function testAction()
{
$collection_of_products = Mage::getModel(‘catalog/product’)->getCollection();
//var_dump($collection_of_products->getSelect()); //might cause a segmentation fault
var_dump(
(string) $collection_of_products->getSelect()
);
}
上面的方法将输出
‘SELECT `e`.* FROM `catalog_product_entity` AS `e`’
有时也会比较复杂,例如
string ‘SELECT `e`.*, `price_index`.`price`, `price_index`.`final_price`, IF(`price_index`.`tier_price`, LEAST(`price_index`.`min_price`, `price_index`.`tier_price`), `price_index`.`min_price`) AS `minimal_price`, `price_index`.`min_price`, `price_index`.`max_price`, `price_index`.`tier_price` FROM `catalog_product_entity` AS `e`
INNER JOIN `catalog_product_index_price` AS `price_index` ON price_index.entity_id = e.entity_id AND price_index.website_id = ‘1’ AND price_index.customer_group_id = 0′
这个差异取决于你选择的字段,同样也涉及到索引和缓存。如果你看过之前的文章,那么你应该知道很多Magento表是使用Eav表结构的,默认情况下一个eav的数据收集器将不会包含所有的对象字段,你可以通过addAttributeToSelect来添加它们。让我们看看例子。
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection()
->addAttributeToSelect(‘*’); //the asterisk is like a SQL SELECT *
或者你也可以只选某一个字段
//or just one
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection()
->addAttributeToSelect(‘meta_title’);
或者更多
//or just one
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection()
->addAttributeToSelect(‘meta_title’)
->addAttributeToSelect(‘price’);
延迟加载(Lazy Loading)
一般情况下,我们在创建sql后需要立刻执行,从而获取数据,例如。
$model = new Customer();
//SQL Calls being made to Populate the Object
echo ‘Done’; //execution continues
但是Magento不是这样的,它采用的是Lazy Loading。延迟加载意味着在程序需要数据前,sql是不执行的,如下。
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection();
在这个时候Magento还没有链接数据库,你可以放心地去做你想要做的事。
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection();
$collection_of_products->addAttributeToSelect(‘meta_title’);
你不必担心每次添加属性的时候Magento都会执行一个sql,去获取数据,sql只有在你需要数据的时候才会被执行。
Magento对数据库连接层做了良好的封装,当然它也考虑到了效率问题。在一般情况下,你没必要去担心sql后台是怎么执行的,只需要专心做你的功能,例如区块、布局等。这是Magento非常优秀的地方。
过滤数据(Filtering Database Collections)
最重要的一个方法是addFieldToFilter。通过这个方法可以添加我们sql中的WHERE语句。
public function testAction()
{
$collection_of_products = Mage::getModel(‘catalog/product’)
->getCollection();
$collection_of_products->addFieldToFilter(‘sku’,’n2610′);
//another neat thing about collections is you can pass them into the count //function. More PHP5 powered goodness
echo “Our collection now has ” . count($collection_of_products) . ‘ item(s)’;
var_dump($collection_of_products->getFirstItem()->getData());
}
addFieldToFilter方法中的第一个参数是你想过滤的字段名称,第二个是你想过滤的值。例如刚刚sku是字段名称,n2610是值。
第二个参数也可以被用来指定某一类型的数据。稍微有些复杂,我们继续往下看。
$collection_of_products->addFieldToFilter(‘sku’,’n2610′);
这个等同于sql中的where条件句
WHERE sku = “n2610”
下面的例子自己尝试下
public function testAction()
{
var_dump(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addFieldToFilter(‘sku’,’n2610′)
->getSelect());
}
将会输出这个
SELECT `e`.* FROM `catalog_product_entity` AS `e` WHERE (e.sku = ‘n2610′)’
但是这个很快会变得很复杂。试着做下面的练习。
var_dump(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addAttributeToSelect(‘*’)
->addFieldToFilter(‘meta_title’,’my title’)
->getSelect()
);
输出的将是下面的sql语句。
SELECT `e`.*, IF(_table_meta_title.value_id>0, _table_meta_title.value, _table_meta_title_default.value) AS `meta_title`
FROM `catalog_product_entity` AS `e`
INNER JOIN `catalog_product_entity_varchar` AS `_table_meta_title_default`
ON (_table_meta_title_default.entity_id = e.entity_id) AND (_table_meta_title_default.attribute_id=’103′)
AND _table_meta_title_default.store_id=0
LEFT JOIN `catalog_product_entity_varchar` AS `_table_meta_title`
ON (_table_meta_title.entity_id = e.entity_id) AND (_table_meta_title.attribute_id=’103′)
AND (_table_meta_title.store_id=’1′)
WHERE (IF(_table_meta_title.value_id>0, _table_meta_title.value, _table_meta_title_default.value) = ‘my title’)
在你有空的时候可以好好研究下上面的sql语句,我们先不转移焦点,继续我们下面的讲解。
其它比较运算符
我确定在刚刚的练习中,你想知道如何实现一个不是”=”的where条件句,例如不等于、大于、小于。刚刚我们有讲过addFieldToFilter的第二个参数允许传入不同”类型”。
其实很简单,只要将一个简单的数组作为第二个参数传入addFieldToFilter方法就可以变换条件句。
数组的键就是”类型”,关联的值就是你想过滤的值。我们改写下上面的代码。
public function testAction()
{
var_dump(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addFieldToFilter(‘sku’,array(‘eq’=>’n2610’))
->getSelect()
);
}
看上面的过滤器
addFieldToFilter(‘sku’,array(‘eq’=>’n2610′))
正如你看到的,第二个参数是一个php的数组。它的键是”eq”,代表等于的意思。
Magento在这个函数中有一系列英语的缩写,这些词的资料可以参考《tear of remembrance》。这些沿用了Perl语言中的一些比较运算符号。
在这里我将Magento所有的条件判断符号列出来供大家参考。
array(“eq”=>’n2610’)
WHERE (e.sku = ‘n2610′)
array(“neq”=>’n2610’)
WHERE (e.sku != ‘n2610′)
array(“like”=>’n2610’)
WHERE (e.sku like ‘n2610′)
array(“nlike”=>’n2610’)
WHERE (e.sku not like ‘n2610′)
array(“is”=>’n2610’)
WHERE (e.sku is ‘n2610’)
array(“in”=>array(‘n2610’))
WHERE (e.sku in (‘n2610’))
array(“nin”=>array(‘n2610’))
WHERE (e.sku not in (‘n2610′))
array(“notnull”=>’n2610′)
WHERE (e.sku is NOT NULL)
array(“null”=>’n2610′)
WHERE (e.sku is NULL)
array(“gt”=>’n2610’)
WHERE (e.sku > ‘n2610′)
array(“lt”=>’n2610’)
WHERE (e.sku < ‘n2610′)
array(“gteq”=>’n2610’)
WHERE (e.sku >= ‘n2610′)
array(“moreq”=>’n2610’) //a weird, second way to do greater than equal
WHERE (e.sku >= ‘n2610′)
array(“lteq”=>’n2610’)
WHERE (e.sku <= ‘n2610’)
array(“finset”=>array(‘n2610’))
WHERE (find_in_set(‘n2610’,e.sku))
array(‘from’=>’10’,’to’=>’20’)
WHERE e.sku >= ’10’ and e.sku <= ’20’
其中大多数是自我的理解,但有几个得特别注意。
in, nin, find_in_set
in and nin 条件句中,语序你传入一个数组作为值。例如:
array(“in”=>array(‘n2610′,’ABC123’)
WHERE (e.sku in (‘n2610′,’ABC123′))
notnull, null
关键字NULL是最特殊的sql句,它将忽略你传入的值。
array(“notnull”=>’n2610’)
WHERE (e.sku is NOT NULL)
from – to 过滤
这是另一种过滤方式,在传入的数组中,允许你传入两个键,是从哪里到哪里的意思,一个数值区间。
public function testAction
{
var_dump(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addFieldToFilter(‘price’,array(‘from’=>’10’,’to’=>’20’))
->getSelect()
);
}
上面等同于
WHERE (_table_price.value >= ’10’ AND _table_price.value <= ’20’)
AND 或者 OR
根据刚才讲的内容,你可以知道,通过多个 addFieldToFilter方法可以获得一个”AND”的条件句。
function testAction()
{
echo(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addFieldToFilter(‘sku’,array(‘like’=>’a%’))
->addFieldToFilter(‘sku’,array(‘like’=>’b%’))
->getSelect()
);
}
等同于下面的子句
WHERE (e.sku LIKE ‘a%’) AND (e.sku LIKE ‘b%’)
但是,聪明的你可以发现,上面的例子不可能返回任何结果,因为一个sku不可能以a开头,同时也以b开头。
我们希望用的应该是”OR”,那么如何实现呢?这又使我们将焦点集中到了addFieldToFilter方法的第二个参数上。
如果你希望构造一个or的语句,首先我们构造两个参数。
public function testAction()
{
$filter_a = array(‘like’=>’a%’);
$filter_b = array(‘like’=>’b%’);
}
然后将它们作为一组参数传入addFieldToFilter方法中,如下。
public function testAction()
{
$filter_a = array(‘like’=>’a%’);
$filter_b = array(‘like’=>’b%’);
echo(
(string)
Mage::getModel(‘catalog/product’)
->getCollection()
->addFieldToFilter(‘sku’,array($filter_a,$filter_b))
->getSelect()
);
}
你可以看到这样的一个子句。
WHERE (((e.sku LIKE ‘a%’) OR (e.sku LIKE ‘b%’)))
总结
恭喜你,你现在已经是一个很不错的Magento开发者了!因为你不需要写任何sql语句,就可以获取几乎所有模型的所有你想要的数据。
深入理解MAGENTO – 第九章 – 数据集合瓦瑞恩的更多相关文章
- Python 数据分析—第九章 数据聚合与分组运算
打算从后往前来做笔记 第九章 数据聚合与分组运算 分组 #生成数据,五行四列 df = pd.DataFrame({'key1':['a','a','b','b','a'], 'key2':['one ...
- 深入理解Magento – 第二章 – Magento请求分发与控制器
深入理解Magento 作者:Alan Storm 翻译:Hailong Zhang 第二章 – Magento请求分发与控制器 Model-View-Controller (MVC) ,模型-视图- ...
- 深入理解Magento - 第一章 - Magento强大的配置系统
深入理解Magento 作者:Alan Storm翻译:zhlmmc 前言第一章 - Magento强大的配置系统第二章 - Magento请求分发与控制器第三章 - 布局,块和模板第四章 - 模型和 ...
- 《利用python进行数据分析》读书笔记--第九章 数据聚合与分组运算(一)
http://www.cnblogs.com/batteryhp/p/5046450.html 对数据进行分组并对各组应用一个函数,是数据分析的重要环节.数据准备好之后,通常的任务就是计算分组统计或生 ...
- 深入理解计算机系统 第九章 虚拟内存 Part1 第二遍
这次花了4小时40分钟,看了第 559~575 页,共 17 页 第一遍对应地址 https://www.cnblogs.com/stone94/p/10264044.html 注意:本章的练习题一定 ...
- 《python for data analysis》第九章,数据聚合与分组运算
# -*- coding:utf-8 -*-# <python for data analysis>第九章# 数据聚合与分组运算import pandas as pdimport nump ...
- 深入理解Magento-第九章-修改、扩展、重写Magento代码
(博主提示:本章应该不是原作者的第九章,仅作补充和参考) 作为一个开发者的你,肯定要修改Magento代码去适应你的业务需求,但是在很多时候我们不希望修改Magento的核心代码,这里有很多原因,例如 ...
- 深入理解Magento - 第六章 - 高级Magento模型
我们讲过Magento有两种模型,简单模型和EAV(Entity Attribute Value)模型.上一章我们讲过所有的Magento模型都是继承自Mage_Core_Model_Abstract ...
- 深入理解Magento – 第四章 – 模型和ORM基础
深入理解Magento 作者:Alan Storm 翻译:Hailong Zhang 第四章 – 模型和ORM基础 对于任何一个MVC架构,模型(Model)层的实现都是占据了很大一部分.对于Mage ...
随机推荐
- 理解CSV格式规范(解析CSV必备)
什么是CSV逗号分隔值(Comma-Separated Values,CSV),其文件以纯文本形式存储表格数据(数字和文本),文件的每一行都是一个数据记录.每个记录由一个或多个字段组成,用逗号分隔.使 ...
- Vue开发环境的搭建及基本开发流程
1.下载并安装node,下载地址. 2.命令行运行以下命令安装npm淘宝镜像; npm install -g cnpm --registry=https://registry.npm.taobao.o ...
- Python--多态与多态性、绑定方法与非绑定方法
多态与多态性 多态 多态指的是一类事物有多种形态,(一个抽象类有多个子类,因而多态的概念依赖于继承) 1. 序列类型有多种形态:字符串,列表,元组. s='hello' l=[,,] t=('a',' ...
- Service系统服务(四):搭建单区域DNS服务器、特殊DNS解析、配置DNS子域授权、搭建并测试缓存DNS
一.搭建单区域DNS服务器 目标: 本例要求要求为DNS区域tedu.cn搭建一台DNS服务器,以便用户能通过域名的方式访问网站.测试阶段主要提供以下正向记录: svr7.tedu.cn ---> ...
- kubernetes批量删除pod
监控页面出现看到有运行失败的pod 1) 查看有哪些不运行的podcustom-metrics-apiserver日志占满空间被驱逐 [root@hadoop03 ~]# kubectl get po ...
- ubuntu下安装apidoc
1.到http://nodejs.cn/download/下载nodejs 可以复制链接 使用wget下载更加快速 选择对应的操作系统位数 下载到/usr/local/src 此处强烈不建议编译安装 ...
- 探索Redis设计与实现6:Redis内部数据结构详解——skiplist
本文转自互联网 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial ...
- (转)使用OpenGL显示图像(七)Android OpenGLES2.0——纹理贴图之显示图片
转:http://blog.csdn.net/junzia/article/details/52842816 前面几篇博客,我们将了Android中利用OpenGL ES 2.0绘制各种形体,并在上一 ...
- T1212:LETTERS
[题目描述] 给出一个R * S的大写字母矩阵,一开始的位置为左上角,你可以向上下左右四个方向移动,并且不能移向曾经经过的字母.问最多可以经过几个字母.roe× [输入] 第一行,输入字母矩阵行数R和 ...
- Xshell与securecrt对比
一.功能对比1.Xshell功能- 支持布局切换- 可调整Script执行顺序- 提供多标签功能- 对linux支持度高- 支持IPv6- 全球用户的多语言支持- 支持用户定义的键映射- 灵活和强大的 ...