网络通信——HTTP接口访问——POST方式调用HTTP接口

妖狐艹你老母 2024-04-03 12:54 211阅读 0赞

GET方式的毛病在于:请求参数只能以“参数名=参数值&参数名=参数值”这样的格式添加到接口地址末尾,使得它无法传送复杂结构的请求报文。

POST方式把接口地址与请求报文分开,允许使用自定义的报文格式(如JSON),由此扩大了该方式的应用场景。

5009f17e6b8c68225074a3406201f9ba.png

dc86eb572a38c4984ed92574020191b8.png

=================================================================================================================

布局:

  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. android:layout_width="match_parent"
  3. android:layout_height="match_parent"
  4. android:orientation="vertical"
  5. android:padding="5dp" >
  6. <LinearLayout
  7. android:layout_width="match_parent"
  8. android:layout_height="40dp" >
  9. <TextView
  10. android:layout_width="wrap_content"
  11. android:layout_height="match_parent"
  12. android:gravity="center"
  13. android:text="请选择要更新的应用:"
  14. android:textColor="@color/black"
  15. android:textSize="17sp" />
  16. <Spinner
  17. android:id="@+id/sp_app_name"
  18. android:layout_width="0dp"
  19. android:layout_height="match_parent"
  20. android:layout_weight="1"
  21. android:spinnerMode="dialog" />
  22. </LinearLayout>
  23. <ImageView
  24. android:id="@+id/iv_app"
  25. android:layout_width="match_parent"
  26. android:layout_height="50dp"
  27. android:scaleType="fitCenter" />
  28. <TextView
  29. android:id="@+id/tv_app_result"
  30. android:layout_width="match_parent"
  31. android:layout_height="wrap_content"
  32. android:textColor="@color/black"
  33. android:textSize="17sp" />
  34. </LinearLayout>

