cari

Rumah  >  Soal Jawab  >  teks badan

Isih mengikut tarikh, gunakan IN untuk mendapatkan entri terkini dan sebelumnya untuk berbilang entri

<p>Matlamat saya: Saya mempunyai senarai <code>stock_id</code>s dan ingin mendapatkan <code>bid</code>s (diisih mengikut tarikh) untuk setiap <code> ;/kod> </p> <p>Untuk imej, ini bermakna saya mahu: </p> <table class="s-table"> <kepala> <tr> <th>stock_id</th> <th>Bida</th> </tr> </kepala> <tbody> <tr> <td>3</td> <td>663.91953</td> </tr> <tr> <td>1</td> <td>46.44281</td> </tr> <tr> <td>2</td> <td>9.02798</td> </tr> </tbody> </table> <p>Satu masalah ialah kami mempunyai stok seperti Gazprom yang digantung, jadi salah satu petikan terakhir mungkin, sebagai contoh, 2021-06-06. </p> <p>Mengambil tempat >quote_day = TARIKH(SEKARANG())</kod> pada <kod tidak berfungsi dalam kes ini. </p> <p>Saya juga memerlukan tarikh yang sama dengan tarikh bawah pertama yang tiada dalam pertanyaan pertama, ini boleh dilakukan dengan pertanyaan kedua. </p> <p>Penyelesaian semasa saya menggunakan PHP. Ini berfungsi, tetapi prestasinya tidak sempurna, seperti 100 stok mengambil masa 5 saat. </p> <p>Saya boleh menggunakan Redis, yang juga mempunyai pilihan untuk menyimpan bida di suatu tempat. </p> <p>Semasa:</p> <pre class="lang-sql prettyprint-override"><kod>pilih `tarikh_sebut harga`, 'saham' sebagai `jenis`, `bida`, `id_saham` sebagai id daripada ( pilih t.*, row_number() over(partition by stock_id order by `quote_date` desc) as rn daripada end_day_quotes_AVG t di mana quote_date <= DATE({$date}) DAN stok_id dalam ({$val}) dan currency_id = {$c_id} ) x di mana rn = 1 </code></pre> <p>Sehari sebelumnya: </p> <pre class="lang-sql prettyprint-override"><kod>pilih `tarikh_sebut harga`, 'saham' sebagai `jenis`, `bida`, `id_saham` sebagai id daripada ( pilih t.*, row_number() over(partition by stock_id order by `quote_date` desc) as rn daripada end_day_quotes_AVG t di mana quote_date < DATE({$date}) DAN stok_id dalam ({$val}) dan currency_id = {$c_id} ) x di mana rn = 1 </code></pre> <p><kod>Stock_id</code>, <code>quote_date</code> dan <code>currency_id</code></p> <p>我想要使用服务器数据的表:10.9.4-MariaDB-1:10.9.4</p> <p>编辑:</p> <p>解释的查询:</p> <pre class="brush:php;toolbar:false;">id select_type table type possible_keys key key_len ref rows Tambahan 1 UTAMA <berasal2> ALL NULL NULL NULL NULL 220896 Menggunakan mana 2 TERUSKAN t SEMUA stock_id,quote_date NULL NULL NULL 2173105 Menggunakan mana; Menggunakan sementara</pre> <p>创建表:</p> <pre class="brush:php;toolbar:false;">BUAT JADUAL `tanda_akhir_hari_AVG` ( `id` int(11) BUKAN NULL, `tarikh_petik` tarikh BUKAN NULL, `bida` perpuluhan(15,5) BUKAN NULL, `id_saham` int(11) LALAI NULL, `etf_id` int(11) LALAI NULL, `id_kripto` int(11) LALAI NULL, `id_sijil` int(11) LALAI NULL, `id_mata wang` int(11) BUKAN NULL ) ENJIN=CHARSET LALAI InnoDB=utf8mb4 COLLATE=utf8mb4_general_ci; INSERT IN TO `end_day_quotes_AVG` (`id`, `quote_date`, `bid`, `stock_id`, `etf_id`, `crypto_id`, `certificate_id`, `currency_id`) NILAI (10537515, '2023-01-02', '16.48286', 40581, NULL, NULL, NULL, 2), (10537514, '2023-01-02', '3.66786', 40569, NULL, NULL, NULL, 2), (10537513, '2023-01-02', '9.38013', 40400, NULL, NULL, NULL, 2), (10537512, '2023-01-02', '8.54444', 40396, NULL, NULL, NULL, 2), ALTER JADUAL `petikan_akhir_hari_AVG` TAMBAHKAN KUNCI UTAMA (`id`), TAMBAHKAN KUNCI `stock_id` (`stock_id`,`currency_id`), TAMBAHKAN KUNCI `etf_id` (`etf_id`,`currency_id`), TAMBAHKAN KUNCI `crypto_id` (`crypto_id`,`currency_id`), TAMBAHKAN KUNCI `id_sijil` (`id_sijil`,`id_mata wang`), TAMBAHKAN KUNCI `tarikh_petikan` (`tarikh_petikan`); ALTER JADUAL `petikan_akhir_hari_AVG` UBAHSUAI `id` int(11) BUKAN NULL AUTO_INCREMENT, AUTO_INCREMENT=10570526;</pre> <p>生成的填充查询:</p> <pre class="brush:php;toolbar:false;">pilih `quote_date`, 'stok' sebagai `type`, `bid`, `stock_id` sebagai id daripada ( pilih t.*, row_number() over(partition by stock_id order by `quote_date` desc) as rn dari tarikh_akhir_hari_AVG t di mana tarikh_sebut harga <= TARIKH('2023-01-02') DAN stok_id dalam (2,23,19,41,40,26,9,43,22, 44,28,32,30,34,20,10,13,17,27,35,8,29,39,16,33,5,36589,25,18,6,38,37,3,45, 7,21,46,15,4,24,31,36,38423,40313, 22561,36787,35770,36600,35766,42,22567,40581,40569,29528,22896,24760,40369,40396,40400,40374,367964,36796 29659,40367,27821,24912,36654,21125,22569,22201, 23133,40373,36697,36718,26340,36653,47,34019,36847,36694) dan currency_id = 2 ) x dengan rn = 1;</pre></p>
P粉653045807P粉653045807436 hari yang lalu634

