0

我有 5 个查询,如下所示,我需要对所有查询执行联合

  if($dv_state_ids!="")//check if no state is entered
    $state_sql="SELECT pkg_id FROM package_state where state_id in ($dv_state_ids)  ";
  if($dv_city_ids!="")//check if no city is entered
    $city_sql="SELECT pkg_id FROM package_city where  city_id in ($dv_city_ids)";
  if($dv_category_ids!="")//check if nocategory is entered
    $category_sql="SELECT pkg_id FROM package_category where category_id in ($dv_category_ids) ";
  if($dv_sub_category_ids!="")//check if no subcategory is found
    $sub_category_sql="SELECT pkg_id FROM package_category where subcategory_id in ($dv_sub_category_ids) ";
  if($dv_category_ids!="")
    $package_sql="SELECT pkg_id FROM package where id in ($dv_category_ids) ";

我可以通过以下语句对上述查询运行联合

$sql_get_any_packages=$state_sql." union ".$city_sql." 
union 
".$category_sql." union ".$sub_category_sql." union ".$package_sql ;

但是如果任何查询为空,那么这将在 mysql 中产生错误。一种方法是我构建 32 个条件(这将由这 5 个查询形成以检查不同的条件)

任何人都可以建议我另一种出路吗

我还需要对这 5 个查询的结果执行交集,并且 sql 不支持 INTERSECT

请在这件事上给予我帮助

4

1 回答 1

1

“查询为空”显然是指 PHP 变量为null. 这绝不是特定于 SQL 或 SQL 联合的。

重复if_

所以你可以做的是:

$sql_get_any_packages = "";
if (!is_null($state_sql))
 $sql_get_any_packages .= " union " . $state_sql;
if (!is_null($city_sql))
 $sql_get_any_packages .= " union " . $city_sql;
if (!is_null($category_sql))
 $sql_get_any_packages .= " union " . $category_sql;
if (!is_null($sub_category_sql))
 $sql_get_any_packages .= " union " . $sub_category_sql;
if (!is_null($package_sql))
 $sql_get_any_packages .= " union " . $package_sql;
$sql_get_any_packages = substr($sql_get_any_packages, strlen(" union "));

这只是收集所有非null部分,包括每个部分的前导" union ",并在最后删除首字母" union "。如果所有查询可能null同时进行,那么最终结果将是FALSE.

使用数组

作为替代方案,您可以将您的零件放入一个数组中,用于array_filter放置null元素,然后implode将结果" union "用作胶水。相应的代码可能如下所示(另请参见ideone 上的示例):

$sql_get_any_packages = array(
    $state_sql,
    $city_sql,
    $category_sql,
    $sub_category_sql,
    $package_sql);
$sql_get_any_packages = array_filter($sql_get_any_packages);
$sql_get_any_packages = implode(" union ", $sql_get_any_packages);

关于相交

如果你想与结果相交,你不想计算他们的联合。一旦你做了一个联合,你就不能再访问它的各个部分,所以你不能将它们相互比较。要实现交集,您可以在 SQL 中使用连接。一个完整的示例可能如下所示:

SELECT package_state.pkg_id
FROM package_state, package_city, package_category, package_category, package
WHERE package_state.state_id IN ($dv_state_ids)
  AND package_city.city_id IN ($dv_city_ids)
  AND package_category.category_id IN ($dv_category_ids)
  AND package_category.subcategory_id IN ($dv_sub_category_ids)
  AND package.id in ($dv_category_ids)
  AND package_state.pkg_id == package_city.pkg_id
  AND package_state.pkg_id == package_category.pkg_id
  AND package_state.pkg_id == package.pkg_id

您可以使用我上面描述的解决方案来构建这样的查询,但它们会变得稍微复杂一些。特别是,package_state上述查询中的表与所有其他表所起的作用有些不同:这是SELECT您输出值的表形式,以及s also the table which occurs on the left hand side of all thepkg_id comparisons, where the other tables appear on the right hand side. So you might want to collect several arrays, one of table names and one offoo IN (bar)` 条件。然后你可以使用表名数组的第一个元素来扮演这个特殊的角色。

小心 SQL 注入

将用户请求中的字符串粘贴到查询中时要小心。它们可能包含会破坏您的数据的丑陋的东西。确保输入是健全的,例如通过检查它是否只是一个以逗号分隔的整数值列表。或者使用prepared statements,它可以让你远离大多数这些问题。

于 2012-11-30T17:33:41.060 回答