How do I execute a MS SQL Server stored procedure in java/jsp, returning table data?

前端 未结 4 1951
隐瞒了意图╮
隐瞒了意图╮ 2020-12-02 18:39

I am having difficulty executing a MS SQL Server stored procedure from Java/jsp. I wish to return a table set of data; the last line of the stored procedure is a regular se

相关标签:
4条回答
  • 2020-12-02 19:16

    Thank to Brian for the code. I was trying to connect to the sql server with {call spname(?,?)} and I got errors, but when I change my code to exec sp... it works very well.

    I post my code in hope this helps others with problems like mine:

    ResultSet rs = null;
    PreparedStatement cs=null;
    Connection conn=getJNDIConnection();
    
    try {
        cs=conn.prepareStatement("exec sp_name ?,?,?,?,?,?,?");
        cs.setEscapeProcessing(true);
        cs.setQueryTimeout(90);
    
        cs.setString(1, "valueA");
    
        cs.setString(2, "valueB");
    
        cs.setString(3, "0418");
    
        //commented, because no need to register parameters out!, I got results from the resultset. 
        //cs.registerOutParameter(1, Types.VARCHAR);
        //cs.registerOutParameter(2, Types.VARCHAR);
    
        rs = cs.executeQuery();
        ArrayList<ObjectX> listaObjectX = new ArrayList<ObjectX>();
        while (rs.next()) {
    
            ObjectX to = new ObjectX();
            to.setFecha(rs.getString(1));
            to.setRefId(rs.getString(2));
            to.setRefNombre(rs.getString(3));
            to.setUrl(rs.getString(4));
    
            listaObjectX.add(to);
    
        }
        return listaObjectX;
         } catch (SQLException se) {
            System.out.println("Error al ejecutar SQL"+ se.getMessage());
            se.printStackTrace();
            throw new IllegalArgumentException("Error al ejecutar SQL: " + se.getMessage());
    
        } finally {
    
            try {
    
                rs.close();
                cs.close();
              con.close();
    
            } catch (SQLException ex) {
                ex.printStackTrace();
            }
        }
    
    0 讨论(0)
  • 2020-12-02 19:23

    Frequently we deal with other fellow java programmers work which create these Stored Procedure. and we do not want to mess around with it. but there is possibility you get the result set where these exec sample return 0 (almost Stored procedure call returning zero).

    check this sample :

    public void generateINOUT(String USER, int DPTID){
    
        try {
    
    
            conUrl = JdbcUrls + dbServers +";databaseName="+ dbSrcNames+";instance=MSSQLSERVER";
    
            con = DriverManager.getConnection(conUrl,dbUserNames,dbPasswords);
            //stat = con.createStatement();
            con.setAutoCommit(false); 
            Statement st = con.createStatement(); 
    
            st.executeUpdate("DECLARE @RC int\n" +
                    "DECLARE @pUserID nvarchar(50)\n" +
                    "DECLARE @pDepartmentID int\n" +
                    "DECLARE @pStartDateTime datetime\n" +
                    "DECLARE @pEndDateTime datetime\n" +
                    "EXECUTE [AccessManager].[dbo].[SP_GenerateInOutDetailReportSimple] \n" +
                    ""+USER +
                    "," +DPTID+
                    ",'"+STARTDATE +
                    "','"+ENDDATE+"'");
    
            ResultSet rs = st.getGeneratedKeys();
    
            while (rs.next()){
                  String userID = rs.getString("UserID");
                  Timestamp timeIN = rs.getTimestamp("timeIN");
                  Timestamp timeOUT = rs.getTimestamp ("timeOUT");
                  int totTime = rs.getInt ("totalTime");
                  int pivot = rs.getInt ("pivotvalue");
    
                  timeINS = sdz.format(timeIN);
                  userIN.add(timeINS);
    
                  timeOUTS = sdz.format(timeOUT);
                  userOUT.add(timeOUTS);
    
                  System.out.println("User : "+userID+" |IN : "+timeIN+" |OUT : "+timeOUT+"| Total Time : "+totTime+" | PivotValue : "+pivot);
            }
    
            con.commit();
    
        }catch (Exception e) {
            e.printStackTrace();
            System.out.println(e);
            if (e.getCause() != null) {
            e.getCause().printStackTrace();}
        }  
    }
    

    I came to this solutions after few days trial and error, googling and get confused ;) it execute below Stored Procedure :

    USE [AccessManager]
    GO
    /****** Object:  StoredProcedure [dbo].[SP_GenerateInOutDetailReportSimple]    
    Script Date: 04/05/2013 15:54:11 ******/
    SET ANSI_NULLS OFF
    GO
    SET QUOTED_IDENTIFIER ON
    GO
    
    
    ALTER PROCEDURE [dbo].[SP_GenerateInOutDetailReportSimple]
    (
    @pUserID nvarchar(50),
    @pDepartmentID int,
    @pStartDateTime datetime,
    @pEndDateTime datetime
    )
    AS
    
    
    Declare @ErrorCode int
    Select @ErrorCode = @@Error
    
    Declare @TransactionCountOnEntry int
    If @ErrorCode = 0
    Begin
        Select @TransactionCountOnEntry = @@TranCount
        BEGIN TRANSACTION
    End
    
    
    If @ErrorCode = 0
    Begin       
    
        -- Create table variable instead of SQL temp table because report wont pick up the temp table
        DECLARE @tempInOutDetailReport TABLE
        (
            UserID nvarchar(50),
            LogDate datetime,   
            LogDay varchar(20), 
            TimeIN datetime,
            TimeOUT datetime,
            TotalTime int,
            RemarkTimeIn nvarchar(100),
            RemarkTimeOut nvarchar(100),
            TerminalIPTimeIn varchar(50),
            TerminalIPTimeOut varchar(50),
            TerminalSNTimeIn nvarchar(50),
            TerminalSNTimeOut nvarchar(50),
            PivotValue int
        )           
    
        -- Declare variables for the while loop
        Declare @LogUserID nvarchar(50)
        Declare @LogEventID nvarchar(50)
        Declare @LogTerminalSN nvarchar(50)
        Declare @LogTerminalIP nvarchar(50)
        Declare @LogRemark nvarchar(50)
        Declare @LogTimestamp datetime  
        Declare @LogDay nvarchar(20)
    
        -- Filter off userID, departmentID, StartDate and EndDate if specified, only process the remaining logs
        -- Note: order by user then timestamp
        Declare LogCursor Cursor For 
        Select distinct access_event_logs.USERID, access_event_logs.EVENTID, 
            access_event_logs.TERMINALSN, access_event_logs.TERMINALIP,
            access_event_logs.REMARKS, access_event_logs.LOCALTIMESTAMP, Datename(dw,access_event_logs.LOCALTIMESTAMP) AS WkDay
        From access_event_logs
            Left Join access_user on access_user.User_ID = access_event_logs.USERID
            Left Join access_user_dept on access_user.User_ID = access_user_dept.User_ID
        Where ((Dept_ID = @pDepartmentID) OR (@pDepartmentID IS NULL))
            And ((access_event_logs.USERID LIKE '%' + @pUserID + '%') OR (@pUserID IS NULL)) 
            And ((access_event_logs.LOCALTIMESTAMP >= @pStartDateTime ) OR (@pStartDateTime IS NULL)) 
            And ((access_event_logs.LOCALTIMESTAMP < DATEADD(day, 1, @pEndDateTime) ) OR (@pEndDateTime IS NULL)) 
            And (access_event_logs.USERID != 'UNKNOWN USER') -- Ignore UNKNOWN USER
        Order by access_event_logs.USERID, access_event_logs.LOCALTIMESTAMP
    
        Open LogCursor
    
        Fetch Next 
        From LogCursor
        Into @LogUserID, @LogEventID, @LogTerminalSN, @LogTerminalIP, @LogRemark, @LogTimestamp, @LogDay
    
        -- Temp storage for IN event details
        Declare @InEventUserID nvarchar(50)
        Declare @InEventDay nvarchar(20)
        Declare @InEventTimestamp datetime
        Declare @InEventRemark nvarchar(100)
        Declare @InEventTerminalIP nvarchar(50)
        Declare @InEventTerminalSN nvarchar(50)
    
        -- Temp storage for OUT event details
        Declare @OutEventUserID nvarchar(50)        
        Declare @OutEventTimestamp datetime
        Declare @OutEventRemark nvarchar(100)
        Declare @OutEventTerminalIP nvarchar(50)
        Declare @OutEventTerminalSN nvarchar(50)
    
        Declare @CurrentUser varchar(50) -- used to indicate when we change user group
        Declare @CurrentDay varchar(50) -- used to indicate when we change day
        Declare @FirstEvent int -- indicate the first event we received     
        Declare @ReceiveInEvent int -- indicate we have received an IN event
        Declare @PivotValue int -- everytime we change user or day - we reset it (reporting purpose), if same user..keep increment its value
        Declare @CurrTrigger varchar(50) -- used to keep track of the event of the current event log trigger it is handling 
        Declare @CurrTotalHours int -- used to keep track of total hours of the day of the user
    
        Declare @FirstInEvent datetime
        Declare @FirstInRemark nvarchar(100)
        Declare @FirstInTerminalIP nvarchar(50)
        Declare @FirstInTerminalSN nvarchar(50)
        Declare @FirstRecord int -- indicate another day of same user
    
        Set @PivotValue = 0 -- initialised
        Set @CurrentUser = '' -- initialised
        Set @FirstEvent = 1 -- initialised
        Set @ReceiveInEvent = 0 -- initialised  
        Set @CurrTrigger = '' -- Initialised    
        Set @CurrTotalHours = 0 -- initialised
        Set @FirstRecord = 1 -- initialised
        Set @CurrentDay = '' -- initialised
    
        While @@FETCH_STATUS = 0
        Begin
            -- use to track current log trigger
            Set @CurrTrigger =LOWER(@LogEventID)
    
            If (@CurrentUser != '' And @CurrentUser != @LogUserID) -- new batch of user
            Begin
                If @ReceiveInEvent = 1  -- previous IN event is not cleared (no OUT is found)
                Begin
                    -- Check day
                    If (@CurrentDay != @InEventDay) -- change to another day                    
                        Set @PivotValue = 0 -- Reset                        
    
                    Else -- same day
                        Set @PivotValue = @PivotValue + 1 -- increment
                    Set @CurrentDay = @InEventDay -- update the day
    
                    -- invalid row (only has IN event)
                    Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, RemarkTimeIn, TerminalIPTimeIn, 
                        TerminalSNTimeIn, PivotValue, LogDate )
                    values( @InEventUserID, @InEventDay, @InEventTimestamp, @InEventRemark, @InEventTerminalIP, 
                        @InEventTerminalSN, @PivotValue, DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))                                                         
                End                 
    
                Set @FirstEvent = 1 -- Reset flag (we are having a new user group)
                Set @ReceiveInEvent = 0 -- Reset
                Set @PivotValue = 0 -- Reset
                --Set @CurrentDay = '' -- Reset
            End
    
            If LOWER(@LogEventID) = 'in' -- IN event
            Begin           
                If @ReceiveInEvent = 1  -- previous IN event is not cleared (no OUT is found)
                Begin
                    -- Check day
                    If (@CurrentDay != @InEventDay) -- change to another day
                    Begin
                        Set @PivotValue = 0 -- Reset
    
                        --Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, TimeOUT, TotalTime, RemarkTimeIn,
                        --  RemarkTimeOut, TerminalIPTimeIn, TerminalIPTimeOut, TerminalSNTimeIn, TerminalSNTimeOut, PivotValue,
                        --  LogDate)
                        --values( @LogUserID, @CurrentDay, @FirstInEvent, @LogTimestamp,  @CurrTotalHours,
                        --  @FirstInRemark, @LogRemark, @FirstInTerminalIP, @LogTerminalIP, @FirstInTerminalSN, @LogTerminalSN, @PivotValue,
                        --  DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))  
                    End
                    Else
                        Set @PivotValue = @PivotValue + 1 -- increment
                    Set @CurrentDay = @InEventDay -- update the day
    
    
                    -- invalid row (only has IN event)
                    Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, RemarkTimeIn, TerminalIPTimeIn, 
                        TerminalSNTimeIn, PivotValue, LogDate )
                    values( @InEventUserID, @InEventDay, @InEventTimestamp, @InEventRemark, @InEventTerminalIP, 
                        @InEventTerminalSN, @PivotValue, DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))                     
                End         
    
                If((@CurrentDay != @LogDay And @CurrentDay != '') Or (@CurrentUser != @LogUserID And @CurrentUser != '') )
                Begin
    
                        Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, TimeOUT, TotalTime, RemarkTimeIn,
                            RemarkTimeOut, TerminalIPTimeIn, TerminalIPTimeOut, TerminalSNTimeIn, TerminalSNTimeOut, PivotValue,
                            LogDate)
                        values( @CurrentUser, @CurrentDay, @FirstInEvent, @OutEventTimestamp, @CurrTotalHours,
                            @FirstInRemark, @OutEventRemark, @FirstInTerminalIP, @OutEventTerminalIP, @FirstInTerminalSN, @LogTerminalSN, @PivotValue,
                            DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))      
    
                    Set @FirstRecord = 1
                End
                -- Save it
                Set @InEventUserID = @LogUserID                         
                Set @InEventDay = @LogDay
                Set @InEventTimestamp = @LogTimeStamp
                Set @InEventRemark = @LogRemark
                Set @InEventTerminalIP = @LogTerminalIP
                Set @InEventTerminalSN = @LogTerminalSN
    
                If (@FirstRecord = 1) -- save for first in event record of the day
                Begin
                    Set @FirstInEvent = @LogTimestamp
                    Set @FirstInRemark = @LogRemark
                    Set @FirstInTerminalIP = @LogTerminalIP
                    Set @FirstInTerminalSN = @LogTerminalSN
                    Set @CurrTotalHours = 0 --initialise total hours for another day
                End
    
                Set @FirstRecord = 0 -- no more first record of the day
                Set @ReceiveInEvent = 1 -- indicate we have received an "IN" event
                Set @FirstEvent = 0  -- no more "first" event
            End
            Else If LOWER(@LogEventID) = 'out' -- OUT event
            Begin
                If @FirstEvent = 1 -- the first OUT record when change users 
                Begin
                    -- Check day
                    If (@CurrentDay != @LogDay) -- change to another day
                        Set @PivotValue = 0 -- Reset
                    Else
                        Set @PivotValue = @PivotValue + 1 -- increment
                    Set @CurrentDay = @LogDay -- update the day
    
                    -- Only an OUT event (no IN event) - invalid record but we show it anyway
                    Insert into @tempInOutDetailReport( UserID, LogDay, TimeOUT, RemarkTimeOut, TerminalIPTimeOut, TerminalSNTimeOut,
                        PivotValue, LogDate )
                    values( @LogUserID, @LogDay, @LogTimestamp, @LogRemark, @LogTerminalIP, @LogTerminalSN, @PivotValue,
                        DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @LogTimestamp)))                                              
    
                    Set @FirstEvent = 0 -- not "first" anymore
                End
                Else -- Not first event
                Begin       
    
                    If @ReceiveInEvent = 1 -- if there are IN event previously
                    Begin           
                        -- Check day
                        If (@CurrentDay != @InEventDay) -- change to another day                        
                            Set @PivotValue = 0 -- Reset                        
                        Else
                            Set @PivotValue = @PivotValue + 1 -- increment
                        Set @CurrentDay = @InEventDay -- update the day     
                        Set @CurrTotalHours = @CurrTotalHours +  DATEDIFF(second,@InEventTimestamp, @LogTimeStamp) -- update total time             
    
                        Set @OutEventRemark = @LogRemark
                        Set @OutEventTerminalIP = @LogTerminalIP
                        Set @OutEventTerminalSN = @LogTerminalSN
                        Set @OutEventTimestamp = @LogTimestamp          
                        -- valid row
                        --Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, TimeOUT, TotalTime, RemarkTimeIn,
                        --  RemarkTimeOut, TerminalIPTimeIn, TerminalIPTimeOut, TerminalSNTimeIn, TerminalSNTimeOut, PivotValue,
                        --  LogDate)
                        --values( @LogUserID, @InEventDay, @InEventTimestamp, @LogTimestamp,  Datediff(second, @InEventTimestamp, @LogTimeStamp),
                        --  @InEventRemark, @LogRemark, @InEventTerminalIP, @LogTerminalIP, @InEventTerminalSN, @LogTerminalSN, @PivotValue,
                        --  DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))                      
    
                        Set @ReceiveInEvent = 0 -- Reset
                    End
                    Else -- no IN event previously
                    Begin
                        -- Check day
                        If (@CurrentDay != @LogDay) -- change to another day
                            Set @PivotValue = 0 -- Reset
                        Else
                            Set @PivotValue = @PivotValue + 1 -- increment
                        Set @CurrentDay = @LogDay -- update the day                         
    
                        -- invalid row (only has OUT event)
                        Insert into @tempInOutDetailReport( UserID, LogDay, TimeOUT, RemarkTimeOut, TerminalIPTimeOut, TerminalSNTimeOut,
                            PivotValue, LogDate )
                        values( @LogUserID, @LogDay, @LogTimestamp, @LogRemark, @LogTerminalIP, @LogTerminalSN, @PivotValue,
                            DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @LogTimestamp)) )             
                    End             
                End         
            End
    
            Set @CurrentUser = @LogUserID -- update user        
    
            Fetch Next 
            From LogCursor
            Into @LogUserID, @LogEventID, @LogTerminalSN, @LogTerminalIP, @LogRemark, @LogTimestamp, @LogDay
        End 
    
        -- Need to handle the last log if its IN log as it will not be processed by the while loop
        if @CurrTrigger='in'
        Begin 
        -- Check day
                If (@CurrentDay != @InEventDay) -- change to another day
                    Set @PivotValue = 0 -- Reset
                Else -- same day
                    Set @PivotValue = @PivotValue + 1 -- increment
                Set @CurrentDay = @InEventDay -- update the day
    
                -- invalid row (only has IN event)
                Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, RemarkTimeIn, TerminalIPTimeIn, 
                    TerminalSNTimeIn, PivotValue, LogDate )
                values( @InEventUserID, @InEventDay, @InEventTimestamp, @InEventRemark, @InEventTerminalIP, 
                        @InEventTerminalSN, @PivotValue, DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))     
        End
        else if @CurrTrigger = 'out'
        Begin
            Insert into @tempInOutDetailReport( UserID, LogDay, TimeIN, TimeOUT, TotalTime, RemarkTimeIn,
            RemarkTimeOut, TerminalIPTimeIn, TerminalIPTimeOut, TerminalSNTimeIn, TerminalSNTimeOut, PivotValue,
            LogDate)
            values( @LogUserID, @CurrentDay, @FirstInEvent, @LogTimestamp,  @CurrTotalHours,
            @FirstInRemark, @LogRemark, @FirstInTerminalIP, @LogTerminalIP, @FirstInTerminalSN, @LogTerminalSN, @PivotValue,
            DATEADD(HOUR, 0, DATEDIFF(DAY, 0, @InEventTimestamp)))  
        End
    
        Close LogCursor
        Deallocate LogCursor
    
        Select * 
        From @tempInOutDetailReport tempTable
            Left Join access_user on access_user.User_ID = tempTable.UserID
        Order By tempTable.UserID, LogDate
    
    
    End
    
    
    
    If @@TranCount > @TransactionCountOnEntry
    Begin
         If @ErrorCode = 0
                COMMIT TRANSACTION
         Else
                ROLLBACK TRANSACTION
    End
    
    return @ErrorCode
    

    you will get the "java SQL Code" by right click on stored procedure in your database. something like this :

    DECLARE @RC int
    DECLARE @pUserID nvarchar(50)
    DECLARE @pDepartmentID int
    DECLARE @pStartDateTime datetime
    DECLARE @pEndDateTime datetime
    
    -- TODO: Set parameter values here.
    
    EXECUTE @RC = [AccessManager].[dbo].[SP_GenerateInOutDetailReportSimple] 
    @pUserID,@pDepartmentID,@pStartDateTime,@pEndDateTime
    GO
    

    check the query String I've done, that is your homework ;) so sorry answering this long, this is my first answer since I register few weeks ago to get answer.

    0 讨论(0)
  • 2020-12-02 19:33

    FWIW, sp_test will not be returning anything but an integer (all SQL Server stored procs just return an integer) and no result sets on the wire (since no SELECT statements). To get the output of the PRINT statements, you normally use the InfoMessage event on the connection (not the command) in ADO.NET.

    0 讨论(0)
  • 2020-12-02 19:35

    Our server calls stored procs from Java like so - works on both SQL Server 2000 & 2008:

    String SPsql = "EXEC <sp_name> ?,?";   // for stored proc taking 2 parameters
    Connection con = SmartPoolFactory.getConnection();   // java.sql.Connection
    PreparedStatement ps = con.prepareStatement(SPsql);
    ps.setEscapeProcessing(true);
    ps.setQueryTimeout(<timeout value>);
    ps.setString(1, <param1>);
    ps.setString(2, <param2>);
    ResultSet rs = ps.executeQuery();
    
    0 讨论(0)
提交回复
热议问题