Anotasi tenang SpringMVC @RequestBody menukar kemahiran json dan object_javascript

WBOY
Lepaskan: 2016-05-16 15:26:22
asal
2527 orang telah melayarinya

Disebabkan Tahun Baharu Cina yang akan datang, pasukan projek tidak mempunyai banyak tugas, jadi saya mempunyai masa lapang untuk belajar panggilan tenang pada musim bunga. Saya mendapati bahawa musim bunga telah menjadi begitu hebat sehingga pengaturcara tidak perlu lagi mengambil berat tentang penukaran data dan permohonan dalam proses menulis antara muka, dan hanya perlu memberi tumpuan kepada perniagaan. Di bawah ini saya rumuskan langkah-langkah dan masalah yang dihadapi semasa proses penyelidikan.

Langkah:

1. git clone https://github.com/spring-guides/gs-rest-service.git Memuat turun kod sumber daripada laman web rasmi spring

2. Susun dengan maven (gradle juga boleh diterima)

3. Lari dan lawati http://localhost:8080/greeting

4. Hasil larian boleh menukar objek menjadi objek json dan mengembalikannya ke halaman

Pada masa ini, saya berfikir tentang cara menukar data yang diminta secara automatik kepada objek java Melalui google, saya mendapati bahawa spring sebenarnya telah menyediakan penukar HttpMessageConverter, dan MappingJackson2HttpMessageConverter (json ~kelas penukaran objek) dimuatkan secara lalai. . Hanya konfigurasikan @RequestBody Greeting gree untuk digunakan.

Kod lapisan pengawal adalah seperti berikut:

@RequestMapping(value = "/greeting", method = RequestMethod.POST,consumes = "application/json")
  public @ResponseBody Greeting greeting(@RequestBody Greeting gree) { 
    System.out.println(gree.getContent());
    return gree;
  }
Salin selepas log masuk

Pada masa ini, saya membuat panggilan melalui pemalam Google (posmen), tetapi panggilan hidup atau mati tidak berjaya!

Menganalisis dan menyelesaikan masalah:

Pada masa ini saya merasakan punca masalah mungkin dalam aspek berikut:

1. Spring tidak memuatkan MappingJackson2HttpMessageConverter secara lalai (saya tidak tahu kaedah pemuatan khusus)

2. MappingJackson2HttpMessageConverter tidak boleh berfungsi selepas memuatkan (saya tidak tahu sebab mengapa ia tidak berfungsi)

Malah, sebab mengapa ia tidak berfungsi pada akhirnya adalah kerana saya terlalu percaya pada kod sumber spring (objek itu tidak menyediakan kaedah yang ditetapkan dengan dua soalan ini, pencari besar-besaran di Internet). tidak dapat mencari hasil yang sepadan. Tidak ada cara selain mencari punca masalah dan melihat kod sumber musim bunga.

Untuk soalan pertama:

Langkah 1: Tulis semula penukar jenis pemuatan secara manual

@Configuration
  @EnableWebMvc
public class WebConfiguration extends WebMvcConfigurerAdapter {
  public void configureMessageConverters(List<HttpMessageConverter<&#63;>> messageConverters) {
    System.out.println("init convert is start !!!!!");
    StringHttpMessageConverter stringConverter = new StringHttpMessageConverter();
    stringConverter.setWriteAcceptCharset(false);
    messageConverters.add(new MappingJackson2HttpMessageConverter());
    System.out.println("init convert is stop !!!!!");
  }
}
Salin selepas log masuk

Ujian mendapati ia masih tidak boleh digunakan, dan kini sebabnya semakin kurang jelas. Anda hanya boleh melihat cara spring memuatkan penukar jenis secara lalai. Ternyata dalam kaedah addDefaultHttpMessageConverters dalam WebMvcConfigurationSupport (kata kunci HttpMessageConverter dicerminkan dan dicari di mana ia digunakan, ia ditemui melalui pertimbangan dan penjejakan) seperti berikut:

@SuppressWarnings("deprecation")
  protected final void addDefaultHttpMessageConverters(List<HttpMessageConverter<&#63;>> messageConverters) {
    StringHttpMessageConverter stringConverter = new StringHttpMessageConverter();
    stringConverter.setWriteAcceptCharset(false);
    messageConverters.add(new ByteArrayHttpMessageConverter());
    messageConverters.add(stringConverter);
    messageConverters.add(new ResourceHttpMessageConverter());
    messageConverters.add(new SourceHttpMessageConverter<Source>());
    messageConverters.add(new AllEncompassingFormHttpMessageConverter());
    if (romePresent) {
      messageConverters.add(new AtomFeedHttpMessageConverter());
      messageConverters.add(new RssChannelHttpMessageConverter());
    }
    if (jaxb2Present) {
      messageConverters.add(new Jaxb2RootElementHttpMessageConverter());
    }
    if (jackson2Present) {
      messageConverters.add(new MappingJackson2HttpMessageConverter());
    }
    else if (jacksonPresent) {
      messageConverters.add(new org.springframework.http.converter.json.MappingJacksonHttpMessageConverter());
    }
  }
