0

0

Java中处理HTTP 301重定向:安全读取JSON数据的实践指南

花韻仙語

花韻仙語

发布时间:2025-07-19 13:46:14

|

583人浏览过

|

来源于php中文网

原创

Java中处理HTTP 301重定向:安全读取JSON数据的实践指南

在Java应用中通过URL读取JSON数据时,若遇到301 Moved Permanently重定向导致JSONException,通常是由于使用了不安全的HTTP协议访问了已迁移至HTTPS的资源。解决方案是直接将URL协议从http更改为https,并建议使用HttpURLConnection进行更健壮的网络请求处理,以有效避免此类数据解析错误。

理解问题:HTTP 301重定向与JSON解析失败

在进行网络请求时,我们可能会遇到http状态码为301 moved permanently的情况。这意味着请求的资源已经被永久性地移动到新的url。当客户端使用http://协议访问一个资源,而该资源已被服务器配置为强制使用https://协议时,服务器会返回一个301响应,告知客户端应使用新的https地址。

原始代码中,使用URL.openStream()方法来读取URL内容:

private static String readUrl(String urlString) throws Exception {
    BufferedReader reader = null;
    try {
        URL url = new URL(urlString);
        reader = new BufferedReader(new InputStreamReader(url.openStream()));
        StringBuffer buffer = new StringBuffer();
        int read;
        char[] chars = new char[1024];
        while ((read = reader.read(chars)) != -1)
            buffer.append(chars, 0, read);

        return buffer.toString();
    } finally {
        if (reader != null)
            reader.close();
    }
}

