Запрос не выдает результирующий набор при попытке удалить по дате. - PullRequest
0 голосов
/ 03 мая 2018

Я пытаюсь создать метод в Java, который будет удалять все записи в базе данных до определенной даты. Прямо сейчас я получаю SQLException: запрос или процедура не возвращает набор результатов.

Это один из операторов вставки, который мой модульный тест использует для заполнения базы данных.

INSERT INTO C35_6 ( xIndex, xcSource, dTimeOn, dTimeOff, bSuccessful ) 
VALUES (74013, 1, '05/01/1972 00:00:00.000', '05/02/1972 00:00:00.000', 1 )

А вот оператор удаления

DELETE FROM C35_6 WHERE dTimeOff <= '12/31/1975 00:00:00.000'

Я вижу в файле журнала, что эти операторы выполняются в правильном порядке, сразу друг за другом. Эти операторы создаются с помощью подготовленного java с использованием setDate ().

Кажется, что sql по какой-то причине не может сравнить эти даты.

Ответы [ 2 ]

0 голосов
/ 05 мая 2018

Ответ от Elevate является верным, поскольку причина вашего исключения не связана с обработкой даты и времени.

Но у вас есть проблемы с обработкой даты и времени.

  • Игнорирование важного вопроса о часовом поясе.
  • Использование немых строк вместо смарт-объектов.

Вот полный пример использования базы данных H2 .

A ZonedDateTime объект представляет собой момент, точку на временной шкале, видимую через время настенных часов, используемое людьми определенного региона ( часовой пояс ).

Мы можем увидеть тот же самый момент, ту же точку на временной шкале через время настенных часов UTC , просто извлекая Instant из нашего ZonedDateTime. Драйвер JDBC для базы данных H2 требует Instant, так как его метод PreparedStatement::setObject не может напрямую работать с ZonedDateTime. Большинство баз данных, включая H2, хранят TIMESTAMP WITH TIME ZONE в UTC. Некоторые корректируют входящие значения из другого часового пояса в UTC, но, по-видимому, H2 требует, чтобы мы выполнили настройку.

package com.basilbourque.example;

import org.h2.jdbcx.JdbcDataSource;

import java.sql.*;
import java.time.*;

public class App {
    public static void main ( String[] args ) {
        App app = new App();
        app.doIt();
    }

