• 前言
  • 问题追踪
  • 总结


前言

前面通过腾讯云ssl证书实现了https的请求,如果想简单了解可参考如何添加SSL证书实现https请求;为了满足之前http请求不受影响,在nginx上面不仅配置了https的443端口的监听,同时监听了80端口。通过浏览器测试确实是没有问题,但是在写代码时,java代码发起get,post请求的时候,出现了“301 Moved Permanently”的错误;网上也没有搜索到好的解决方案,经过一下午的分析,找到了一个突破口,但是总觉得并不是一个好的方式,再此分享一下,如果有人能有更好的方式解决此问题,烦请告知一下。

问题追踪

  • 获取HeaderFields
    当我们通过一个http的地址得到一个URLConnection连接之后;通过getHeaderFields()获取这个连接的响应头,代码如下:
String urlName = url + "?" + param;
    URL realUrl = new URL(urlName);
    // 打开和URL之间的连接
    URLConnection conn = realUrl.openConnection();
    Map<String, List<String>> map = conn.getHeaderFields();
    // 遍历所有的响应头字段
    for (String key : map.keySet())
    {
        System.out.println(key + "--->" + map.get(key).get(0));
    }
  • 如果是http的请求我们可以看到如下信息:
  • nginx返回code加messge nginx返回301_nginx返回code加messge

  • 这里给出了新的重定向后的地址
  • 如果是https的请求,就不会出现这个Location字段;并且能正常请求成功,效果如下:
  • nginx返回code加messge nginx返回301_rewrite_02

  • 浏览器能正常请求个人认为是在出现错误之后,浏览器发现存在新地址Location字段;就取到了新地址重新发起了一次新的请求从而实现了正常的响应。
  • 解决方案
    既然上面说道浏览器检测到Location字段,拿到新的地址重新发起了一次新的请求,那么代码也同样是可以这么实现的,因此调整了代码,写了如下工具类:
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.URL;
import java.net.URLConnection;
import java.util.List;
import java.util.Locale;
import java.util.Map;

import org.apache.commons.httpclient.util.HttpURLConnection;
import org.apache.http.ParseException;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils;

public class HttpClientUtil
{

    /**
     * 向指定URL发送GET方法的请求
     * 
     * @param url
     *            发送请求的URL
     * @param param
     *            请求参数,请求参数应该是name1=value1&name2=value2的形式。
     * @return URL所代表远程资源的响应
     */

    public static String sendGet(String url, String param)
    {
        String result = "";
        BufferedReader in = null;
        try
        {
            String urlName = url + "?" + param;
            URL realUrl = new URL(urlName);
            // 打开和URL之间的连接
            URLConnection conn = realUrl.openConnection();

            String newUrl = urlName;
            Map<String, List<String>> map = conn.getHeaderFields();
            // 遍历所有的响应头字段
            for (String key : map.keySet())
            {
                //如果发现有重定向了新的地址
                if ("Location".equals(key))
                {
                    //获取新地址
                    newUrl = map.get(key).get(0) + "?" + param;
                    break;
                }
            }
            // 重新实例化url对象
            realUrl = new URL(newUrl);
            // 重新打开和URL之间的连接
            conn = realUrl.openConnection();

            // 设置通用的请求属性
            conn.setRequestProperty("accept", "*/*");
            conn.setRequestProperty("connection", "Keep-Alive");
            conn.setRequestProperty("Accept-Charset", "UTF-8");
            conn.setRequestProperty("contentType", "UTF-8");
            conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)");
            conn.setRequestProperty("Content-type", "application/x-www-form-urlencoded");
            conn.setRequestProperty("Accept-Language", Locale.getDefault().toString());
            // 建立实际的连接
            conn.connect();
            // 定义BufferedReader输入流来读取URL的响应
            in = new BufferedReader(new InputStreamReader(conn.getInputStream(), "UTF-8"));
            String line;
            while ((line = in.readLine()) != null)
            {
                result += line;
            }
        }
        catch (Exception e)
        {
            System.out.println("发送GET请求出现异常!" + e);
            e.printStackTrace();
        }
        // 使用finally块来关闭输入流
        finally
        {
            try
            {
                if (in != null)
                {
                    in.close();
                }
            }
            catch (IOException ex)
            {
                ex.printStackTrace();
            }
        }
        return result;
    }


    /**
     * post请求
     */
    public static String sendPost(String url, String param)
    {
        PrintWriter out = null;
        BufferedReader in = null;
        String result = "";
        try
        {
            URL realUrl = new URL(url);
            // 打开和URL之间的连 ?
            URLConnection conn = realUrl.openConnection();
            Map<String, List<String>> map = conn.getHeaderFields();
            // 遍历所有的响应头字段
            String newUrl = url;
            for (String key : map.keySet())
            {
                //如果发现有重定向了新的地址
                if ("Location".equals(key))
                {
                    //获取新地址
                    newUrl = map.get(key).get(0);
                    break;
                }
            }
            realUrl = new URL(newUrl);
            // 打开和URL之间的连接
            conn = realUrl.openConnection();

            // 设置通用的请求属 ?
            conn.setRequestProperty("accept", "*/*");
            conn.setRequestProperty("connection", "Keep-Alive");
            conn.setRequestProperty("Accept-Charset", "UTF-8");
            conn.setRequestProperty("contentType", "UTF-8");
            conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)");
            conn.setRequestProperty("Content-type", "application/x-www-form-urlencoded;charset=UTF-8");
            conn.setRequestProperty("Accept-Language", Locale.getDefault().toString());
            // 发起POST请求必须设置如下两行
            conn.setDoOutput(true);
            conn.setDoInput(true);
            // 获取URLConnection对象对应的输出流
            out = new PrintWriter(new OutputStreamWriter(conn.getOutputStream(), "UTF-8"));
            // 发 ?请求参数
            out.print(param);
            // flush输出流的缓冲
            out.flush();
            // 定义BufferedReader输入流来读取URL的响应
            in = new BufferedReader(new InputStreamReader(conn.getInputStream(), "UTF-8"));
            String line;
            while ((line = in.readLine()) != null)
            {
                result += line;
            }
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        // 使用finally块来关闭输出流 ?输入 ?
        finally
        {
            try
            {
                if (out != null)
                {
                    out.close();
                }
                if (in != null)
                {
                    in.close();
                }
            }
            catch (IOException ex)
            {
                ex.printStackTrace();
            }
        }
        return result;
    }
}

总结

虽说通过调整客户端请求的时候通过调整,实现重定向的功能,但是我认为这并不是最好的解决方式;个人是希望能通过nginx那块的配置实现http到https的重定向,从而不必要修改客户端请求时的代码即可完成这块的动作,如果有那位大牛知道怎么做,烦请留言告知,万分感谢!