Сериализованные данные формы в Java [дубликат]

private void f(Button b, final int a[]) {

    b.addClickHandler(new ClickHandler() {

        @Override
        public void onClick(ClickEvent event) {
            a[0] = a[0] * 5;

        }
    });
}
194
задан Nathan 22 January 2016 в 22:13
поделиться

13 ответов

Если вы ищете способ его достижения без использования внешней библиотеки, вам поможет следующий код.

public static Map<String, String> splitQuery(URL url) throws UnsupportedEncodingException {
    Map<String, String> query_pairs = new LinkedHashMap<String, String>();
    String query = url.getQuery();
    String[] pairs = query.split("&");
    for (String pair : pairs) {
        int idx = pair.indexOf("=");
        query_pairs.put(URLDecoder.decode(pair.substring(0, idx), "UTF-8"), URLDecoder.decode(pair.substring(idx + 1), "UTF-8"));
    }
    return query_pairs;
}

Вы можете получить доступ к возвращенной карте с помощью <map>.get("client_id"), с URL-адресом заданный в вашем вопросе, это вернет «SS».

UPDATE URL-декодирование добавлено

UPDATE. Поскольку этот ответ все еще довольно популярен, я сделал улучшенную версию вышеописанного метода, который

public static Map<String, List<String>> splitQuery(URL url) throws UnsupportedEncodingException {
  final Map<String, List<String>> query_pairs = new LinkedHashMap<String, List<String>>();
  final String[] pairs = url.getQuery().split("&");
  for (String pair : pairs) {
    final int idx = pair.indexOf("=");
    final String key = idx > 0 ? URLDecoder.decode(pair.substring(0, idx), "UTF-8") : pair;
    if (!query_pairs.containsKey(key)) {
      query_pairs.put(key, new LinkedList<String>());
    }
    final String value = idx > 0 && pair.length() > idx + 1 ? URLDecoder.decode(pair.substring(idx + 1), "UTF-8") : null;
    query_pairs.get(key).add(value);
  }
  return query_pairs;
}

UPDATE Версия Java8

public Map<String, List<String>> splitQuery(URL url) {
    if (Strings.isNullOrEmpty(url.getQuery())) {
        return Collections.emptyMap();
    }
    return Arrays.stream(url.getQuery().split("&"))
            .map(this::splitQueryParameter)
            .collect(Collectors.groupingBy(SimpleImmutableEntry::getKey, LinkedHashMap::new, mapping(Map.Entry::getValue, toList())));
}

public SimpleImmutableEntry<String, String> splitQueryParameter(String it) {
    final int idx = it.indexOf("=");
    final String key = idx > 0 ? it.substring(0, idx) : it;
    final String value = idx > 0 && it.length() > idx + 1 ? it.substring(idx + 1) : null;
    return new SimpleImmutableEntry<>(key, value);
}

Выполнение вышеуказанного метода с помощью URL

https://stackoverflow.com?param1=value1&param2=&param3=value3&param3

возвращает эту карту:

{param1=["value1"], param2=[null], param3=["value3", null]}
262
ответ дан Stephan 21 August 2018 в 06:43
поделиться
  • 1
    Вы забываете декодировать имена и параметры, одна из причин, почему обычно лучше позволить библиотекам выполнять общие задачи. – Juan Mendes 27 November 2012 в 22:52
  • 2
    действительно, вы правы ... но я лично предпочитаю писать такие «простые». задачи самостоятельно, вместо того, чтобы использовать собственную библиотеку для каждой отдельной задачи, которую я должен выполнить. – Pr0gr4mm3r 27 November 2012 в 22:59
  • 3
    если у вас есть несколько параметров с одинаковым именем / ключом, использование этой функции переопределит значение, имеющее аналогичный ключ. – snowball147 8 July 2013 в 07:24
  • 4
    @Chris Вы запутываете xml / html экранирование с кодировкой URL. Ваш пример должен быть: a.com/q?1=a%26b&2=b%26c – sceaj 19 May 2015 в 23:40
  • 5
    Если вы используете статический импорт, пожалуйста, покажите им – Tarmo 14 July 2017 в 12:36

Простое обновление к API версии Java 8