    private void doIt () {
        JdbcDataSource ds = new JdbcDataSource();
        ds.setURL( "jdbc:h2:mem:bogus;DATABASE_TO_UPPER=FALSE;DB_CLOSE_DELAY=-1" );  // Create in-memory database that disappears when this JVM exits.
        ds.setUser( "sa" );  // Arbitrary.
        ds.setPassword( "sa" );

        ZoneId z = ZoneId.of( "America/Montreal" ); // Arbitrarily chosen time zone.

        // Define "event" table with 3 columns: id number auto-generated, beginning of event, ending of event.
        String sql = "CREATE TABLE event_  ( id_ INTEGER GENERATED ALWAYS AS IDENTITY PRIMARY KEY , start_ TIMESTAMP WITH TIME ZONE NOT NULL, stop_ TIMESTAMP WITH TIME ZONE NOT NULL ) ;";
        try (
                Connection conn = ds.getConnection() ;
                Statement stmt = conn.createStatement() ;
        ) {
            stmt.executeUpdate( sql );
        } catch ( SQLException e ) {
            e.printStackTrace();
        }

        // Insert rows for years 1972, 1977, & 1973.
        sql = "INSERT INTO event_ (  start_ , stop_ ) VALUES ( ? , ? ) ; ";
        try (
                Connection conn = ds.getConnection() ;
                PreparedStatement ps = conn.prepareStatement( sql ) ;
        ) {
            ps.setObject( 1 , LocalDate.of( 1972 , Month.MAY , 1 ).atStartOfDay( z ).toInstant() );
            ps.setObject( 2 , LocalDate.of( 1972 , Month.MAY , 2 ).atStartOfDay( z ).toInstant() );
            ps.executeUpdate();

            ps.setObject( 1 , LocalDate.of( 1977 , Month.JUNE , 1 ).atStartOfDay( z ).toInstant() );
            ps.setObject( 2 , LocalDate.of( 1977 , Month.JULY , 2 ).atStartOfDay( z ).toInstant() );
            ps.executeUpdate();

            ps.setObject( 1 , LocalDate.of( 1973 , Month.AUGUST , 1 ).atStartOfDay( z ).toInstant() );
            ps.setObject( 2 , LocalDate.of( 1973 , Month.AUGUST , 2 ).atStartOfDay( z ).toInstant() );
            ps.executeUpdate();


        } catch ( SQLException e ) {
            e.printStackTrace();
        }


        // Dump table.
        System.out.println( "Dumping event_ table. " );
        sql = "SELECT * FROM event_  ; ";
        try (
                Connection conn = ds.getConnection() ;
                PreparedStatement ps = conn.prepareStatement( sql ) ;
                ResultSet rs = ps.executeQuery() ;
        ) {
            while ( rs.next() ) {
                System.out.println( "id_:  " + rs.getInt( 1 ) + " | " + rs.getObject( 2 , Instant.class ).atZone( z ) + " to " + rs.getObject( 3 , Instant.class ).atZone( z ) );
            }
        } catch ( SQLException e ) {
            e.printStackTrace();
        }

        // Deleting rows for events starting before beginning of 1976. Should delete 2 of our 3 rows, leaving 1 row.
        System.out.println( "Deleting rows. " );
        sql = "DELETE FROM event_ WHERE start_ < ? ; ";
        try (
                Connection conn = ds.getConnection() ;
                PreparedStatement ps = conn.prepareStatement( sql ) ;
        ) {
            ps.setObject( 1 , Year.of( 1976 ).atDay( 1 ).atStartOfDay( z ).toInstant() );  // First moment of 1976 in our time zone.
            ps.executeUpdate();
        } catch ( SQLException e ) {
            e.printStackTrace();
        }

        // Dump table.
        System.out.println( "Dumping event_ table. " );
        sql = "SELECT * FROM event_  ; ";
        try (
                Connection conn = ds.getConnection() ;
                PreparedStatement ps = conn.prepareStatement( sql ) ;
                ResultSet rs = ps.executeQuery() ;
        ) {
            while ( rs.next() ) {
                System.out.println( "id_:  " + rs.getInt( 1 ) + " | " + rs.getObject( 2 , Instant.class ).atZone( z ) + " to " + rs.getObject( 3 , Instant.class ).atZone( z ) );
            }
        } catch ( SQLException e ) {
            e.printStackTrace();
        }


    }
}

при запуске.

Сброс таблицы событий.

id_: 1 | 1972-05-01T00: 00-04: 00 [Америка / Монреаль] до 1972-05-02T00: 00-04: 00 [Америка / Монреаль]

id_: 2 | 1977-06-01T00: 00-04: 00 [Америка / Монреаль] - 1977-07-02T00: 00-04: 00 [Америка / Монреаль]

id_: 3 | 1973-08-01T00: 00-04: 00 [Америка / Монреаль] - 1973-08-02T00: 00-04: 00 [Америка / Монреаль]

Удаление строк.

Сброс таблицы событий.

id_: 2 | 1977-06-01T00: 00-04: 00 [Америка / Монреаль] - 1977-07-02T00: 00-04: 00 [Америка / Монреаль]


О java.time

Фреймворк java.time встроен в Java 8 и более поздние версии. Эти классы вытесняют проблемные старые устаревшие классы даты и времени, такие как java.util.Date, Calendar, & SimpleDateFormat.

Проект Joda-Time , теперь в режиме обслуживания , рекомендует выполнить переход на классы java.time .

Чтобы узнать больше, см. Oracle Tutorial . И поиск переполнения стека для многих примеров и объяснений. Спецификация JSR 310 .

Вы можете обмениваться java.time объектами напрямую с вашей базой данных. Используйте драйвер JDBC , совместимый с JDBC 4.2 или более поздней версии. Нет необходимости в строках, нет необходимости в java.sql.* классах.

Где получить классы java.time?

Проект ThreeTen-Extra расширяет java.time дополнительными классами. Этот проект является полигоном для возможных будущих дополнений к java.time. Здесь вы можете найти некоторые полезные классы, такие как Interval, YearWeek, YearQuarter и more .

0 голосов
/ 03 мая 2018

Я думаю, тот факт, что вы устанавливаете дату, не имеет значения. Сообщение об исключении говорит, что «запрос не возвращает набор результатов». Это правда. Оба ваших запроса не возвращают результатов.

Смысл в том, что вы выполняете их неправильно. Я предполагаю, что вы запускаете их с preparedStatement.executeQuery() (что для SELECT запросов и ожидает возврата ResultSet), а не preparedStatement.execute(), который можно использовать для запросов любого типа.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...