Salin selepas log masuk

Penukar lalai yang sepadan telah dimuatkan. Penyahpepijatan titik putus menunjukkan bahawa tiada masalah dengan konfigurasi lalai.

Ia hanya boleh dikatakan berpunca daripada masalah kedua, tetapi saya tidak tahu mengapa ia menyebabkan masalah ini (masalah data json, atau masalah lain, tanpa mengetahui masalah, saya hanya boleh melihat permintaan). minta dan lihat cara penukar berfungsi. Kerana saya tidak tahu banyak tentang musim bunga, saya tidak tahu prinsipnya. Dalam kes ini, penggunaan yang sepadan hanya boleh didapati berdasarkan kelas kunci (HttpMessageConverter). Gunakan pengalaman untuk menilai dan nyahpepijat. Didapati bahawa kaedah readWithMessageConverters dalam AbstractMessageConverterMethodArgumentResolver ialah kaedah pemprosesan untuk permintaan permintaan untuk melaksanakan penukaran jenis.

protected <T> Object readWithMessageConverters(HttpInputMessage inputMessage,
      MethodParameter methodParam, Type targetType) throws IOException, HttpMediaTypeNotSupportedException {
    MediaType contentType;
    try {
      contentType = inputMessage.getHeaders().getContentType();
    }
    catch (InvalidMediaTypeException ex) {
      throw new HttpMediaTypeNotSupportedException(ex.getMessage());
    }
    if (contentType == null) {
      contentType = MediaType.APPLICATION_OCTET_STREAM;
    }
    Class<&#63;> contextClass = methodParam.getContainingClass();
    Class<T> targetClass = (Class<T>) ResolvableType.forType(targetType,
        ResolvableType.forMethodParameter(methodParam)).resolve();
    for (HttpMessageConverter<&#63;> converter : this.messageConverters) {
      if (converter instanceof GenericHttpMessageConverter) {
        GenericHttpMessageConverter<&#63;> genericConverter = (GenericHttpMessageConverter<&#63;>) converter;
        if (genericConverter.canRead(targetType, contextClass, contentType)) {
          if (logger.isDebugEnabled()) {
            logger.debug("Reading [" + targetType + "] as \"" +
                contentType + "\" using [" + converter + "]");
          }
          return genericConverter.read(targetType, contextClass, inputMessage);
        }
      }
      if (targetClass != null) {
        if (converter.canRead(targetClass, contentType)) {
          if (logger.isDebugEnabled()) {
            logger.debug("Reading [" + targetClass.getName() + "] as \"" +
                contentType + "\" using [" + converter + "]");
          }
          return ((HttpMessageConverter<T>) converter).read(targetClass, inputMessage);
        }
      }
    }
    throw new HttpMediaTypeNotSupportedException(contentType, allSupportedMediaTypes);
  }
Salin selepas log masuk

Pada masa ini, didapati bahawa penukar mesej jenis sepadan MappingJackson2HttpMessageConverter telah ditemui mengikut kaedah canRead HttpMessageConverter, dan penukaran telah bermula, tetapi pengecualian masa jalan telah dibuang. Kerana pengecualian bukan output pada konsol. Melalui penyahpepijatan titik putus, saya mendapati bahawa kaedah readJavaType MappingJackson2HttpMessageConverter melemparkan pengecualian masa jalan Melalui kod sumber, saya mendapati bahawa lapisan bawah dikendalikan oleh objectMapper Jackson adalah seperti berikut:

try {
      return this.objectMapper.readValue(inputMessage.getBody(), javaType);
    }
    catch (IOException ex) {
      throw new HttpMessageNotReadableException("Could not read JSON: " + ex.getMessage(), ex);
    }
Salin selepas log masuk

Jika saya mengeluarkan kod secara berasingan dan menjalankannya dalam kaedah utama, tetapi ia masih tidak berfungsi, maka saya boleh mencari masalahnya. Sama ada jenis salah atau data input salah. Selepas pemeriksaan yang teliti, saya mendapati bahawa tidak ada masalah dengan data json, dan ia juga boleh ditukar menggunakan jsonobject. Pada masa ini, ia hanya boleh dinilai bahawa terdapat masalah dengan javaType yang masuk. Jika saya membukanya dan mendapati objek itu (Salam) tidak mempunyai kaedah yang ditetapkan, saya tertanya-tanya adakah kerana jakson tidak boleh berfungsi (prinsipnya tidak jelas). Jika ya, saya menyediakan kaedah yang ditetapkan untuk objek ini dan ia boleh dijalankan semula. Selepas berpusing-pusing dalam bulatan, saya akhirnya menyelesaikan masalah itu, tetapi melalui masalah ini, saya menjadi lebih sedar tentang mekanisme kerja spring's restful.

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan
Tentang kita Penafian Sitemap
Laman web PHP Cina:Latihan PHP dalam talian kebajikan awam,Bantu pelajar PHP berkembang dengan cepat!