public Map<String, List<String>> splitQuery(URL url) {
    if (Strings.isNullOrEmpty(url.getQuery())) {
        return Collections.emptyMap();
    }
    return Arrays.stream(url.getQuery().split("&"))
            .map(this::splitQueryParameter)
            .collect(Collectors.groupingBy(SimpleImmutableEntry::getKey, LinkedHashMap::new, **Collectors**.mapping(Map.Entry::getValue, **Collectors**.toList())));
}

и toList () необходимо использовать с коллекторами, которые не были указаны в верхнем ответе. В противном случае это приведет к ошибке компиляции в IDE

0
ответ дан Aarish Ramesh 21 August 2018 в 06:43
поделиться

Самый короткий путь, который я нашел, следующий:

MultiValueMap<String, String> queryParams =
            UriComponentsBuilder.fromUriString(url).build().getQueryParams();

UPDATE: UriComponentsBuilder происходит из Spring. Здесь ссылка .

9
ответ дан Dmitry Senkovich 21 August 2018 в 06:43
поделиться
  • 1
    Не зная, откуда взялся этот класс UriComponentsBuilder, это не очень полезно. – Thomas Mortagne 6 November 2017 в 11:50
  • 2
    Возможно, стоит отметить, что это только хорошая идея, если вы уже используете Spring. Если вы не используете Spring, вам следует избегать. samatkinson.com/why-i-hate-spring – Nick 8 April 2018 в 11:05
  • 3
    Это именно то, что я искал, спасибо. – TachikomaGT 23 April 2018 в 11:54

Java 8 один оператор

С учетом URL-адреса для анализа:

URL url = new URL("https://google.com.ua/oauth/authorize?client_id=SS&response_type=code&scope=N_FULL&access_type=offline&redirect_uri=http://localhost/Callback");

Это решение собирает список пар:

List<AbstractMap.SimpleEntry<String, String>> list = 
        Pattern.compile("&").splitAsStream(url.getQuery())
        .map(s -> Arrays.copyOf(s.split("="), 2))
        .map(o -> new AbstractMap.SimpleEntry<String, String>(decode(o[0]), decode(o[1])))
        .collect(toList());

Это решение с другой стороны, собирает карту (учитывая, что в URL-адресе может быть больше параметров с одним и тем же именем, но с разными значениями).

Map<String, List<String>> list = 
        Pattern.compile("&").splitAsStream(url.getQuery())
        .map(s -> Arrays.copyOf(s.split("="), 2))
        .collect(groupingBy(s -> decode(s[0]), mapping(s -> decode(s[1]), toList())));

Оба решения должны использовать служебную функцию для правильного декодирования параметров.

private static String decode(final String encoded) {
    try {
        return encoded == null ? null : URLDecoder.decode(encoded, "UTF-8");
    } catch(final UnsupportedEncodingException e) {
        throw new RuntimeException("Impossible: UTF-8 is a required encoding", e);
    }
}
5
ответ дан freedev 21 August 2018 в 06:43
поделиться
  • 1
    Это скорее подход Java 8 , чем Java 8 oneliner. – Stephan 16 June 2017 в 08:36
  • 2
    @Stephan хорошо :) Может быть, оба. Но мне больше интересно понять, нравится ли вам это решение. – freedev 16 June 2017 в 08:39
  • 3
    ИМО, онлайнер должен быть коротким и не должен охватывать несколько строк. – Stephan 16 June 2017 в 08:54
  • 4
    IMO a oneliner - это когда это одно утверждение. Я написал таким образом для удобства чтения. – freedev 16 June 2017 в 09:29
  • 5
    Здесь есть несколько заявлений. – Stephan 16 June 2017 в 09:47
  • 6
    – Abhijit Sarkar 9 August 2018 в 18:26

Если вы используете Java 8, и вы готовы написать несколько методов многократного использования, вы можете сделать это в одной строке.

private Map<String, List<String>> parse(final String query) {
    return Arrays.asList(query.split("&")).stream().map(p -> p.split("=")).collect(Collectors.toMap(s -> decode(index(s, 0)), s -> Arrays.asList(decode(index(s, 1))), this::mergeLists));
}

private <T> List<T> mergeLists(final List<T> l1, final List<T> l2) {
    List<T> list = new ArrayList<>();
    list.addAll(l1);
    list.addAll(l2);
    return list;
}

private static <T> T index(final T[] array, final int index) {
    return index >= array.length ? null : array[index];
}

