我正试图根据用户的条件编写一个产品页面。默认情况下传递以下字段:
- qString(用于匹配产品名称/描述)
- maxPrice/minPrice(在表中的价格之间执行)
- minDate/maxDate(搜索不同日期间的产品)
上面给出了我的返回值。
接下来的几个字段是可选的,并根据类别通过jQuery添加:
- Category:Category_id,它与deals_deals_Category_id匹配(这仍然有效)
根据类别,可能会有其他属性(如移动交易有服务提供商、制造商),这些属性将在前端动态可见。从这里开始,客户可以通过复选框选择他们感兴趣的制造商和服务提供商。
我的表格包含:
- Deals(包含具有deal_category_id的产品)
- 类别(所有交易都链接到一个类别)
- Category_attributes(每个类别的附加属性。如果在类别中加载产品,则还会为其提供附加属性选择器
- Deal_attribute_options(包含基于属性的关系链接选项(其中属性可以是服务提供商,属性选项可以具有AT&T、T-Mobile等)
- Deal_attribute_values(包含创建产品时所选属性选项的值。假设我创建了一部新手机并选择了一个类别的手机,则产品将被要求提供attribute_options的值。他们将收到移动网络或制造商的列表,并将产品制造商链接到特定制造商
到目前为止,我在数据库中有两个产品,每个产品都链接到一个类别,根据链接到该类别的属性,每个产品将为每个属性选项选择一个值。如果有2个属性(制造商、移动网络),则每个产品都将在deal_attribute_values中有一个值,链接deal_id、attribute_id,然后链接所选值(因此产品可以是三星制造商和沃达丰作为服务提供商)。
我现在尝试做的是基于搜索查询,基于前端过滤返回关联的行。一旦前端的表单发生更改,它将序列化表单的值,并通过ajax根据这些查询返回正确的产品。
除了attribute_values之外,我正在让它完美地工作。我怀疑这与查询中的IN()子句有关,但希望有人能帮助我解决我做错的地方。
以下是模型方法:
public function getDeals() {
$select = array();
$tables = array();
$where = array();
$values = array();
// Default to get all deal rows:
$select[] = "`deals`.*";
$tables[] = "`deals`";
// Get the category Name AND ID
$select[] = "`deal_categories`.`name` AS `catName`";
$tables[] = "`deal_categories`";
$where[] = "`deals`.`deal_category_id`=`deal_categories`.`id`";
// Look if a category was selected:
$cat = $_POST['category'];
if ($cat != "") {
$where[] = "`deals`.`deal_category_id`=?";
$values[] = $cat;
}
// Assign a query of the deal by string:
if ($_POST['searchDeals'] != "") {
$where[] = "CONCAT_WS(' ',`deals`.`name`,`deals`.`description`) LIKE ?";
$values[] = "%" . str_replace(" ", "%", htmlspecialchars($_POST['searchDeals'])) . "%";
}
// Process Min / Max PRicing:
if (isset($_POST['minPrice'])) {
$minPrice = intval($_POST['minPrice']);
$where[] = "`deals`.`price` >= ?";
$values[] = $minPrice;
}
if (isset($_POST['maxPrice'])) {
$maxPrice = intval($_POST['maxPrice']);
$where[] = "`deals`.`price` <= ?";
$values[] = $maxPrice;
}
// PRocess the min/max dates:
if (isset($_POST['minDate'])) {
$minDate = $_POST['minDate'];
$where[] = "`deals`.`start_date` >= ?";
$values[] = $minDate;
}
if (isset($_POST['maxDate'])) {
$maxDate = $_POST['maxDate'];
$where[] = "`deals`.`end_date` <= ?";
$values[] = $maxDate;
}
if (isset($_POST['attr']) && valid_array($_POST['attr'])) {
$tables[] = "`deal_attribute_values`";
foreach ($_POST['attr'] AS $attrID => $checked) {
$values[] = $attrID;
$where_condition = "`deals`.`id`=`deal_attribute_values`.`deal_id`
AND `deal_attribute_values`.`deal_attribute_id`=?
AND `deal_attribute_values`.`value` IN (";
$bind_placeholder = array();
foreach ($checked as $val) {
$bind_placeholder[] = "?";
$values[] = $val;
}
$where_condition .= implode(',', $bind_placeholder) . ")";
$where[] = $where_condition;
}
}
$sql = "SELECT " . implode(", ", $select) . " FROM " . implode(",", $tables) . " WHERE " . implode(" AND ", $where);
return $this->get($sql, $values);
}
下面是一个包含属性的示例POST(注意,键3和4)表示属性集合(制造商和服务提供商)。其中的值是选中的选项,我们希望IN()通过这些选项进行操作。
Array
(
[searchDeals] =>
[category] => 2
[attr] => Array
(
[3] => Array
(
[0] => 8
[1] => 9
[2] => 12
[3] => 10
)
[4] => Array
(
[0] => 4
[1] => 7
[2] => 5
)
)
[minPrice] => 100000
[maxPrice] => 9500
[minDate] => 2014-10-26 00:00:00
[maxDate] => 2014-11-30 00:00:00
)
这里是基于上述输出的SQL(基于代码更新更新):
SELECT `deals`.*, `deal_categories`.`name` AS `catName` FROM `deals`,`deal_categories`,`deal_attribute_values` WHERE `deals`.`deal_category_id`=`deal_categories`.`id` AND `deals`.`deal_category_id`=? AND `deals`.`price` >= ? AND `deals`.`price` <= ? AND `deals`.`start_date` >= ? AND `deals`.`end_date` <= ? AND `deals`.`id`=`deal_attribute_values`.`deal_id` AND `deal_attribute_values`.`deal_attribute_id`=? AND `deal_attribute_values`.`value` IN (?,?,?,?) AND `deals`.`id`=`deal_attribute_values`.`deal_id` AND `deal_attribute_values`.`deal_attribute_id`=? AND `deal_attribute_values`.`value` IN (?,?,?)
这是PDO:的值
Array
(
[0] => 2
[1] => 100000
[2] => 9500
[3] => 2014-10-26 00:00:00
[4] => 2014-11-30 00:00:00
[5] => 3
[6] => 8
[7] => 9
[8] => 12
[9] => 10
[10] => 4
[11] => 4
[12] => 7
[13] => 5
)
在引入属性之前,我可以确认我正在得到结果。如果我取消勾选所有附件(制造商和服务提供商),只选择一个属性,例如"苹果",然后返回一个产品,我仍然可以得到结果,但如果我选择相关的服务提供商,则没有结果。
从本质上讲,对于属性,如果attribute_values表中乘积的值在in()子句中的属性中,我想选择一行。由于一个类别会有多个属性,如果我将苹果、三星和Vodacom/MTN保留为SP,则应返回多个产品。如果该产品是三星产品,并且Vodacom/MTN应返回,或者如果没有选择制造商,并且我们只选择Vodacom/MTN作为SP,则它应返回MTN/Vodacom的所有手机。即使没有Vodacom,它至少应该返回所有的MTN。
如果您需要任何其他帮助/信息,请告诉我。我试着在这里有尽可能多的细节,希望每个人都能理解:)
**答案更新**
我对亚历克斯发布的代码做了2次更改,这对我来说很有效!谢谢Alex:)
if (isset($_POST['attr']) && valid_array($_POST['attr'])) {
foreach ($_POST['attr'] AS $attrID => $checked) {
$current_table = "`deal_attribute_values` AS `dav" . $attrID."`";
$asName = "`dav".$attrID."`";
$values[] = $attrID;
$where_condition = "`deals`.`id`=".$asName.".`deal_id`
AND " . $asName . ".`deal_attribute_id`=?
AND " . $asName . ".`value` IN (";
$bind_placeholder = array();
foreach ($checked as $val) {
$bind_placeholder[] = "?";
$values[] = $val;
}
$where_condition .= implode(',', $bind_placeholder) . ")";
$where[] = $where_condition;
$tables[] = $current_table;
}
}
问题出在IN
子句中,我相信您是对的。为IN
子句准备的语句和提供的值存在问题。不能将整个IN
子句条件作为一个参数传递,因为它会自动将其作为一个整体转义,而不是作为单个值转义。
所以你有:
if (isset($_POST['attr']) && valid_array($_POST['attr'])) {
$tables[] = "`deal_attribute_values`";
foreach ($_POST['attr'] AS $attrID => $checked) {
$str = implode(", ", $checked);
$where[] = "`deals`.`id`=`deal_attribute_values`.`deal_id`
AND `deal_attribute_values`.`deal_attribute_id`=?
AND `deal_attribute_values`.`value` IN (?)";
$values[] = $attrID;
$values[] = $str;
}
}
当您将$str
参数绑定到语句时,它将用引号封装整个$str
,因此IN
只有一个值,而不是用逗号分隔的值。
您有两个选项:
- 您可以将
$str
附加到查询而不绑定它 - 您可以将足够的
?
添加到IN
子句中,用于您想要的每个IN
值,如下所示:
foreach($_POST['attr']AS$attrID=>$已选中){$values[]=$attrID;$where_condition="`deals`.`id`=`deal_attribute_values`.`deal_id`AND `deal_attribute_values``deal_attribute_id`=?AND `deal_attribute_values``值`IN(";$bind_placeholder=array();foreach($checked as$val){$bind_placeholder[]="?";$values[]=$val;}$where_condition.=内爆(',',$bind_placeholder)。")";$where[]=$where_condition;}
后续回答:您的查询中存在逻辑错误。当引入超过1个属性类别时,以下条件deal_attribute_values.deal_attribute_id=?
会出现两次。这意味着对于同一行,deal_attribute_id
必须同时等于2个值,这是不可能的。
您必须修改查询,以便对于要添加到查询中的每个属性类别/集,都必须将deal_attribute_values
添加到联接列表中。因此,对于上次更新,查询应该如下所示:
SELECT deals.*, deal_categories.name AS catName
FROM deals,deal_categories,deal_attribute_values, deal_attribute_values AS dav2
WHERE
deals.deal_category_id=deal_categories.id AND deals.deal_category_id=? AND deals.price >= ? AND deals.price <= ? AND deals.start_date >= ? AND deals.end_date <= ? AND
deals.id=deal_attribute_values.deal_id AND deal_
attribute_values.deal_attribute_id=? AND deal_attribute_values.value IN (?,?,?,?) AND
deals.id=dav2.deal_id AND dav2.deal_attribute_id=? AND dav2.value IN (?,?,?)
注意,我添加了deal_attribute_values
作为dav2
,这也是为了匹配第二组属性。
对于为此所需的代码更改,它可能是这样的(请参阅对$tables[]
部分的更改):
if(isset($_POST['attr'])&&valid_array($_POST['attr'])){foreach($_POST['attr']AS$attrID=>$已选中){current_table="`deal_attribute_values`AS dav"$attrID;$values[]=$attrID;$where_condition="`deals`.`id`=`".$current_table."`.`deal_id`AND `".$current_table."`.`deal_attribute_id`=?AND `".$current_table."`.`value`IN(";$bind_placeholder=array();foreach($checked as$val){$bind_placeholder[]="?";$values[]=$val;}$where_condition.=内爆(',',$bind_placeholder)。")";$where[]=$where_condition;$tables[]=$current_table;}}