主代码:

  1. package com.example.myapplication;
  2. import android.annotation.SuppressLint;
  3. import android.os.Bundle;
  4. import android.text.TextUtils;
  5. import android.view.View;
  6. import android.widget.AdapterView;
  7. import android.widget.ArrayAdapter;
  8. import android.widget.ImageView;
  9. import android.widget.Spinner;
  10. import android.widget.TextView;
  11. import android.widget.Toast;
  12. import androidx.appcompat.app.AppCompatActivity;
  13. import com.example.myapplication.bean.PackageInfo;
  14. import com.example.myapplication.constant.ApkConstant;
  15. import com.example.myapplication.task.CheckUpdateTask;
  16. import com.example.myapplication.task.req.CheckUpdateReq;
  17. import com.example.myapplication.task.resp.CheckUpdateResp;
  18. import com.google.gson.Gson;
  19. @SuppressLint("DefaultLocale")
  20. public class MainActivity extends AppCompatActivity implements CheckUpdateTask.OnCheckUpdateListener
  21. {
  22. private static final String TAG = "HttpPostActivity";
  23. private Spinner sp_app_name; // 应用名称的下拉框
  24. private ImageView iv_app;
  25. private TextView tv_app_result;
  26. private boolean isFirstSelect = true; // 是否首次选择
  27. @Override
  28. protected void onCreate(Bundle savedInstanceState)
  29. {
  30. super.onCreate(savedInstanceState);
  31. setContentView(R.layout.activity_main);
  32. iv_app = findViewById(R.id.iv_app);
  33. tv_app_result = findViewById(R.id.tv_app_result);
  34. initAppSpinner(); // 初始化应用名称的下拉框
  35. }
  36. // 初始化应用名称的下拉框
  37. private void initAppSpinner() {
  38. ArrayAdapter<String> apkNameAdapter = new ArrayAdapter<String>(this,
  39. R.layout.item_select, ApkConstant.NAME_ARRAY);
  40. sp_app_name = findViewById(R.id.sp_app_name);
  41. sp_app_name.setPrompt("请选择要更新的应用");
  42. sp_app_name.setAdapter(apkNameAdapter);
  43. sp_app_name.setOnItemSelectedListener(new AppNameSelectedListener());
  44. sp_app_name.setSelection(0);
  45. }
  46. class AppNameSelectedListener implements AdapterView.OnItemSelectedListener {
  47. public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
  48. if (isFirstSelect) { // 刚打开页面时不需要执行下载动作
  49. isFirstSelect = false;
  50. return;
  51. }
  52. queryAppInfo(arg2); // 查询应用的详细信息
  53. }
  54. public void onNothingSelected(AdapterView<?> arg0) {}
  55. }
  56. // 查询应用的详细信息
  57. private void queryAppInfo(int pos) {
  58. iv_app.setImageResource(ApkConstant.ICON_ARRAY[pos]); // 设置图像视图的资源图片
  59. CheckUpdateReq req = new CheckUpdateReq(); // 创建检查更新的请求对象
  60. req.package_list.add(new PackageInfo(ApkConstant.PACKAGE_ARRAY[pos]));
  61. String content = new Gson().toJson(req); // 把检查更新的请求对象转换为json字符串
  62. CheckUpdateTask task = new CheckUpdateTask(); // 创建一个检查应用更新的异步任务
  63. task.setCheckUpdateListener(this); // 设置应用更新检查的监听器
  64. task.execute(content); // 把应用更新检查任务加入到处理队列
  65. }
  66. // 在结束应用更新检查时触发
  67. @Override
  68. public void finishCheckUpdate(String resp) {
  69. if (TextUtils.isEmpty(resp)) {
  70. Toast.makeText(this, "应用检查更新失败", Toast.LENGTH_SHORT).show();
  71. return;
  72. }
  73. // 把JSON串转换为对应结构的实体对象
  74. CheckUpdateResp checkResp = new Gson().fromJson(resp, CheckUpdateResp.class);
  75. if (checkResp!=null && checkResp.package_list!=null && checkResp.package_list.size()>0) {
  76. PackageInfo info = checkResp.package_list.get(0);
  77. String desc = String.format("应用检查更新结果如下:\n应用名称:%s\n应用包名:%s\n最新版本:%s\n下载地址:%s",
  78. info.app_name, info.package_name, info.new_version, info.download_url);
  79. tv_app_result.setText(desc); // 显示当前选中应用的检查更新结果
  80. }
  81. }
  82. }
  83. CheckUpdateResp
  84. package com.example.myapplication.task.resp;
  85. import com.example.myapplication.bean.PackageInfo;
  86. import java.util.ArrayList;
  87. import java.util.List;
  88. public class CheckUpdateResp
  89. {
  90. public List<PackageInfo> package_list = new ArrayList<PackageInfo>();
  91. }
  92. PackageInfo
  93. package com.example.myapplication.bean;
  94. public class PackageInfo
  95. {
  96. public String app_name; // 应用名称
  97. public String package_name; // 应用包名
  98. public String download_url; // 下载地址
  99. public String new_version; // 新版本号
  100. public PackageInfo()
  101. {
  102. app_name = "";
  103. package_name = "";
  104. download_url = "";
  105. new_version = "";
  106. }
  107. public PackageInfo(String package_name)
  108. {
  109. this.package_name = package_name;
  110. }
  111. }
  112. UrlConstant
  113. package com.example.myapplication.constant;
  114. public class UrlConstant
  115. {
  116. // 以下是HTTP接口调用的服务地址,根据实际情况修改IP和端口
  117. public static final String REQUEST_URL = "http://192.168.1.7:8080/NetServer";
  118. //public static final String REQUEST_URL = "http://192.168.1.2:6001/NetServer";
  119. // 检查应用更新的服务地址
  120. public static final String CHECK_UPDATE_URL = REQUEST_URL + "/checkUpdate";
  121. // 上传文件的服务地址
  122. public static final String UPLOAD_URL = REQUEST_URL + "/uploadServlet";
  123. // 获取旅游图片的服务地址
  124. public static final String GET_PHOTO_URL = REQUEST_URL + "/getPhoto";
  125. // 根据经纬度查询详细地址的网址(天地图)
  126. public static final String GET_ADDRESS_URL = "https://api.tianditu.gov.cn/geocoder?postStr={'lon':%f,'lat':%f,'ver':1}&type=geocode&tk=145897399844a50e3de2309513c8df4b";
  127. // 请求图片验证码的网址
  128. public static final String IMAGE_CODE_URL = "http://yx12.fjjcjy.com/Public/Control/GetValidateCode?time=";
  129. }
  130. CheckUpdateTask
  131. package com.example.myapplication.task;
  132. import android.os.AsyncTask;
  133. import android.util.Log;
  134. import com.example.myapplication.constant.UrlConstant;
  135. import com.example.myapplication.util.HttpUtil;
  136. // 检查应用更新的异步任务
  137. public class CheckUpdateTask extends AsyncTask<String, Void, String>
  138. {
  139. private final static String TAG = "CheckUpdateTask";
  140. // public CheckUpdateTask() {
  141. // super();
  142. // }
  143. // 线程正在后台处理
  144. protected String doInBackground(String... params)
  145. {
  146. String req = params[0]; // HTTP请求内容
  147. Log.d(TAG, "req = " + req);
  148. // 发送HTTP请求信息,并获得HTTP应答内容。检查更新的接口地址见UrlConstant.java
  149. String resp = HttpUtil.post(UrlConstant.CHECK_UPDATE_URL, req, null);
  150. Log.d(TAG, "resp = " + resp);
  151. return resp; // 返回HTTP应答内容
  152. }
  153. // 线程已经完成处理
  154. protected void onPostExecute(String resp)
  155. {
  156. mListener.finishCheckUpdate(resp); // HTTP调用完毕,触发监听器的结束检查事件
  157. }
  158. private OnCheckUpdateListener mListener; // 声明一个结束更新检查的监听器对象
  159. // 设置结束更新检查的监听器
  160. public void setCheckUpdateListener(OnCheckUpdateListener listener)
  161. {
  162. mListener = listener;
  163. }
  164. // 定义一个结束更新检查的监听器接口
  165. public interface OnCheckUpdateListener
  166. {
  167. void finishCheckUpdate(String resp);
  168. }
  169. }
  170. HttpUtil
  171. package com.example.myapplication.util;
  172. import android.graphics.Bitmap;
  173. import android.graphics.BitmapFactory;
  174. import android.util.Log;
  175. import java.io.ByteArrayOutputStream;
  176. import java.io.DataOutputStream;
  177. import java.io.FileInputStream;
  178. import java.io.IOException;
  179. import java.io.InputStream;
  180. import java.io.OutputStream;
  181. import java.net.HttpURLConnection;
  182. import java.net.URL;
  183. import java.security.SecureRandom;
  184. import java.security.cert.X509Certificate;
  185. import java.util.Map;
  186. import java.util.zip.GZIPInputStream;
  187. import javax.net.ssl.HostnameVerifier;
  188. import javax.net.ssl.HttpsURLConnection;
  189. import javax.net.ssl.SSLContext;
  190. import javax.net.ssl.SSLSession;
  191. import javax.net.ssl.TrustManager;
  192. import javax.net.ssl.X509TrustManager;
  193. public class HttpUtil {
  194. private final static String TAG = "HttpUtil";
  195. private final static int CONNECT_TIMEOUT = 15000;
  196. private final static int READ_TIMEOUT = 15000;
  197. // 兼容https开头的调用地址
  198. private static void compatibleSSL(String callUrl) throws Exception {
  199. if (callUrl.toLowerCase().startsWith("https")) {
  200. SSLContext sc = SSLContext.getInstance("TLS");
  201. sc.init(null, new TrustManager[]{
  202. new X509TrustManager() {
  203. @Override
  204. public X509Certificate[] getAcceptedIssuers() {
  205. return null;
  206. }
  207. @Override
  208. public void checkServerTrusted(X509Certificate[] arg0, String arg1) {
  209. }
  210. @Override
  211. public void checkClientTrusted(X509Certificate[] arg0, String arg1) {
  212. }
  213. }}, new SecureRandom());
  214. HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());
  215. HttpsURLConnection.setDefaultHostnameVerifier(new HostnameVerifier() {
  216. @Override
  217. public boolean verify(String hostname, SSLSession session) {
  218. return true;
  219. }
  220. });
  221. }
  222. }
  223. // 对指定接口地址发起GET调用
  224. public static String get(String callUrl, Map<String, String> headers)
  225. {
  226. String resp = ""; // 应答内容
  227. try
  228. {
  229. Log.d(TAG, "请求地址:"+callUrl);
  230. compatibleSSL(callUrl); // 兼容https开头的调用地址
  231. URL url = new URL(callUrl); // 根据网址字符串构建URL对象
  232. // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
  233. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  234. conn.setRequestMethod("GET"); // 设置请求方式
  235. setConnHeader(conn, headers);// 设置HTTP连接的头部信息
  236. conn.connect(); // 开始连接
  237. // 打印HTTP调用的应答内容长度、内容类型、压缩方式
  238. Log.d(TAG, String.format("应答内容长度=%d, 内容类型=%s, 压缩方式=%s", conn.getContentLength(), conn.getContentType(), conn.getContentEncoding()) );
  239. // 对输入流中的数据解压和字符编码,得到原始的应答字符串
  240. resp = getUnzipString(conn);
  241. // 打印HTTP调用的应答状态码和应答报文
  242. Log.d(TAG, String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
  243. conn.disconnect(); // 断开连接
  244. }
  245. catch (Exception e)
  246. {
  247. e.printStackTrace();
  248. }
  249. return resp;
  250. }
  251. // 从指定url获取图片
  252. public static Bitmap getImage(String callUrl, Map<String, String> headers) {
  253. Bitmap bitmap = null; // 位图对象
  254. try {
  255. Log.d(TAG, "请求地址:"+callUrl);
  256. compatibleSSL(callUrl); // 兼容https开头的调用地址
  257. URL url = new URL(callUrl); // 根据网址字符串构建URL对象
  258. // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
  259. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  260. conn.setRequestMethod("GET"); // 设置请求方式
  261. setConnHeader(conn, headers);// 设置HTTP连接的头部信息
  262. conn.connect(); // 开始连接
  263. // 打印图片获取的应答内容长度、内容类型、压缩方式
  264. Log.d(TAG, String.format("应答内容长度=%d, 内容类型=%s, 压缩方式=%s",
  265. conn.getContentLength(), conn.getContentType(), conn.getContentEncoding()) );
  266. // 对输入流中的数据解码,得到位图对象
  267. bitmap = BitmapFactory.decodeStream(conn.getInputStream());
  268. // 打印图片获取的应答状态码和位图大小
  269. Log.d(TAG, String.format("应答状态码=%d, 位图大小=%s", conn.getResponseCode(), bitmap.getByteCount()) );
  270. conn.disconnect(); // 断开连接
  271. } catch (Exception e) {
  272. e.printStackTrace();
  273. }
  274. return bitmap;
  275. }
  276. // 对指定接口地址发起POST调用
  277. public static String post(String callUrl, String req, Map<String, String> headers) {
  278. String resp = ""; // 应答内容
  279. try {
  280. Log.d(TAG, "请求地址:"+callUrl+", 请求报文="+req);
  281. compatibleSSL(callUrl); // 兼容https开头的调用地址
  282. URL url = new URL(callUrl); // 根据网址字符串构建URL对象
  283. // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
  284. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  285. conn.setRequestMethod("POST"); // 设置请求方式
  286. setConnHeader(conn, headers);// 设置HTTP连接的头部信息
  287. conn.setRequestProperty("Content-Type", "application/json"); // 请求报文为json格式
  288. conn.setDoOutput(true); // 准备让连接执行输出操作。默认为false,POST方式需要设置为true
  289. //conn.setDoInput(true); // 准备让连接执行输入操作。默认为true
  290. conn.connect(); // 开始连接
  291. OutputStream os = conn.getOutputStream(); // 从连接对象中获取输出流
  292. os.write(req.getBytes()); // 往输出流写入请求报文
  293. // 打印HTTP调用的应答内容长度、内容类型、压缩方式
  294. Log.d(TAG, String.format("应答内容长度=%s, 内容类型=%s, 压缩方式=%s",
  295. conn.getHeaderField("Content-Length"), conn.getHeaderField("Content-Type"),
  296. conn.getHeaderField("Content-Encoding")) );
  297. // 对输入流中的数据解压和字符编码,得到原始的应答字符串
  298. resp = getUnzipString(conn);
  299. // 打印HTTP调用的应答状态码和应答报文
  300. Log.d(TAG, String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
  301. conn.disconnect(); // 断开连接
  302. } catch (Exception e) {
  303. e.printStackTrace();
  304. }
  305. return resp;
  306. }
  307. // 把文件上传给指定的URL
  308. public static String upload(String uploadUrl, String uploadFile, Map<String, String> headers)
  309. {
  310. String resp = ""; // 应答内容
  311. // 从本地文件路径获取文件名
  312. String fileName = uploadFile.substring(uploadFile.lastIndexOf("/"));
  313. String end = "\r\n"; // 结束字符串
  314. String hyphens = "--"; // 连接字符串
  315. String boundary = "WUm4580jbtwfJhNp7zi1djFEO3wNNm"; // 边界字符串
  316. try (FileInputStream fis = new FileInputStream(uploadFile)) {
  317. Log.d(TAG, "上传地址:"+uploadUrl+", 上传文件="+uploadFile);
  318. compatibleSSL(uploadUrl); // 兼容https开头的调用地址
  319. URL url = new URL(uploadUrl); // 根据网址字符串构建URL对象
  320. // 打开URL对象的网络连接,并返回HttpURLConnection连接对象
  321. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  322. conn.setRequestMethod("POST"); // 设置请求方式
  323. setConnHeader(conn, headers);// 设置HTTP连接的头部信息
  324. conn.setDoOutput(true); // 准备让连接执行输出操作。默认为false,POST方式需要设置为true
  325. //conn.setDoInput(true); // 准备让连接执行输入操作。默认为true
  326. conn.setRequestProperty("Connection", "Keep-Alive"); // 连接过程要保持活跃
  327. // 请求报文要求分段传输,并且各段之间以边界字符串隔开
  328. conn.setRequestProperty("Content-Type", "multipart/form-data;boundary=" + boundary);
  329. // 根据连接对象的输出流构建数据输出流
  330. DataOutputStream ds = new DataOutputStream(conn.getOutputStream());
  331. // 以下写入请求报文的头部
  332. ds.writeBytes(hyphens + boundary + end);
  333. ds.writeBytes("Content-Disposition: form-data; "
  334. + "name=\"file\";filename=\"" + fileName + "\"" + end);
  335. ds.writeBytes(end);
  336. // 以下写入请求报文的主体
  337. byte[] buffer = new byte[1024];
  338. int length;
  339. // 先将文件数据写入到缓冲区,再将缓冲数据写入输出流
  340. while ((length = fis.read(buffer)) != -1) {
  341. ds.write(buffer, 0, length);
  342. }
  343. ds.writeBytes(end);
  344. // 以下写入请求报文的尾部
  345. ds.writeBytes(hyphens + boundary + hyphens + end);
  346. ds.close(); // 关闭数据输出流
  347. // 打印HTTP调用的应答内容长度、内容类型、压缩方式
  348. Log.d(TAG, String.format("应答内容长度=%s, 内容类型=%s, 压缩方式=%s",
  349. conn.getHeaderField("Content-Length"), conn.getHeaderField("Content-Type"),
  350. conn.getHeaderField("Content-Encoding")) );
  351. // 对输入流中的数据解压和字符编码,得到原始的应答字符串
  352. resp = getUnzipString(conn);
  353. // 打印HTTP上传的应答状态码和应答报文
  354. Log.d(TAG, String.format("应答状态码=%d, 应答报文=%s", conn.getResponseCode(), resp) );
  355. conn.disconnect(); // 断开连接
  356. }
  357. catch (Exception e)
  358. {
  359. e.printStackTrace();
  360. }
  361. return resp;
  362. }
  363. // 设置HTTP连接的头部信息
  364. private static void setConnHeader(HttpURLConnection conn, Map<String, String> headers)
  365. {
  366. conn.setConnectTimeout(CONNECT_TIMEOUT); // 设置连接的超时时间,单位毫秒
  367. conn.setReadTimeout(READ_TIMEOUT); // 设置读取应答数据的超时时间,单位毫秒
  368. conn.setRequestProperty("Accept", "*/*"); // 设置数据格式
  369. conn.setRequestProperty("Accept-Language", "zh-CN"); // 设置文本语言
  370. conn.setRequestProperty("Accept-Encoding", "gzip, deflate"); // 设置编码格式
  371. // 设置用户代理,包括操作系统版本、浏览器版本等等
  372. conn.setRequestProperty("User-Agent", "Mozilla/5.0 (Windows NT 6.1; WOW64; rv:33.0) Gecko/20100101 Firefox/33.0");
  373. if (headers != null)
  374. {
  375. for (Map.Entry<String, String> item : headers.entrySet())
  376. {
  377. conn.setRequestProperty(item.getKey(), item.getValue());
  378. }
  379. }
  380. }
  381. // 把输入流中的数据按照指定字符编码转换为字符串。处理大量数据需要使用本方法
  382. private static String isToString(InputStream is, String charset)
  383. {
  384. String result = "";
  385. // 创建一个字节数组的输出流对象
  386. try (ByteArrayOutputStream baos = new ByteArrayOutputStream())
  387. {
  388. int i = -1;
  389. while ((i = is.read()) != -1) // 循环读取输入流中的字节数据
  390. {
  391. baos.write(i); // 把字节数据写入字节数组输出流
  392. }
  393. byte[] data = baos.toByteArray(); // 把字节数组输出流转换为字节数组
  394. result = new String(data, charset); // 将字节数组按照指定的字符编码生成字符串
  395. }
  396. catch (Exception e)
  397. {
  398. e.printStackTrace();
  399. }
  400. return result; // 返回转换后的字符串
  401. }
  402. // 从HTTP连接中获取已解压且重新编码后的应答报文
  403. private static String getUnzipString(HttpURLConnection conn) throws IOException
  404. {
  405. String contentType = conn.getContentType(); // 获取应答报文的内容类型(包括字符编码)
  406. String charset = "UTF-8"; // 默认的字符编码为UTF-8
  407. if (contentType != null)
  408. {
  409. if (contentType.toLowerCase().contains("charset=gbk")) // 应答报文采用gbk编码
  410. {
  411. charset = "GBK"; // 字符编码改为GBK
  412. }
  413. else if (contentType.toLowerCase().contains("charset=gb2312")) // 采用gb2312编码
  414. {
  415. charset = "GB2312"; // 字符编码改为GB2312
  416. }
  417. }
  418. String contentEncoding = conn.getContentEncoding(); // 获取应答报文的压缩方式
  419. InputStream is = conn.getInputStream(); // 获取HTTP连接的输入流对象
  420. String result = "";
  421. if (contentEncoding != null && contentEncoding.contains("gzip")) // 应答报文使用gzip压缩
  422. {
  423. // 根据输入流对象构建压缩输入流
  424. try (GZIPInputStream gis = new GZIPInputStream(is))
  425. {
  426. // 把压缩输入流中的数据按照指定字符编码转换为字符串
  427. result = isToString(gis, charset);
  428. }
  429. catch (Exception e)
  430. {
  431. e.printStackTrace();
  432. }
  433. }
  434. else
  435. {
  436. // 把输入流中的数据按照指定字符编码转换为字符串
  437. result = isToString(is, charset);
  438. }
  439. return result; // 返回处理后的应答报文
  440. }
  441. }

ca16966b92d2e3e0acdaf0d3c7345dc5.png

0e76a6070ee3c40378a71ac28e7146e8.png

e74d71fb991f42ac1c84ba6666df0148.png

c26c6dfb98347e878a760de4d20c4845.png

c953fef0c8ff5db9910aa38e616516cc.png

5338036212c2441e5314b3dad98c6f6e.png

37d075e39df41a1d2799e42ea08b6968.png

d8cbf41bcb450a9ab27869c047eb2c0b.png

发表评论

表情:
评论列表 (有 0 条评论,211人围观)

还没有评论,来说两句吧...

相关阅读