4

我试图了解 ProxySelector 类的工作原理。我当前的代码如下所示:

    URI uri = new URI("http://google.com");
    proxySelector.select(uri);

我知道在调用它时proxySelector.select(uri);假设返回相应 URI 的代理列表。但我看不到如何为每个 URI 设置代理。

我知道我可以使用该setDefault()方法设置默认代理,但据我所知,这将设置系统范围的代理,而不是特定URI.

我可能在这里遗漏了一些基本点,但是如何为 url A 设置一个代理(例如http://google.com)和为 url B 设置一个不同的代理(例如http://ebay.com)然后有每次连接到对应的url时系统会自动选择正确的代理吗?

4

2 回答 2

4
  1. ProxySelector.select(URI uri)实现自定义逻辑以为 URI 选择正确的代理或代理列表的覆盖方法。

  2. ProxySelector通过调用将新的自定义设置为系统范围ProxySelector.setDefault(customProxySelector)

    的任何子类都URLConnection将使用ProxySelector,例如:

    URLConnection conn = url.openConnection();
    
  3. 或配置您将用于调用远程 URI 的框架,例如 Spring RestTemplate:

    HttpRoutePlanner routePlanner = new SystemDefaultRoutePlanner(new MyProxySelector());
    
    HttpComponentsClientHttpRequestFactory clientHttpRequestFactory 
        = new HttpComponentsClientHttpRequestFactory(
            HttpClientBuilder.create()
                .setRoutePlanner(routePlanner)
                .build());
    restTemplate = new RestTemplate(clientHttpRequestFactory);
    

select(URI uri)如果自定义逻辑无法为 uri 确定合适的代理,则回退到自定义中的默认 ProxySelector 是一种很好的做法。

请参阅我对ProxySelector 示例的其他答案。

网络和代理在Java 网络和代理第 4 段 ProxySelector)和ProxySelector Java 文档中有很好的解释。

于 2017-11-07T19:52:16.793 回答
1

实际上,只需使用增强的 ProxySelector,它就可以很好地工作而无需任何反射,它支持特定 url 的白名单并将其余部分传递给委托(也称为默认值)ProxySelector。

public class DelegatingProxySelector extends ProxySelector {

    private final Set<URI> allProxiedUri = new HashSet<>();

    private String proxyHost;
    private Integer proxyPort;

    private final ProxySelector delegate;

    public DelegatingProxySelector(
        String proxyHost, Integer proxyPort, ProxySelector delegate) {
        this.proxyHost = proxyHost;
        this.proxyPort = proxyPort;
        this.delegate = delegate;
    }

    @Override
    public List<Proxy> select(final URI uri) {
        if (allProxiedUri.contains(uri)) {
            final InetSocketAddress proxyAddress = InetSocketAddress
                .createUnresolved(proxyHost, proxyPort);
            return Collections.singletonList(new Proxy(Type.HTTP, proxyAddress));
        }
        return delegate.select(uri);
    }

    @Override
    public void connectFailed(final URI uri, final SocketAddress sa, final IOException ioe) {
        System.err.println("Unable to reach uri " + uri + " via proxy: " + ioe.getMessage());
    }
    
    public void addProxiedUri(URI uri) {
        allProxiedUri.add(uri);
    }
}

...

final DelegatingProxySelector delegatingProxySelector = new DelegatingProxySelector("localhost", 3128, ProxySelector.getDefault());
ProxySelector.setDefault(delegatingProxySelector);
于 2020-11-13T09:27:36.227 回答