当请求的URL(例如http://webservice.fanart.tv/v3/movies/...)触发301重定向时,URL.openStream()在某些情况下可能会尝试自动跟踪重定向。然而,问题在于,服务器返回的301响应体通常包含一个简单的HTML页面,指示资源已移动,而不是预期的JSON数据。当后续代码尝试将这个HTML内容解析为JSONObject时,就会抛出org.json.JSONException: Value of type java.lang.String cannot be converted to JSONObject异常。这清楚地表明,接收到的数据不是JSON格式,而是HTML。

核心解决方案:强制使用HTTPS协议

解决此问题的最直接和最有效的方法是,在构建URL时,直接使用https://协议而非http://。许多API提供商为了数据传输的安全性,会强制要求使用HTTPS。当客户端直接使用HTTPS请求时,就避免了最初的HTTP到HTTPS的重定向过程,从而直接获取到正确的JSON数据。

将原始URL字符串:

立即学习Java免费学习笔记(深入)”;

String url = "http://webservice.fanart.tv/v3/movies/" + movie.id + "?api_key=" + apikey;

修改为:

String url = "https://webservice.fanart.tv/v3/movies/" + movie.id + "?api_key=" + apikey;

通过这一简单的更改,应用程序将直接向API的HTTPS端点发起请求,绕过不必要的重定向,并有望直接接收到期望的JSON响应。

Cutout.Pro抠图
Cutout.Pro抠图

AI批量抠图去背景

下载

更健壮的网络请求:使用HttpURLConnection

尽管直接切换到HTTPS可以解决当前问题,但为了构建更健壮、更可控的网络请求逻辑,建议使用java.net.HttpURLConnection而非简单的URL.openStream()。HttpURLConnection提供了对HTTP请求的更多控制,包括获取响应状态码、设置请求头、处理重定向等。

以下是使用HttpURLConnection改进readUrl方法的示例:

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;

public class JsonDataReader {

    private static String readUrl(String urlString) throws Exception {
        HttpURLConnection connection = null;
        BufferedReader reader = null;
        try {
            URL url = new URL(urlString);
            connection = (HttpURLConnection) url.openConnection();

            // 设置请求方法
            connection.setRequestMethod("GET");
            // 允许自动重定向(HttpURLConnection默认开启,但明确设置更清晰)
            connection.setInstanceFollowRedirects(true);
            // 设置连接和读取超时(可选,但推荐)
            connection.setConnectTimeout(5000); // 5秒连接超时
            connection.setReadTimeout(10000);  // 10秒读取超时

            int responseCode = connection.getResponseCode();
            // 检查响应码,确保是成功状态(200 OK)或重定向后成功
            if (responseCode >= 200 && responseCode < 300) {
                reader = new BufferedReader(new InputStreamReader(connection.getInputStream()));
                StringBuilder buffer = new StringBuilder();
                String line;
                while ((line = reader.readLine()) != null) {
                    buffer.append(line);
                }
                return buffer.toString();
            } else {
                // 处理非成功响应,例如打印错误信息
                String errorMessage = "HTTP Error: " + responseCode + " - " + connection.getResponseMessage();
                // 如果需要,可以读取错误流
                try (BufferedReader errorReader = new BufferedReader(new InputStreamReader(connection.getErrorStream()))) {
                    String errorLine;
                    StringBuilder errorBuffer = new StringBuilder();
                    while ((errorLine = errorReader.readLine()) != null) {
                        errorBuffer.append(errorLine);
                    }
                    errorMessage += "\nError Body: " + errorBuffer.toString();
                } catch (Exception e) {
                    // 忽略读取错误流的异常
                }
                throw new Exception(errorMessage);
            }
        } finally {
            if (reader != null) {
                try {
                    reader.close();
                } catch (Exception e) {
                    e.printStackTrace(); // 打印关闭流的异常
                }
            }
            if (connection != null) {
                connection.disconnect(); // 关闭连接
            }
        }
    }

    public static void main(String[] args) {
        // 假设 movie.id 和 apikey 已定义
        String movieId = "629542"; // 示例ID
        String apikey = "YOUR_API_KEY"; // 请替换为你的API密钥
        String url = "https://webservice.fanart.tv/v3/movies/" + movieId + "?api_key=" + apikey;

        try {
            String jsonString = readUrl(url);
            // 示例:解析JSON数据
            org.json.JSONObject json = new org.json.JSONObject(jsonString);
            org.json.JSONArray jsonArray = json.getJSONArray("hdmovielogo");
            java.util.List enClearLogos = new java.util.ArrayList<>();
            for(int i = 0; i < jsonArray.length(); i++){
                org.json.JSONObject movieObject = jsonArray.getJSONObject(i);
                if (movieObject.getString("lang").equalsIgnoreCase("en"))
                    enClearLogos.add(movieObject.getString("url"));
            }
            System.out.println("英文高清Logo URLs: " + enClearLogos);

        } catch (Exception e) {
            System.err.println("Error reading JSON data: " + e.getMessage());
            e.printStackTrace();
        }
    }
}

代码改进点说明:

  1. 使用HttpURLConnection: 提供更细粒度的控制,可以检查HTTP响应码。
  2. 响应码检查: 在读取输入流之前,先检查connection.getResponseCode()。只有当响应码为2xx(成功)时才尝试解析JSON。对于3xx重定向,HttpURLConnection通常会默认自动跟随,但明确检查有助于调试。对于4xx或5xx错误,可以抛出更具体的异常。
  3. 超时设置: setConnectTimeout()和setReadTimeout()有助于防止网络请求无限期挂起。
  4. 资源关闭: 在finally块中确保BufferedReader和HttpURLConnection都被正确关闭,释放资源。

JSON数据解析与异常处理

在获取到JSON字符串后,进行解析时也需要注意健壮性。org.json库在遇到非JSON格式的数据时会抛出JSONException。因此,将JSON解析部分放入try-catch块中是至关重要的。

// ... (前略,已获取到 jsonString)
try {
    JSONObject json = new JSONObject(jsonString);
    // 确保键存在且是JSONArray类型
    if (json.has("hdmovielogo") && json.get("hdmovielogo") instanceof JSONArray) {
        JSONArray jsonArray = json.getJSONArray("hdmovielogo");
        List enClearLogos = new ArrayList<>();
        for(int i = 0; i < jsonArray.length(); i++){
            JSONObject movieObject = jsonArray.getJSONObject(i);
            if (movieObject.has("lang") && movieObject.getString("lang").equalsIgnoreCase("en")) {
                if (movieObject.has("url")) {
                    enClearLogos.add(movieObject.getString("url"));
                }
            }
        }
        // ... 处理 enClearLogos
    } else {
        System.err.println("JSON structure invalid: 'hdmovielogo' not found or not a JSONArray.");
    }
} catch (JSONException e) {
    System.err.println("Error parsing JSON data: " + e.getMessage());
    e.printStackTrace();
} catch (Exception e) { // 捕获其他可能的异常
    System.err.println("An unexpected error occurred: " + e.getMessage());
    e.printStackTrace();
}

JSON解析注意事项:

  • has()方法: 在尝试获取JSON对象或数组之前,使用JSONObject.has("key")方法检查键是否存在,避免JSONException。
  • 类型检查: 使用instanceof或get(key)返回的类型判断,确保获取到的值是预期的类型(如JSONArray或JSONObject)。
  • 细致的异常处理: 不仅捕获JSONException,还要考虑其他网络或IO相关的异常。

注意事项与最佳实践

  1. 始终验证API文档: 在集成任何第三方API时,仔细阅读其官方文档至关重要。文档会明确指出推荐的协议(HTTP/HTTPS)、认证方式、请求限制以及可能返回的错误码。
  2. 网络请求超时: 为网络连接和数据读取设置合理的超时时间,防止因网络问题导致应用程序长时间无响应。
  3. 使用现代HTTP客户端库: 对于更复杂的应用,考虑使用成熟的第三方HTTP客户端库,如Apache HttpClient、OkHttp或Spring的RestTemplate/WebClient。这些库提供了更高级的功能(如连接池、拦截器、异步请求、更好的错误处理机制)和更简洁的API,可以大大简化网络请求的开发和维护。
  4. 日志记录: 记录网络请求的URL、响应状态码、耗时以及任何异常信息,这对于问题诊断和性能监控非常有帮助。

通过遵循这些实践,可以有效避免因URL重定向和不当的JSON解析而导致的常见问题,确保应用程序能够稳定、安全地与外部API进行交互。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

825

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

724

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

728

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

395

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

428

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16861

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
RunnerGo从入门到精通
RunnerGo从入门到精通

共22课时 | 1.7万人学习

尚学堂Mahout视频教程
尚学堂Mahout视频教程

共18课时 | 3.2万人学习

Linux优化视频教程
Linux优化视频教程

共14课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号