membalas semua(2)saya akan balas

  • P粉340980243

    P粉3409802432023-09-04 18:30:57

    Adakah anda sedang mencari Dua sebut harga terkini untuk setiap tawaran pada tarikh tertentu? Jika ya, anda hanya boleh mengubah suai pertanyaan pertama untuk membenarkan nombor baris 1 dan 2:

    select `quote_date`, 'stocks' as `type`, `bid`, `stock_id` as id 
    from ( 
        select t.*, row_number() over(partition by stock_id order by quote_date desc) as rn f
        from end_day_quotes_AVG t 
        where quote_date <= DATE(?) AND stock_id in (?)  and currency_id = ? 
    ) x 
    where rn <= 2  -- the latest two

    balas
    0
  • P粉899950720

    P粉8999507202023-09-04 12:31:54

    Untuk mendapatkan bida terakhir (sebelum tarikh tertentu) dan bida kedua terakhir untuk setiap mata wang/saham dalam satu pertanyaan, dan dengan cekap menggunakan indeks pada currency_id, stock_id, quote_date, anda boleh melakukan ini secara berperingkat: mula-mula cari Tarikh maksimum untuk setiap mata wang /saham ( akan menggunakan indeks), kemudian cari tarikh sebelumnya (sekali lagi, dengan cara yang sama seperti menggunakan indeks), kemudian cari bida sebenar:

    with stock_ids(stock_id) as (
        values (2),(23),(19),(41),(40),(26),(9),(43),
               (22),(44),(28),(32),(30),(34),(20),(10),
               (13),(17),(27),(35),(8),(29),(39),(16),
               (33),(5),(36589),(25),(18),(6),(38),(37),
               (3),(45),(7),(21),(46),(15),(4),(24),
               (31),(36),(38423),(40313),(22561),(36787),(35770),(36600),
               (35766),(42),(22567),(40581),(40569),(29528),(22896),(24760),
               (40369),(40396),(40400),(40374),(36799),(1),(27863),(29659),
               (40367),(27821),(24912),(36654),(21125),(22569),(22201),(23133),
               (40373),(36697),(36718),(26340),(36653),(47),(34019),(36847),
               (36694)
    ),
    last_dates as (
        select t.currency_id, t.stock_id, max(t.quote_date) as quote_date
        from stock_ids
        join end_day_quotes_AVG t on
            t.currency_id=2 and
            t.stock_id=stock_ids.stock_id and
            t.quote_date <= '2023-01-31'
        group by t.currency_id,t.stock_id
    ),
    next_to_last_dates as (
        select t.currency_id, t.stock_id, max(t.quote_date) as quote_date
        from last_dates l
        join end_day_quotes_AVG t on
            t.currency_id=l.currency_id and
            t.stock_id=l.stock_id and
            t.quote_date < l.quote_date
        group by t.currency_id,t.stock_id
    )
    select 'last' as 'when', currency_id, stock_id, quote_date, bid
    from last_dates
    join end_day_quotes_AVG using (currency_id, stock_id, quote_date)
    union all
    select 'next-to-last', currency_id, stock_id, quote_date, bid
    from next_to_last_dates
    join end_day_quotes_AVG using (currency_id, stock_id, quote_date)

    Jika anda mahukan lebih daripada sekadar dua tarikh terbaharu untuk setiap stok, anda mungkin boleh menggantikan tarikh_akhir/seterusnya_tarikh_akhir dengan cte rekursif yang mengandungi bilangan hari (terhad kepada bilangan hari yang anda mahu kumpulkan).

    Violin

    balas
    0
  • Batalbalas