java 如何读取text/event-stream


在Java中读取`text/event-stream`(Server-Sent Events, SSE)可以使用`HttpURLConnection`或第三方库如`OkHttp`。以下是两种实现方式: 1. 使用原生HttpURLConnection ```java import java.io.BufferedReader; import java.io.InputStreamReader; import java.net.HttpURLConnection; import java.net.URL; public class SSEClient { public static void main(String[] args) throws Exception { URL url = new URL("http://example.com/sse-endpoint"); HttpURLConnection connection = (HttpURLConnection) url.openConnection(); connection.setRequestMethod("GET"); connection.setRequestProperty("Accept", "text/event-stream"); try (BufferedReader reader = new BufferedReader( new InputStreamReader(connection.getInputStream()))) { String line; while ((line = reader.readLine()) != null) { // 处理SSE事件(格式为 "data: {json}" 或 "event: xxx") if (line.startsWith("data:")) { System.out.println("Received: " + line.substring(5).trim()); } } } } } ``` 2. 使用OkHttp(推荐) ```java import okhttp3.OkHttpClient; import okhttp3.Request; import okhttp3.Response; import okhttp3.sse.EventSource; import okhttp3.sse.EventSourceListener; import okhttp3.sse.EventSources; public class OkHttpSSE { public static void main(String[] args) { OkHttpClient client = new OkHttpClient(); Request request = new Request.Builder() .url("http://example.com/sse-endpoint") .header("Accept", "text/event-stream") .build(); EventSourceListener listener = new EventSourceListener() { @Override public void onEvent(EventSource eventSource, String id, String type, String data) { System.out.println("SSE Event: " + data); } }; EventSources.createFactory(client).newEventSource(request, listener); } } ``` 关键点说明: 1. 连接保持:SSE是长连接,服务器会持续推送数据,需确保连接不会超时或被关闭。 2. 事件格式:SSE数据以`data:`开头,可能包含`event:`(事件类型)、`id:`(事件ID)等字段。 3. 错误处理:需监听连接中断或错误事件(如`onFailure`方法中实现重连逻辑)。 --- 额外注意事项: - 如果使用Spring框架,可考虑`SseEmitter`或WebFlux的`ServerSentEvent`类。 - 生产环境建议添加连接超时和重试机制。 - 注意线程阻塞问题,异步处理更高效。 [2025-06-30 10:38:40 | AI写代码神器 | 599点数解答]
- skill={ trigger:{ global:"useCard1", }, audio:2, forced:true, firstDo:true, filter:function(event,player,card){ if(get.color(event.card)!='black') return false; return event.card.name=='nanman'&&player!=event.player||event.card.name=='wanjian'&&player!=event.player||event.card.name=='taoyuan'&&player.hp<player.maxHp||event.card.name=='wugu'; }, content:function(){}, mod:{ targetEnabled:function(card){ if((get.type(ca(211点数解答 | 2025-02-01 13:23:26)173
- import javax.swing.*; import java.awt.*; import java.awt.datatransfer.clipboard; import java.awt.datatransfer.stringselection; import java.awt.datatransfer.transferable; import java.awt.event.*; import java.io.*; import java.nio.file.files; import java.nio.file.path; import java.nio.file.paths; import java.time.localdatetime; import java.util.hashmap; import java.util.list; import java.util.map; import java.util.random; public class copy { static private final jtextarea textarea = new jtext(1497点数解答 | 2024-08-25 09:40:33)292
- content: async function(event, trigger, player) { const [target] = event.targets; const [card] = event.cards; trigger.cancel(); await player.discard(event.cards); const { result } = await player.chooseControlList( true, function(event, player) { const target = _status.event.target; let att = get.attitude(player, target); if (target.hasSkillTag("maihp")) att = -att; return att > 0 ? 0 : 1; }, ["令" (179点数解答 | 2025-05-17 20:49:48)108
- 这是无名杀的一个技能,如何让他去除不可声明限定技、觉醒技、隐匿技、使命技、主公技等特殊技能的限制skill={ unique: true, audio: 2, trigger: { global: "phaseBefore", player: ["enterGame","phaseBegin","phaseEnd"], }, filter(event, player, name) { if (event.name != "phase") return true; if (name == "phaseBefore") return game.phaseNumber == 0; return player.storage.rehuashen?.character?.length > 0; }, async cost(event, trigger, player) { if (trigger.name !== "phase" || event.triggername === "phaseBefore") { event.result = { bool: true, cost_data: ["(556点数解答 | 2025-06-29 08:59:58)84
- skill={ audio:2, trigger:{ player:"useCard", }, frequent:true, filter:function (event,player){ if(!event.cards||event.cards.length!=1) return false; if(_status.currentPhase!=player) return false; if(!player.storage.jianying) return false; return get.suit(player.storage.jianying)==get.suit(event.cards[0])|| player.storage.jianying.number==event.cards[0].number; }, content:function (){ player.draw(); }, (713点数解答 | 2025-07-05 23:27:14)67
- #include "videoplayer.h" #include "log.h" #pragma warning(disable: 4996) //buffer: static int audio_len = 0; static uint8* audio_pos; void videoplayer::audiocallback(void* userdata, uint8_t* stream, int len) { sdl_memset(stream, 0, len); if (audio_len <= 0) return; len = (len > audio_len ? audio_len : len); /* mix as much data as possible */ sdl_mixaudio(stream, audio_pos, len, sdl_mix_maxvolume); audio_pos += len; audio_len -= len; } videoplayer::videop(549点数解答 | 2024-08-08 15:59:25)194
- #include<iostream> #include<complex.h> #include <complex> #include <math.h> using namespace std; #define pi 3.14159265358979323846 int main() { file* stream; stream = fopen("4000-red.raw","rb"); unsigned char* image = (unsigned char*)malloc(4000 * 4000 * sizeof(unsigned char)); fread(image,4000*4000,sizeof(unsigned char),stream); unsigned char* image2 = (unsigned char*)malloc(4000 * 4000 * sizeof(unsigned char)); unsigned char* image3 = (unsigned char*)malloc(4000 * 4000 * sizeof(unsigned(413点数解答 | 2024-11-02 18:16:35)188
- dm.jdbc.driver.dmexception: 数据溢出 at dm.jdbc.driver.dberror.throwexception(dberror.java:636) at dm.jdbc.c.b.p.s(msg.java:225) at dm.jdbc.c.b.p.p(msg.java:185) at dm.jdbc.c.b.p.o(msg.java:166) at dm.jdbc.c.a.a(dbaccess.java:776) at dm.jdbc.c.a.a(dbaccess.java:327) at dm.jdbc.c.a.a(dbaccess.java:435) at dm.jdbc.driver.dmdbpreparedstatement.executeinner(dmdbpreparedstatement.java:284) at dm.jdbc.driver.dmdbpreparedstatement.do_executequery(dmdbpreparedstatement.java:383) at dm.jdbc.driver.(598点数解答 | 2024-10-15 15:48:38)195
- 问题排查:dm.jdbc.driver.dmexception: 数据溢出 at dm.jdbc.driver.dberror.throwexception(dberror.java:636) at dm.jdbc.c.b.p.s(msg.java:225) at dm.jdbc.c.b.p.p(msg.java:185) at dm.jdbc.c.b.p.o(msg.java:166) at dm.jdbc.c.a.a(dbaccess.java:776) at dm.jdbc.c.a.a(dbaccess.java:327) at dm.jdbc.c.a.a(dbaccess.java:435) at dm.jdbc.driver.dmdbpreparedstatement.executeinner(dmdbpreparedstatement.java:284) at dm.jdbc.driver.dmdbpreparedstatement.do_executequery(dmdbpreparedstatement.java:383) at dm.jdbc.dr(403点数解答 | 2024-10-15 15:48:41)843
- jingyu: { audio: 2, trigger: { global: ["useskill", "logskillbegin", "usecard", "respond"], }, filter(event, player) { if (["global", "equip"].includes(event.type)) return false; let skill = event.sourceskill || event.skill; if (!skill || skill === "jingyu") return false; let info = get.info(skill); while (true) { if (!info || info.charlotte || info.equipskill) return false; if (info && !info.sourceskill) break; skill = info.sourceskill; info = get.info((15点数解答 | 2024-06-06 17:40:23)248
- content() { "step 0"; player.loseHp(); event.target = trigger.player; "step 1"; event.addIndex = 0; var list = [], num = target.countCards("h") - player.countCards("h"); event.num = num; if (num > 0 && target.countCards("h") > 0) list.push("令其弃置" + get.cnNumber(num) + "张手牌"); else event.addIndex++; if (target.hp > player.hp) list.push("令其失去" + get.cnNumber(target.hp - player.hp) + "点体力"); if (!lis(313点数解答 | 2025-05-28 16:03:47)104
- skill={ unique: true, audio: 2, trigger: { global: "phaseBefore", player: ["enterGame","phaseBegin","phaseEnd"], }, filter(event, player, name) { if (event.name != "phase") return true; if (name == "phaseBefore") return game.phaseNumber == 0; return player.storage.rehuashen?.character?.length > 0; }, async cost(event, trigger, player) { if (trigger.name !== "phase" || event.triggername === "phaseBefore") { ev(1573点数解答 | 2025-06-29 08:55:47)57