php - MySQL 根据附加参数 PDO 返回行

标签 php mysql arrays pdo

我正在尝试根据用户的标准编写产品页面。默认传递以下字段:

  • qString(匹配产品名称/描述)
  • maxPrice/minPrice(在表中的价格之间做一个)
  • minDate/maxDate(搜索可用日期之间的产品)

  • 以上给了我返回的值。

    接下来的几个字段是可选的,并根据类别通过 jQuery 添加:
  • 类别:category_id 匹配回交易。deals_category_id(这仍然有效)

  • 根据类别,可能会有其他属性(例如移动交易具有服务提供商、制造商),这些属性将在前端动态显示。从这里客户可以通过复选框选择他们感兴趣的制造商和服务提供商。

    我的表包含:
  • Deals(包含带有 deal_category_id 的产品)
  • 类别(所有交易都链接到一个类别)
  • Category_attributes(每个类别的附加属性。如果在该类别中加载了产品,它们还将提供其他属性选择器
  • Deal_attribute_options(包含基于属性的关系链接选项(其中属性可能是服务提供商,属性选项可能有 AT&T、t-Mobile 等)。
  • Deal_attribute_values(包含产品创建时所选属性选项的值。假设我创建了一个新手机并选择了一个类别手机,产品将被要求为attribute_options提供值。他们将收到一个say mobile的列表网络或制造商,并将产品制造商链接到该特定制造商。

  • 所以到现在为止,我在数据库中有几个产品,每个产品都链接到一个类别,并且基于链接到该类别的属性,每个产品都会为每个属性选项选择一个值。如果有 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
     )
    

    在引入属性之前,我可以确认我正在获得结果。如果我取消勾选所有属性(对于制造商和服务提供商)并且只选择 1 个属性,例如“Apple”,然后返回一个产品,我仍然可以获得结果,但是如果我选择关联的服务提供商,则没有结果。

    本质上,对于属性,如果 attribute_values 表中该产品的值在 IN() 子句中的属性内,我想选择一行。因为如果我将 Apple、Samsung 和 Vodacom/MTN 保留为 SP,则基于一个类别将有多个属性,并且应该返回多个产品。如果产品是三星并且 Vodacom/MTN 应该退回,或者如果没有选择制造商而我们只选择 Vodacom/MTN 作为 SP,它应该退回所有 MTN/Vodacom 手机。即使没有Vodacom,它至少应该返回所有MTN。

    如果您需要任何其他帮助/信息,请告诉我。我试图在这里提供尽可能多的细节,所以希望每个人都能理解:)

    ** 答案更新 **

    我对 Alex 发布的代码进行了 2 处更改,这对我有用!谢谢亚历克斯:)
     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;
        }
    }
    

    当您绑定(bind) $str语句的参数它将封装整个$str因此在引号中 IN只有一个值而不是用逗号分隔的值。

    您有 2 个选择:
  • 您可以附加 $str到查询而不绑定(bind)它
  • 你加够了?IN每个IN的条款你想要的值(value)观是这样的:

  • 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;
    }

    跟进回答:
    您的查询存在逻辑错误。当您引入超过 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_valuesdav2这也是为了匹配第二组属性。

    对于为此所需的代码更改,它应该是这样的(请参阅对 $tables[] 部分的更改):

    if (isset($_POST['attr']) && valid_array($_POST['attr'])) {
    foreach ($_POST['attr'] AS $attrID => $checked) {
    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 .= implode(',', $bind_placeholder).")";
    $where[] = $where_condition;
    $tables[] = $current_table;
    }
    }

    关于php - MySQL 根据附加参数 PDO 返回行,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26603843/

    相关文章:

    php - cakephp 从一列中的字符串中选择多个字符串

    php - 从父子中计算子级 php mysql 5 级深

    mysql - 优化这些 MySQL 查询的更好方法

    java - Java读取JSON二维数组

    c++ - 如何在编译时初始化一个 constexpr 多维数组?

    python - 计算落在一组 x、y、z 坐标之间的值的数量

    php - sql insert - 神秘地破坏

    php - Mysql - 查询和索引

    mysql - CONCAT 或 REPLACE 基于 MySQL 中值的存在

    javascript - 将数据从 php 保存到 ajax 并在插入新数据时更改 div 的颜色?