Java使用代理进行网络连接

需求是这样的:

一、界面上要有这样几种代理类型可以选。

1.HTTP代理
2.Socks代理
3.不使用代理(直连)
4.使用浏览器设置(浏览器也是HTTP、Socks、直连三种)。

可参考QQ登录设置里的代理能,其实跟qq的代理功能是一样的。

二、测试使用所填写的代理配置信息是否可连接

三、记录用户上次选择的代理配置,默认使用用户上次使用的代理配置进行网络连接。

程序运行环境是WindowsXP、Windows7、Windows8系统。
使用的技术为Java7,Swing,CXF。

难点:

1.如何进行全居的代理设置:

/**
 * 网络代理Bean
 * 
 * @author tang
 */
public class NetworkBean implements Serializable {

    private static final long serialVersionUID = 1L;
    // private static sun.misc.BASE64Encoder base64Encoder = new sun.misc.BASE64Encoder();

    private Proxy.Type type;// 代理类型
    private String address;// ip 地址
    private String port;// 端口号
    private String username;// 代理服务器用户名
    private String password;// 代理服务器用户密码
    private String domain;// 域
    private String typeText;// 代理类型显示的文本

    public NetworkBean() {
    }

    public NetworkBean(Type type, String address, String port, String username, String password) {
        this.type = type;
        this.address = address;
        this.port = port;
        this.username = username;
        this.password = password;
    }

    public NetworkBean(Type type, String address, String port, String username, String password, String domain) {
        super();
        this.type = type;
        this.address = address;
        this.port = port;
        this.username = username;
        this.password = password;
        this.domain = domain;
    }

    public Proxy.Type getType() {
        return type;
    }

