【问题标题】:Return NULL instead of repeated value in certain columns在某些列中返回 NULL 而不是重复值
【发布时间】:2016-01-12 19:06:19
【问题描述】:

我的查询必须允许某些列中的重复值,但我希望它在出现重复值时在查询中返回空值。

以下两张图片显示了我的问题:

现在返回的内容

我希望它返回什么:

SQL 查询:

SELECT *
FROM completo_sem_saldo AS a
LEFT JOIN posicao_contabil AS b ON (a.Cliente = b.Cliente_Posicao)
LEFT JOIN saldo_analitico AS c ON (a.Cliente = c.Cliente_Saldo_Analitico)
LEFT JOIN titulos_em_ser AS d ON (a.Cliente = d.Cliente_Titulos_Em_Ser) 

【问题讨论】:

  • 尽量让你的问题更简单。你的问题只是 SQL 部分,所以删除 php 部分并专注于修复查询。您已经提供了结果,还包括每个表的 db 架构和示例数据。
  • 顺便说一句,我只是重新格式化您的查询,看起来最后有额外的括号
  • 只需编辑问题。让它更容易理解。
  • 再次简单,意思是让它成为一个只是 MySQL 的问题,删除 php 查询。如果在 MySql 中工作将在 php 中工作。仍然需要数据库架构和数据样本
  • 正确。你知道如何解决这个问题吗?

标签: php mysql join left-join


【解决方案1】:

您可以在 PHP 中使用执行重复检查并在重复时返回 "" 的函数来执行此操作:

function blankIfRepeat($result, $key, &$map) {
    if (isset($map[$key]) && $map[$key] == $result[$key]) {
        return ""; // it is the same value we encountered last time
    }
    $map[$key] = $result[$key];
    return $result[$key];
}

注意最后一个参数之前的&,以确保调用者获得对该参数所做的更改。

你可以这样使用它:

$map = array(); // initialise the variable used to track repetitions
while($resul1 = mysqli_fetch_assoc($query1)) {
    $nome = $resul1['Nome'];
    // call the function for the fields that need blanks for repeated values:
    $sld_dev_ctbl = blankIfRepeat($resul1, 'Sld_dev_ctbl', $map);
    $saldo = blankIfRepeat($resul1, 'Saldo', $map);

    $vlr_atual = $resul1['Vlr_atual'];

    $tabela .= '<tr>';
    $tabela .= '<td>'.$nome.'</td>';
    $tabela .= '<td>'.$sld_dev_ctbl.'</td>';
    $tabela .= '<td>'.$saldo.'</td>';
    $tabela .= '<td>'.$vlr_atual.'</td>';
    $tabela .= '</tr>';
}
$tabela .= '</table>';

【讨论】:

  • 不客气。我将 PHP 标记恢复为您的问题,因为现在很明显您也对 PHP 解决方案感兴趣。
  • 谢谢!我是软件开发的新手,这就是为什么我很难解决这个问题。我很抱歉英语中的错误单词,我来自巴西,仍在学习英语。 ;)
【解决方案2】:

使用变量只显示第一行。

CASE 的默认值为 null,所以我没有包含 ELSE

这假设没有负萨尔多所以起始值为-1

SELECT CASE WHEN rn = 1 THEN Sld_dev_ctbl END as Sld_dev_ctbl,
       CASE WHEN rn = 1 THEN Saldo END as Saldo           
       Vlr_actual,
FROM ( 
        SELECT *,
               (@rownum := if( @saldo = Saldo,
                               @rownum + 1, -- increase the counter
                               if (@saldo := Saldo,
                                   0,  -- reset the counter for next block
                                   0
                                  )
                             )
               ) as rn
        FROM completo_sem_saldo AS a
        LEFT JOIN posicao_contabil AS b ON (a.Cliente = b.Cliente_Posicao)
        LEFT JOIN saldo_analitico AS c ON (a.Cliente = c.Cliente_Saldo_Analitico)
        LEFT JOIN titulos_em_ser AS d ON (a.Cliente = d.Cliente_Titulos_Em_Ser) 
        CROSS JOIN (select @rownum := 0, @saldo := -1) params
        ORDER BY --<provide some field to choose which one will be first>
) T

【讨论】:

  • 请注意,数据中有一些记录,其中 Saldo 值与之前的记录相比没有变化,但 Sld_dev_ctbl 值会发生变化。请参阅所给示例的末尾附近。
  • @trincot 你是对的。我仍然相信纯 SQL 更好,但看起来 OP 对您的解决方案很满意。干得好
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-09-18
  • 1970-01-01
  • 2012-12-11
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多