private static String decode(final String encoded) {
    try {
        return encoded == null ? null : URLDecoder.decode(encoded, "UTF-8");
    } catch(final UnsupportedEncodingException e) {
        throw new RuntimeException("Impossible: UTF-8 is a required encoding", e);
    }
}

Но это довольно жестокая строка.

8
ответ дан Fuwjax 21 August 2018 в 06:43
поделиться

У меня был вариант Kotlin, который показал, что это лучший результат в Google.

@Throws(UnsupportedEncodingException::class)
fun splitQuery(url: URL): Map<String, List<String>> {

    val queryPairs = LinkedHashMap<String, ArrayList<String>>()

    url.query.split("&".toRegex())
            .dropLastWhile { it.isEmpty() }
            .map { it.split('=') }
            .map { it.getOrEmpty(0).decodeToUTF8() to it.getOrEmpty(1).decodeToUTF8() }
            .forEach { (key, value) ->

                if (!queryPairs.containsKey(key)) {
                    queryPairs[key] = arrayListOf(value)
                } else {

                    if(!queryPairs[key]!!.contains(value)) {
                        queryPairs[key]!!.add(value)
                    }
                }
            }

    return queryPairs
}

И методы расширения

fun List<String>.getOrEmpty(index: Int) : String {
    return getOrElse(index) {""}
}

fun String.decodeToUTF8(): String { 
    URLDecoder.decode(this, "UTF-8")
}
0
ответ дан Graham Smith 21 August 2018 в 06:43
поделиться

используйте google Guava и делайте это в двух строках:

import java.util.Map;
import com.google.common.base.Splitter;

public class Parser {
    public static void main(String... args) {
        String uri = "https://google.com.ua/oauth/authorize?client_id=SS&response_type=code&scope=N_FULL&access_type=offline&redirect_uri=http://localhost/Callback";
        String query = uri.split("\\?")[1];
        final Map<String, String> map = Splitter.on('&').trimResults().withKeyValueSeparator("=").split(query);
        System.out.println(map);
    }
}

, который дает вам

