PHP MySQL - 如何在预准备语句中插入默认值

时间:2016-02-19 23:32:27

标签: php mysql insert prepared-statement

假设我有一个准备好的声明,如下所示:

$insertSql = "";
$insertSql .= "INSERT INTO table";
$insertSql .= "(tag,month_interval,month_interval_lastupdated,date_due,date_due_lastupdated,date_completion,date_completion_lastupdated,";
$insertSql .= "activeNotification,date_lastmodified) ";
$insertSql .= "VALUES (?,?,NOW(),?,NOW(),?,NOW(),?,NOW())";

但有时这些问号中的一些不会被设定。

我是否可以在准备好的语句中替换一个关键字,告诉MySQL插入默认值?

类似的东西:

if($stmt = $mysqli->prepare($insertSql)) {
   if(!isset($tag)) {
      // code to find first question mark and replace w/ default value
      }
      $stmt->bind_param('sisss',$tag,$month_interval,$date_due,$date_completion);
      $stmt->execute();
}

日期字段的默认值为“1000-01-01 00:00:00”,而month_interval的默认值为0,其余字段的默认值为NULL。

3 个答案:

答案 0 :(得分:3)

我有同样的问题,所以我将这个答案https://stackoverflow.com/a/13867665/1251063改编为php预备声明:

  modelBuilder.Entity("WebApplication10.Data.JobCategory", b =>
            {
                b.Property<int>("ID")
                    .ValueGeneratedOnAdd();

                b.Property<string>("JobCategoryName");

                b.Property<int>("JobCategoryParentID");

                b.Property<string>("jobCategoryDetails");

                b.HasKey("ID");

                b.ToTable("JobCategory");
            });

答案 1 :(得分:2)

只是一个想法,如果你有一个条件,当你不知道所有的值,你应该准备语句甚至不添加这些值。

同样在某些情况下(例如在自动增量字段中),您只需插入NULL即可触发要插入的默认值, 但是 如果字段定义允许NULL's插入NULL会插入NULL,请务必小心。

答案 2 :(得分:1)

在我看来,你的问题涉及一个有趣的主题。

通过快速搜索,我找不到任何预定义的解决方案,但我会在接下来的几天继续搜索。

我已经详细说明了解决方案,但它有点扭曲而且不完全清洁。

首先,我使用占位符重新定义查询:

$query = "
    INSERT INTO table
                (
                    tag,
                    month_interval,
                    month_interval_lastupdated,
                    date_due,
                    date_due_lastupdated,
                    date_completion,
                    date_completion_lastupdated,
                    activeNotification,
                    date_lastmodified
                )
         VALUES (<tag>,<month_interval>,NOW(),<date_due>,NOW(),<date_completion>,NOW(),<activeNotification>,NOW())
";

在此示例中,我使用<>围绕占位符,但您可以选择首选占位符,确保它们不会与查询的其他元素混淆。

然后,我初始化一个关联数组,其中键作为占位符名称和要绑定的值,以及一个带有完整绑定类型集的变量($types):

$values = array( 'tag'=>$tag, 'month_interval'=>$month_interval, 'date_due'=>$date_due, 'date_completion'=>$date_completion, 'activeNotification'=>$activeNotification );
$types = 'sisss';

然后,最重要的一句话。我以这种奇怪的方式初始化一个数组:

$bind = array( Null, '' );

索引0设置为Null,因为是为stmt对象保留的(在下面创建);索引1是一个空字符串,将填充必要的绑定类型。

现在我在数组foreach的所有元素中执行$values循环:

$i = 0;
foreach( $values as $key => $val )
{
    if( is_null( $val ) )
    {
        $query = str_replace( '<'.$key.'>', 'DEFAULT', $query );
    }
    else
    {
        $query    = str_replace( '<'.$key.'>', '?', $query );
        $bind[1] .= $types[$i];
        $bind[]   = &$val;
    }
    $i++;
}

在该循环中,如果值为null,则将相应查询的占位符替换为mySQL关键字DEFAULT。否则,我将占位符替换为?,我将相应的$types子字符串添加到$bind[1],然后将值(通过引用)附加到$bind数组。

此时,真正的查询已准备就绪,我可以准备它:

$stmt = $db->prepare( $query ) or die( $stmt->error );

并且,如果没有默认值(count($bind)>2),我可以绑定它:

if( count($bind)>2 )
{
    $bind[0] = $stmt;
    call_user_func_array( 'mysqli_stmt_bind_param', $bind );
}

如前所述,我将$bind[0]设置为->prepare返回的stmt对象,然后我使用call_user_func_array执行mysqli_stmt_bind_param。我不能直接调用->bind_param因为我有一个变量参数号。

在这个古怪的过程结束时,我终于可以执行查询了:

$stmt->execute() or die( $stmt->error );

我已经测试了这段代码,但它确实有用。

主要问题在于is_null( $val ):使用数组,我不能使用isset作为测试,因为数组的每个元素都是设置的。如果包含允许Null值的占位符的所有字段都具有Null广告默认值,则代码将起作用。