2017-11-04 147 views
0

ERROR: structure of query does not match function result type DETAIL: Returned type double precision does not match expected type integer in column 1. CONTEXT: PL/pgSQL function get_analysis1_data(date,date) line 38 at RETURN QUERY ********** Error **********「查詢結構與函數結果類型不匹配。返回類型雙精度與列1中的預期類型整數不匹配。」?

postgresql程序代碼: -

CREATE 
OR REPLACE FUNCTION public.get_analysis1_data(IN date, IN date) RETURNS TABLE (loc_no integer, loc_d_share double precision, loc_id_share double precision, loc_id_share_per double precision, loc_a integer, loc_m integer, loc_l integer, loc_oneway integer, loc_round integer, loc_replacement integer, loc_oncall integer, loc_avg_idshare double precision, out_no integer, out_d_share double precision, out_id_share double precision, out_id_share_per double precision, out_a integer, out_m integer, out_l integer, out_oneway integer, out_round integer, out_replacement integer, out_oncall integer, out_avg_idshare double precision) AS $ BODY $ 
DECLARE in_from_date ALIAS FOR $1; 
in_to_date ALIAS FOR $2; 
loc_no integer; 
loc_d_share double precision; 
loc_id_share double precision; 
loc_id_share_per double precision; 
loc_a integer; 
loc_m integer; 
loc_l integer; 
loc_oneway integer; 
loc_round integer; 
loc_replacement integer; 
loc_oncall integer; 
loc_avg_idshare double precision; 
out_no integer; 
out_d_share double precision; 
out_id_share double precision; 
out_id_share_per double precision; 
out_a integer; 
out_m integer; 
out_l integer; 
out_oneway integer; 
out_round integer; 
out_replacement integer; 
out_oncall integer; 
out_avg_idshare double precision; 
BEGIN 
    RETURN Query 
    SELECT 
(( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     (
      is_outstation = FALSE 
     ) 
     and 
     (
      reporting_date BETWEEN in_from_date AND in_to_date 
     ) 
) 
     UNION 
( 
     SELECT 
     coalesce(round(SUM(driver_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(SUM(id_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round((SUM(id_share)/SUM(driver_share + id_share))*100), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'A' 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'M' 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'L' 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     is_round_trip = FALSE 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     is_round_trip = TRUE 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     trip_type = 'Replacement' 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     trip_type = 'OnCall' 
     AND is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(AVG(id_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = FALSE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(SUM(driver_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(SUM(id_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(((SUM(id_share)/SUM(driver_share + id_share))*100)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'A' 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'M' 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     car_type = 'L' 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     is_round_trip = FALSE 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     is_round_trip = TRUE 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     trip_type = 'Replacement' 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     count(*) 
     FROM 
     bookings 
     WHERE 
     trip_type = 'OnCall' 
     AND is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date) 
     UNION 
( 
     SELECT 
     coalesce(round(AVG(id_share)), 0) 
     FROM 
     bookings 
     WHERE 
     is_outstation = TRUE 
     AND reporting_date BETWEEN in_from_date AND in_to_date)) ; 
END 
$ BODY $ LANGUAGE plpgsql VOLATILE COST 100 ROWS 1000; 



ALTER FUNCTION public.get_analysis1_data(date, date) OWNER TO 
postgres; 

回答

0

我想你看到的錯誤將是許多第一。你的功能表明它希望看到一個記錄回來,看起來像這樣:

CREATE OR REPLACE FUNCTION public.get_analysis1_data(IN date, IN date) 
    RETURNS TABLE (
    loc_no integer, 
    loc_d_share double precision, 
    loc_id_share double precision, 
    loc_id_share_per double precision, 
    loc_a integer, 
    loc_m integer, 
    loc_l integer, 
    loc_oneway integer, 
    loc_round integer, 
    loc_replacement integer, 
    loc_oncall integer, 
    loc_avg_idshare double precision, 
    out_no integer, 
    out_d_share double precision, 
    out_id_share double precision, 
    out_id_share_per double precision, 
    out_a integer, 
    out_m integer, 
    out_l integer, 
    out_oneway integer, 
    out_round integer, 
    out_replacement integer, 
    out_oncall integer, 
    out_avg_idshare double precision) AS 

但據我所知,你的查詢只返回一列回(混合數據類型啓動)。其他20多個領域呢?

錯誤本身幾乎無所謂 - 你的查詢返回一個雙精度(可能是因爲這個:coalesce(round((SUM(id_share)/ SUM(driver_share + id_share)) * 100),),但我不認爲這是你真正的問題。我認爲你的問題是你的查詢看起來不像函數期望的那樣。

如果我不得不猜測,您希望您的查詢結果在一系列聯合查詢中是水平的(可調整的)而不是垂直的。

我不建議這一點,但你的函數的構建方式,好像你要做到這一點:

SELECT count(*) 
INTO loc_m 
FROM bookings 
WHERE car_type = 'A' 
AND is_outstation = FALSE 
AND reporting_date BETWEEN in_from_date AND in_to_date; 

(注意,除了第二線,INTO

然後你的查詢的最後一行會顯示如下:

return query select ... loc_m, ... 

但我認爲一個單一的數據透視查詢可能會更好。