    public void setType(Proxy.Type type) {
        this.type = type;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    public String getPort() {
        return port;
    }

    public void setPort(String port) {
        this.port = port;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    public String getDomain() {
        return domain;
    }

    public void setDomain(String domain) {
        this.domain = domain;
    }

    public String getTypeText() {
        return typeText;
    }

    public void setTypeText(String typeText) {
        this.typeText = typeText;
    }

    /**
     * return domain + "\" + username
     */
    public String getDomainAndUsername() {
        return (Utils.toString(domain).trim().isEmpty()) ? username : domain.trim() + "\\" + username;
    }

    /**
     * return domain + "\" + username + ":" + password
     */
    public String getDomainAndUsernameAndPassword() {
        return getDomainAndUsername() + ":" + password;
    }

    /**
     * return username + ":" + password
     */
    public String getUsernameAndPassword() {
        return username + ":" + password;
    }

    /**
     * return (domain + "\" + username + ":" + password) to 64 bit
     */
    public String getDomainAndUsernameAndPassword64() {
        return org.apache.commons.codec.binary.Base64.encodeBase64String(getDomainAndUsernameAndPassword().getBytes());
    }

    @Override
    public String toString() {
        return "NetworkBean [type=" + type + ", typeText=" + typeText + ", address=" + address + ", port=" + port + ", username=" + username + ", password="
                + password + ", domain=" + domain + "]";
    }
}
    /**
     * 根据指定的代理信息设置系统全局的网络代理
     * 
     * @param networkBean
     */
    public static void setNetworkProxyBySystem(NetworkBean networkBean) {
        System.out.println("System Set Proxy : " + networkBean);
        if (isUserProxy(networkBean)) {
            if (networkBean.getType() == Proxy.Type.SOCKS) {
                System.getProperties().remove("http.proxyHost");
                System.getProperties().remove("http.proxyPort");

                System.getProperties().setProperty("socksProxyHost", networkBean.getAddress());
                System.getProperties().setProperty("socksProxyPort", networkBean.getPort());
            } else {
                System.getProperties().setProperty("http.proxyHost", networkBean.getAddress());
                System.getProperties().setProperty("http.proxyPort", networkBean.getPort());
            }

            Authenticator.setDefault(new BairuiAuthenticator(networkBean.getDomainAndUsername(), networkBean.getPassword()));
        } else if (networkBean != null) {
            System.getProperties().remove("proxySet");
            System.getProperties().remove("socksProxySet");

            System.getProperties().remove("http.proxyHost");
            System.getProperties().remove("http.proxyPort");

            System.getProperties().remove("socksProxyHost");
            System.getProperties().remove("socksProxyPort");
        }
    }

    /**
     * 网络用户名密码校验提供者
     */
    public static class BairuiAuthenticator extends Authenticator {
        private String username, password;

        public BairuiAuthenticator(String username, String password) {
            this.username = username;
            this.password = password;
        }

        protected PasswordAuthentication getPasswordAuthentication() {
            return new PasswordAuthentication(username, password == null ? null : password.toCharArray());
        }
    }

2.如何让CXF的Service使用系统的代理:

    /**
     * 为WebService接口添加网络代理支持:httpClientPolicy.setAllowChunking(false);
     * 
     * @param client
     */
    public static void setWebServiceSupportProxy(Client client) {
        HTTPConduit conduit = (HTTPConduit) client.getConduit();
        HTTPClientPolicy httpClientPolicy = new HTTPClientPolicy();
        httpClientPolicy.setAllowChunking(false);
        conduit.setClient(httpClientPolicy);
    }

3.使用浏览器设置:

/**
     * 读取用户注册表获取浏览器的代理设置
     * 
     * @return 该方法不会返回null值也不会抛出异常
     */
    public static NetworkBean getBrowserProxy() {
        NetworkBean networkBean = new NetworkBean();
        networkBean.setTypeText(typeTexts[3]);
        networkBean.setType(Proxy.Type.DIRECT);
        String key = "reg query \"HKEY_CURRENT_USER\\Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\"";// 注册表浏览器代理key
        try {
            Process exec = Runtime.getRuntime().exec(key);
            try (InputStreamReader i = new InputStreamReader(exec.getInputStream()); BufferedReader ir = new BufferedReader(i)) {
                for (String line = ir.readLine(); line != null; line = ir.readLine()) {
                    if (line.indexOf("ProxyServer") >= 0) {
                        String[] split1 = line.split("    ");
                        if (split1.length > 3) {
                            String[] split2 = split1[3].trim().split(":");
                            if (split2.length > 1) {
                                if (!Utils.toString(split2[0]).isEmpty() && !Utils.toString(split2[1]).isEmpty()) {
                                    networkBean.setAddress(split2[0]);
                                    networkBean.setPort(split2[1]);
                                    networkBean.setType(Proxy.Type.HTTP);
                                    break;
                                }
                            }
                        }
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        } catch (Exception e) {// 从注册表读取失败
            e.printStackTrace();
        }
        return networkBean;
    }

4.测试代理是否可用

因为程序中使用了服务器的两个不同的端口,所以需要测试两个端口是否都可连。
如果使用了多台服务器,更加需要分别测试了。


    /**
     * 测试网络代理是否能通过连接,如果不通过抛出异常
     * 
     * @throws Exception
     */
    private static void testNetworkProxy() throws Exception {
        testWebService();
        testURLConnection();
    }

    /**
     * 测试CXF Service接口50367端口
     * 
     * @param networkBean
     * @throws Exception
     */
    public static void testWebService() throws Exception {
        JcptLoginService service = WebServiceUtils.getService(JcptLoginService.class, GeneralWebServiceAddress.LOGIN_SERVICE_URL_ADD);
        String result = service.getLoginPicture();
        System.out.println(result);
    }

    /**
     * 从HttpURLConnection对象读取指定字符,如果不匹配则抛出异常
     * 
     * @param connection
     * @throws Exception
     */
    private static void checkConnectionContent(HttpURLConnection connection) throws Exception {
        try (InputStream inputStream = connection.getInputStream()) {
            byte[] b = new byte[1024];
            boolean success = false;
            StringBuffer sb = new StringBuffer();
            for (int i = inputStream.read(b); i > 0; i = inputStream.read(b)) {
                String tempStr = new String(b, 0, i);
                sb.append(tempStr);
                if (tempStr.indexOf("3,file not found") >= 0) {// service固定返回这个字符串,如果service作了更改此处也应更改
                    success = true;
                    break;
                }
            }
            if (!success) {
                String str = sb.toString();
                if (str.length() > 3) {
                    char char0 = str.charAt(0);
                    char char1 = str.charAt(1);
                    if (Utils.isNumber(char0 + "") && char1 == ',') {
                        success = true;
                    }
                }
            }
            if (!success) {
                throw new RuntimeException("result content does not meet expectations.");
            }
        } catch (Exception ex) {
            throw ex;
        }
    }

    /**
     * 测试文件下载接口9067端口
     * 
     * @param networkBean
     * @throws Exception
     */
    public static void testURLConnection() throws Exception {
        HttpURLConnection connection = (HttpURLConnection) new URL(GeneralWebServiceAddress.FILE_DOWN_URL_ADD + "path=").openConnection();
        checkConnectionContent(connection);
    }
    /**
     * 测试代理服务器连接
     */
    private void testProxyConnection() {
        NetworkBean readNetworkBean = NetworkProxyTool.readNetworkBean();// 先获得正在使用的NetworkBean
        try {
            NetworkBean networkBean = createNetworkBean();// 根据用户填写的信息创建的NetworkBean对象
            showTestResultDialog(NetworkProxyTool.testNetworkProxyBySystem(networkBean));
        } catch (Exception ex) {
            showTestResultDialog(false);
        }
        NetworkProxyTool.setNetworkProxyBySystem(readNetworkBean);// 测试结束,还原原来的NetworkBean
    }

5.因为java连接网络时,如果使用当前的代理连接失败,那么就会使用操作系统中缓存的代理进行网络连接,如何是测试连接时不使用操作系统缓存,但测试结束后使用操作系统缓存。

    /**
     * 设置长连接和验证信息缓存是否开启
     * 
     * @param keepAlive
     */
    public static void setKeepAliveAndAuthCache(boolean keepAlive) {
        System.setProperty("http.keepAlive", keepAlive + "");// 设置是否开始长连接,如果为false可以防止连接被缓存(如果连接被缓存,用户名密码等所有信息都会被缓存)
        if (keepAlive) {
            AuthCacheValue.setAuthCache(new AuthCacheImpl());
        } else {
            // 设置一个空的实现AuthCache可以防止用户名密码信息被缓存
            AuthCacheValue.setAuthCache(new AuthCache() {
                public void remove(String pkey, AuthCacheValue entry) {
                }

                public void put(String pkey, AuthCacheValue value) {
                }

                public AuthCacheValue get(String pkey, String skey) {
                    return null;
                }
            });
        }
    }

在登录之前:

    NetworkProxyTool.setKeepAliveAndAuthCache(false);
    new.Login();

登录成功后:

    NetworkProxyTool.setKeepAliveAndAuthCache(true);

6.保存用户的代理配置:
因为直接序列化自定义类型的对象,会存在版本问题(比如这个类的包名、类名改了,就死定了)。
所以不能直接序列化自定义类的对象,二是将自定义类对象的属性转成字典(Map),然后序列化map。

    /**
     * 读取本地网络代理设置配置文件
     * 
     * @return
     */
    public static NetworkBean readNetworkBean() {
        NetworkBean networkBean = getCurrNetworkBean(readNetworkBeanMap());
        if (networkBean == null) {
            networkBean = new NetworkBean();
            networkBean.setType(Proxy.Type.DIRECT);
        }
        return networkBean;
    }

    /**
     * 获取用户上次选择网络代理设置
     * 
     * @param map
     * @return
     */
    public static NetworkBean getCurrNetworkBean(Map<String, Object> map) {
        putBrowserProxy(map);
        return (NetworkBean) map.get(getTypeMapKey(map));
    }

    /**
     * 将浏览器的信息存放入代理信息总配置map
     * 
     * @param map
     */
    private static void putBrowserProxy(Map<String, Object> map) {
        if (browserProxyBean == null) {
            browserProxyBean = getBrowserProxy();
        }
        NetworkBean networkBeanBrowser = (NetworkBean) map.get(key_browser);
        if (networkBeanBrowser == null) {
            networkBeanBrowser = browserProxyBean;
        }
        if ((Utils.toString(browserProxyBean.getAddress()).isEmpty() || !browserProxyBean.getAddress().equals(networkBeanBrowser.getAddress()))
                || (Utils.toString(browserProxyBean.getPort()).isEmpty() || !browserProxyBean.getPort().equals(networkBeanBrowser.getPort()))) {
            networkBeanBrowser.setUsername(null);
            networkBeanBrowser.setPassword(null);
            networkBeanBrowser.setDomain(null);
        }
        networkBeanBrowser.setType(browserProxyBean.getType());
        networkBeanBrowser.setTypeText(browserProxyBean.getTypeText());
        networkBeanBrowser.setAddress(browserProxyBean.getAddress());
        networkBeanBrowser.setPort(browserProxyBean.getPort());
        map.put(key_browser, networkBeanBrowser);
    }

在登录之前:

    NetworkBean networkBean = NetworkProxyTool.readNetworkBean();
    NetworkProxyTool.setNetworkProxyBySystem(networkBean);

    NetworkProxyTool.setKeepAliveAndAuthCache(false);
    new.Login();

在用户配置完代理点击确定时:

    /**
     * 点击确定
     */
    private void confirm() {

        if ((isHttp() || isSocks()) && !checkIpAndPortNotNull()) {
            return;
        }

        NetworkBean networkBean = createNetworkBean();
        if (isHttp()) {// HTTP代理
            if (networkBean.getDomain() != null) {
                networkBean.setDomain(networkBean.getDomain().trim());
            }
            proxySettingMap.put(key_http, networkBean);
            proxySettingMap.put(key_proxy_type, key_http);
        } else if (isSocks()) {// SOCKS代理
            proxySettingMap.put(key_socks, networkBean);
            proxySettingMap.put(key_proxy_type, key_socks);
        } else if (isBrowser()) {
            proxySettingMap.put(key_browser, networkBean);
            proxySettingMap.put(key_proxy_type, key_browser);
        } else {
            proxySettingMap.put(key_direct, networkBean);
            proxySettingMap.put(key_proxy_type, key_direct);
        }

        userCurrShowNetworkBean = networkBean;

        isConfirm = true;
        setVisible(false);

        NetworkProxyTool.saveNetworkProxyConfig(proxySettingMap);
    }

System Properties

Java System Properties网络设置有哪些key

http://docs.oracle.com/javase/7/docs/technotes/guides/net/properties.html

http://docs.oracle.com/javase/7/docs/api/java/net/doc-files/net-properties.html

Java System Properties优先级说明

host和port比set优先级高,也就是说set可以不用去设置。
如存在http.proxyHost和http.proxyPort时,proxySet设为false无效,系统仍然可以使用http代理。
存在socksProxyHost和socksProxyPort时,socksProxySet设为false无效,系统仍然可以使用socks代理。

http比socks优先级高,即存在http.proxyHost和http.proxyPort时,socksProxySet、socksProxyHost、socksProxyPort都会失效,系统会使用http代理。

使用java.net.Proxy

如果只是单个的连接需要使用代理,那么可以采用Proxy类进行代理连接。
说明:
1.Socket只能使用socks代理不能使用http代理。
2.Socket使用使用Authenticator来进行验证。
3.URLConnection 使用RequestProperty是只对当前连接有效,不会缓存,Authenticator是全局性的,对所有网络请求都有效,会缓存,但RequestProperty的优先级比Authenticator高。
4.只有RequestProperty需要64位编码,Authenticator不需要。
5.需要清除验证缓存的,不使用验证缓存上面已经有代码了,这里就不重复写了。
6.其实,建议采用全局性的代理连接,不管是URLConnection 还是Socket 都简单方便、统一。

代码:

NetworkBean httpBean = new NetworkBean(Proxy.Type.HTTP, "192.168.77.5", "8888", "tzc", "123", null);
Proxy httpProxy = new Proxy(httpBean.getType(), new InetSocketAddress(httpBean.getAddress(), Integer.parseInt(httpBean.getPort())));

NetworkBean socksBean = new NetworkBean(Proxy.Type.SOCKS, "192.168.77.5", "9999", "tzc", "123", "ttt");
Proxy socksProxy = new Proxy(socksBean.getType(), new InetSocketAddress(socksBean.getAddress(), Integer.parseInt(socksBean.getPort())));

URLConnection httpProxyConnection = new URL("http://www.baidu.com/").openConnection(httpProxy);
//或者Authenticator.setDefault(new BairuiAuthenticator(networkBean.getDomainAndUsername(), networkBean.getPassword()));
httpProxyConnection.setRequestProperty("Proxy-Authorization", "Basic "+httpBean.getDomainAndUsernameAndPassword64());

URLConnection socksProxyConnection = new URL("http://www.baidu.com/").openConnection(socksProxy);
//或者Authenticator.setDefault(new BairuiAuthenticator(networkBean.getDomainAndUsername(), networkBean.getPassword()));
socksProxyConnection.setRequestProperty("Proxy-Authorization", "Basic "+httpBean.getDomainAndUsernameAndPassword64());

Socket socket = new Socket(socksProxy);
Authenticator.setDefault(new BairuiAuthenticator(networkBean.getDomainAndUsername(), networkBean.getPassword()));
socket.connect(new InetSocketAddress("www.baidu.com", 5555));
  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要实现Java网络代理,可以使用Java提供的Socket类和ServerSocket类。以下是一个简单的示例代码: ```java import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.ServerSocket; import java.net.Socket; public class ProxyServer { private static final int PROXY_PORT = 8080; private static final String TARGET_HOST = "目标主机的IP地址"; private static final int TARGET_PORT = 80; public static void main(String[] args) { try { ServerSocket serverSocket = new ServerSocket(PROXY_PORT); while (true) { Socket clientSocket = serverSocket.accept(); Thread proxyThread = new Thread(() -> { try { // 连接目标主机 Socket targetSocket = new Socket(TARGET_HOST, TARGET_PORT); // 处理客户端请求 Thread clientThread = new Thread(() -> handleRequest(clientSocket.getInputStream(), targetSocket.getOutputStream())); clientThread.start(); // 处理目标主机响应 handleResponse(targetSocket.getInputStream(), clientSocket.getOutputStream()); // 等待线程完成 clientThread.join(); // 关闭连接 targetSocket.close(); clientSocket.close(); } catch (IOException | InterruptedException e) { e.printStackTrace(); } }); proxyThread.start(); } } catch (IOException e) { e.printStackTrace(); } } private static void handleRequest(InputStream clientInput, OutputStream targetOutput) { try { byte[] buffer = new byte[4096]; int bytesRead; while ((bytesRead = clientInput.read(buffer)) != -1) { targetOutput.write(buffer, 0, bytesRead); targetOutput.flush(); } } catch (IOException e) { e.printStackTrace(); } } private static void handleResponse(InputStream targetInput, OutputStream clientOutput) { try { byte[] buffer = new byte[4096]; int bytesRead; while ((bytesRead = targetInput.read(buffer)) != -1) { clientOutput.write(buffer, 0, bytesRead); clientOutput.flush(); } } catch (IOException e) { e.printStackTrace(); } } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值