java - 作为 byte[] 发送的字符串内容将其内容转义为查询字符串

标签 java string spring http urlconnection

我需要通过网络发送一个字符串,并且我在 Spring Boot 的 REST Controller 的另一端获取它。

我使用以下几行发送数据:

byte[] content = sw.toString().getBytes(StandardCharsets.UTF_8); //sw is StringWriter
httpUrlConnection = (HttpURLConnection)new URL("http://127.0.0.1:8080/upload").openConnection();
httpUrlConnection.setDoInput(false);
httpUrlConnection.setDoOutput(true);
httpUrlConnection.setRequestMethod("POST");
httpUrlConnection.setFixedLengthStreamingMode(content.length);
httpUrlConnection.connect();
httpUrlConnection.getOutputStream().write(content);
httpUrlConnection.getOutputStream().flush();

我收到的是这样的:

@RequestMapping(value="/upload", method = RequestMethod.POST)
public ResponseEntity<String> upload(@RequestBody byte[] uploadedData)
    throws NoSuchAlgorithmException, InvalidKeyException, IOException {
  if(uploadedData == null) {
    log.info("Uploaded data was null.");
    return new ResponseEntity<String>("Data was null after upload.", HttpStatus.INTERNAL_SERVER_ERROR);
  }

  String receivedData = new String(uploadedData, StandardCharsets.UTF_8);
  log.info("" + receivedData);

但是虽然我发送的是这个...

-----BEGIN CERTIFICATE REQUEST-----
MIIB0DCCATkCAQAwgY8xCzAJBgNVBAYTAkFVMSgwJgYDVQQKDB9UaGUgTGVnaW9u
IG9mIHRoZSBCb3VuY3kgQ2FzdGxlMRIwEAYDVQQHDAlNZWxib3VybmUxETAPBgNV
BAgMCFZpY3RvcmlhMS8wLQYJKoZIhvcNAQkBFiBmZWVkYmFjay1jcnlwdG9AYm91
bmN5Y2FzdGxlLm9yZzCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAqE+pEyKS
21JXgeldS83E+SSmQ4s2xYPqV7Yw2ebUKsxnbBz1KcXHdHS1MJ9PYzBJgogdigZW
6hEQ83edP/Ay/EQzGqeKUzFqNEsQh3PSdbF9N5k7b81tQHUbfIbNu1ofSBNa/Eit
MkOj1NAmwivpW0AA8aPZhGzYLYWcp0lsC78CAwEAAaAAMA0GCSqGSIb3DQEBBQUA
A4GBAB0p0ySmfMkm3z8H4P8WwWJ8bMO2RNXEx0i9fU2ncJfdY0zEPYvM6zpUhwJP
T9DsQBPdSy+VLbJ/PtYoiKIcupd+vriGYn3mqckXy7RBLqpiVsnw1rGE28oG4I9N
u0p2AwDuC+KNuHgtrGxYrRnFTRKZpj2AoGuW1a6eSaNOhPeq
-----END CERTIFICATE REQUEST-----

我得到的是这样的:

-----BEGIN+CERTIFICATE+REQUEST-----%0D%0AMIIB0DCCATkCAQAwgY8xCzAJBgNVBAYTAkFVMSgwJgYDVQQKDB9UaGUgTGVnaW9u%0D%0AIG9mIHRoZSBCb3VuY3kgQ2FzdGxlMRIwEAYDVQQHDAlNZWxib3VybmUxETAPBgNV%0D%0ABAgMCFZpY3RvcmlhMS8wLQYJKoZIhvcNAQkBFiBmZWVkYmFjay1jcnlwdG9AYm91%0D%0AbmN5Y2FzdGxlLm9yZzCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAqE+pEyKS%0D%0A21JXgeldS83E+SSmQ4s2xYPqV7Yw2ebUKsxnbBz1KcXHdHS1MJ9PYzBJgogdigZW%0D%0A6hEQ83edP%2FAy%2FEQzGqeKUzFqNEsQh3PSdbF9N5k7b81tQHUbfIbNu1ofSBNa%2FEit%0D%0AMkOj1NAmwivpW0AA8aPZhGzYLYWcp0lsC78CAwEAAaAAMA0GCSqGSIb3DQEBBQUA%0D%0AA4GBAB0p0ySmfMkm3z8H4P8WwWJ8bMO2RNXEx0i9fU2ncJfdY0zEPYvM6zpUhwJP%0D%0AT9DsQBPdSy+VLbJ%2FPtYoiKIcupd+vriGYn3mqckXy7RBLqpiVsnw1rGE28oG4I9N%0D%0Au0p2AwDuC+KNuHgtrGxYrRnFTRKZpj2AoGuW1a6eSaNOhPeq%0D%0A-----END+CERTIFICATE+REQUEST-----%0D%0A=

这不好,因为它会杀死 PEM 解析器。

但是,我可以通过使用以下转换来克服这个问题,并且字符串变得可解析......

  receivedData = receivedData.replace("-----BEGIN+CERTIFICATE+REQUEST-----", "-----BEGIN CERTIFICATE REQUEST-----");
  receivedData = receivedData.replace("-----END+CERTIFICATE+REQUEST-----", "-----END CERTIFICATE REQUEST-----");
  receivedData = receivedData.replace("%0D%0A", "\r\n");
  receivedData = receivedData.replace("%2F", "/");
  receivedData = receivedData.substring(0, receivedData.length()-1);

在这种情况下解析成功...

2014-10-18 16:26:43.864  INFO 2432 --- [nio-8080-exec-1] demo.HelloController                     : PemParser returned: org.spongycastle.pkcs.PKCS10CertificationRequest@f5c81fb9
2014-10-18 16:26:43.870  INFO 2432 --- [nio-8080-exec-1] demo.HelloController                     : SUCCESS

...但我不认为这是一种真正可行的做事方式,因为它看起来不太稳定。一定有更好的方法。我尝试了 URLEncoder.decode(receivedData, "UTF-8"); 但它从实际数据中删除了 + 字符,这也不好,因为它使数据无法解析。

当我通过网络发送字符串时,是否有人对如何防止此URL转义有任何有用的想法?

最佳答案

嗯,我仍然不知道为什么会发生这种情况,但我确实做了一个解决方法。

我不是仅仅发送 String byte[ ] 内容,而是使用 https://jsonp.java.net/index.html 将数据包装在 JSON 对象中。根据http://docs.oracle.com/javaee/7/tutorial/doc/jsonp003.htm .

JsonObject model = Json.createObjectBuilder().add("data", sw.toString()).build();

httpsUrlConnection = (HttpsURLConnection)new URL("http://127.0.0.1:8443/upload").openConnection();

httpsUrlConnection.setDoInput(true);
httpsUrlConnection.setDoOutput(true);
httpsUrlConnection.setRequestProperty("Content-Type", "application/json");
httpsUrlConnection.setRequestMethod("POST");
httpsUrlConnection.connect();
try(JsonWriter jsonWriter = Json.createWriter(httpsUrlConnection.getOutputStream()))
{
    jsonWriter.write(model);  
}
httpsUrlConnection.getOutputStream().flush();
System.out.println("Response code: " + httpsUrlConnection.getResponseCode());

然后,在 Spring Boot 端,我进行了设置,以便按照 https://stackoverflow.com/a/8946142/2413303 接收 JSON 对象。 :

public class DataObject {
  private String data;

  public String getData() {
    return data;
  }

  public void setData(String data) {
    this.data = data;
  }
}

在 Controller 中:

@RequestMapping(value="/upload", method = RequestMethod.POST, consumes="application/json")
  public ResponseEntity<String> upload(@RequestBody DataObject uploadedData)
    throws NoSuchAlgorithmException, InvalidKeyException, IOException {
    if(uploadedData == null) {
      log.info("Uploaded data was null.");
      return new ResponseEntity<String>("Data was null after upload.", HttpStatus.INTERNAL_SERVER_ERROR);
    }
    String receivedData = uploadedData.getData();
    log.info("" + receivedData);

这样字符就不会被转义:

2014-10-20 13:03:15.550  INFO 6924 --- [nio-8443-exec-3] demo.HelloController                     : -----BEGIN CERTIFICATE REQUEST-----

编辑:我被告知问题是在没有设置内容类型的情况下,Spring默认为application/x-www-form-urlencoded,因为它应该已application/octet-stream

编辑2:使用octet-stream,将其通过转义的\r\n发送到\\r\\n,并出于某种原因在字符串前后添加了 "。不知道为什么。

关于java - 作为 byte[] 发送的字符串内容将其内容转义为查询字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26440897/

相关文章:

Java:Master Gain 不支持异常

java - 带有 BindingAdapter 的 MutableLiveData 不更新 View 的可见性

java - Animate ball JPanel subclass is not abstract错误

java - Spring在Struts + Spring + Hibernate中的作用是什么?

spring - 是否可以使用 spring 重定向 url

forms - Spring 形式 :options tag with enum

java - void 方法的单元测试应该是什么样子?

数组元素上的 Javascript IndexOf

c++ - 我如何重新排列它以便输入 Ctrl-z(文件结束信号)中断循环,同时仍在 C++ 中测试有效输入

c++ - 为什么 string::data() 不提供可变字符 *?