{client_id=SS, response_type=code, scope=N_FULL, access_type=offline, redirect_uri=http://localhost/Callback}
39
ответ дан Kirby 21 August 2018 в 06:43
поделиться
  • 1
    Как насчет декодирования URL, описанного в выбранном ответе? – Clint Eastwood 5 June 2014 в 21:38
  • 2
    Это также подозрительно для нескольких ключей с тем же именем. Согласно javadocs, это вызовет исключение IllegalArgumentException – jontro 30 December 2014 в 15:42
  • 3
    Вместо ручного разделения uri вы должны использовать new java.net.URL(uri).getQuery(), так как это покупает вам бесплатную проверку ввода в URL-адресе. – avgvstvs 6 March 2015 в 14:07
  • 4
    Для декодирования: final Map & lt; String, String & gt; queryVars = Maps.transformValues ​​(map, new Function & lt; String, String & gt; () {@Override public String apply (String value) {try {return URLDecoder.decode (значение, "UTF-8");} catch (UnsupportedEncodingException e ) {// TODO Автогенерируемый блок catch e.printStackTrace ();} возвращаемое значение;}}); – phreakhead 22 July 2015 в 18:22
  • 5
    ПРЕДУПРЕЖДЕНИЕ!! Это НЕ безопасно делать, поскольку splitter.split() будет бросать IllegalArgumentException, если в строке запроса есть дубликат. См. stackoverflow.com/questions/1746507/… – Anderson 27 November 2015 в 04:57

Если вы используете Spring Framework:

public static void main(String[] args) {
    String uri = "http://youhost.com/test?param1=abc&param2=def&param2=ghi";
    MultiValueMap<String, String> parameters =
            UriComponentsBuilder.fromUriString(uri).build().getQueryParams();
    List<String> param1 = parameters.get("param1");
    List<String> param2 = parameters.get("param2");
    System.out.println("param1: " + param1.get(0));
    System.out.println("param2: " + param2.get(0) + "," + param2.get(1));
}

Вы получите:

param1: abc
param2: def,ghi
54
ответ дан kkonrad 21 August 2018 в 06:43
поделиться

Если вы используете Spring, добавьте аргумент типа @RequestParam Map<String,String> к вашему методу контроллера, а Spring построит для вас карту!

0
ответ дан mancini0 21 August 2018 в 06:43
поделиться

Если вы используете сервлет doGet, попробуйте это

request.getParameterMap()

Возвращает java.util.Map параметров этого запроса.

Возвращает: неизменяемая java.util.Map, содержащая имена параметров в качестве ключей и значений параметров в качестве значений карты. Ключи в карте параметров имеют тип String. Значения в карте параметров имеют тип String array.

( Java doc )

233
ответ дан Roshana Pitigala 21 August 2018 в 06:43
поделиться
  • 1
    Могу ли я получить значение по его имени, не передавая все элементы? Я имею в виду что-то вроде этого: System.out.print (params [& quot; one & quot;]); – Sergey Shafiev 27 November 2012 в 22:22
  • 2
    @SergeyShafiev. Тривиально преобразовать List<NameValuePair> в Map<String,String>. Java не имеет доступ к скобкам для хэш-карт, это будет выглядеть как map.get("one"). Если вы не знаете, как это сделать, это должен быть другой вопрос (но сначала попробуйте сами). Мы предпочитаем оставлять вопросы в Slim здесь на SO – Juan Mendes 27 November 2012 в 22:24
  • 3
    Будьте осторожны, если у вас в вашем URL два параметра одинаковый (то есть? A = 1 & amp; a = 2) URLEncodedUtils выдаст исключение IllegalArgumentException – Crystark 28 April 2014 в 13:45
  • 4
    @Crystark Начиная с httpclient 4.3.3 строка запроса с дублируемыми именами не генерирует никаких исключений. Он работает так, как ожидалось. System.out.println(URLEncodedUtils.parse(new URI("http://example.com/?foo=bar&foo=baz"), "UTF-8")); напечатает [foo = bar, foo = baz] . – Akihiro HARAI 2 June 2014 в 07:24
  • 5
    Начиная с Android 6, клиентская библиотека Apache HTTP была удалена. Это означает, что URLEncodedUtils and NameValuePair` больше недоступны (если вы не добавили зависимость от старой библиотеки Apache, как описано здесь здесь ). – Ted Hopp 10 December 2015 в 18:20

Используя вышеупомянутые комментарии и решения, я сохраняю все параметры запроса, используя Map & lt; String, Object & gt; где Objects может быть строкой или Set & lt; String & gt ;. Решение приведено ниже. Рекомендуется использовать какой-либо валидатор URL для проверки подлинности URL-адреса, а затем вызвать метод convertQueryStringToMap.

private static final String DEFAULT_ENCODING_SCHEME = "UTF-8";

public static Map<String, Object> convertQueryStringToMap(String url) throws UnsupportedEncodingException, URISyntaxException {
    List<NameValuePair> params = URLEncodedUtils.parse(new URI(url), DEFAULT_ENCODING_SCHEME);
    Map<String, Object> queryStringMap = new HashMap<>();
    for(NameValuePair param : params){
        queryStringMap.put(param.getName(), handleMultiValuedQueryParam(queryStringMap, param.getName(), param.getValue()));
    }
    return queryStringMap;
}

private static Object handleMultiValuedQueryParam(Map responseMap, String key, String value) {
    if (!responseMap.containsKey(key)) {
        return value.contains(",") ? new HashSet<String>(Arrays.asList(value.split(","))) : value;
    } else {
        Set<String> queryValueSet = responseMap.get(key) instanceof Set ? (Set<String>) responseMap.get(key) : new HashSet<String>();
        if (value.contains(",")) {
            queryValueSet.addAll(Arrays.asList(value.split(",")));
        } else {
            queryValueSet.add(value);
        }
        return queryValueSet;
    }
}
2
ответ дан SUMIT 21 August 2018 в 06:43
поделиться

Для Android, если вы используете OkHttp в своем проекте. Вы можете взглянуть на это. Это просто и полезно.

final HttpUrl url = HttpUrl.parse(query);
if (url != null) {
    final String target = url.queryParameter("target");
    final String id = url.queryParameter("id");
}
3
ответ дан THANN Phearum 21 August 2018 в 06:43
поделиться
  • 1
    HttpUrl - это странное имя, но это именно то, что мне нужно. Благодарю. – GuiSim 21 June 2018 в 21:29
243
ответ дан Roshana Pitigala 1 November 2018 в 01:52
поделиться
Другие вопросы по тегам:

Похожие вопросы: