最近由于项目需求,需要完成批量照片上传,折腾了一段时间,终于完成了,达到了如下效果

主界面主要有GridView组成和按钮组成,当按下一个格点时,会调用相机或者相册,拍照或者选择相册照片,选择完成之后,将缩略图显示在GridView,在这里说明一下,如果GridView显示不出来,说明图片太大了,需要压缩,在我的上一篇博客,详细讲解了图片压缩的原理与过程,这里不再赘述。

下面贴上代码,

主界面:

package com.qian.pos;import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Picture;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.GridView;
import android.widget.ImageView;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;import com.qian.pos.util.BitmapUtil;
import com.qian.pos.util.FileUtils;
import com.qian.pos.util.PictureUtil;
import com.qian.pos.util.UploadUtil;
import com.qian.pos.util.UploadUtil.OnUploadProcessListener;
import com.qian.servletasynchttp.R;public class ImageUploadActivity extends Activity// implements OnUploadProcessListener
{private static final String TAG = "uploadImage";protected static final int TO_UPLOAD_FILE = 1;  protected static final int UPLOAD_FILE_DONE = 2;  public static final int TO_SELECT_PHOTO = 3;private static final int UPLOAD_INIT_PROCESS = 4;private static final int UPLOAD_IN_PROCESS = 5;private static String requestURL = "http://114.55.72.18/UnionPay/UploadAction";private Button uploadButton;//private ProgressBar progressBar;private String picPath = null;private ProgressDialog progressDialog;private GridView list_gv;private MyAdapter adapter;private HashMap<Integer,Bitmap> imageMap = new HashMap<Integer, Bitmap>();private HashMap<Integer,String> filePathMap = new HashMap<Integer, String>();private Handler handler = new Handler(){@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case TO_UPLOAD_FILE:toUploadFile();break;case UPLOAD_INIT_PROCESS://progressBar.setMax(msg.arg1);break;case UPLOAD_IN_PROCESS://progressBar.setProgress(msg.arg1);break;case UPLOAD_FILE_DONE:String result = "响应码:"+msg.arg1+"\n响应信息:"+msg.obj+"\n耗时:"+UploadUtil.getRequestTime()+"秒";break;default:break;}super.handleMessage(msg);}};@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);initView();}private void initView() {uploadButton = (Button) this.findViewById(R.id.uploadImage);uploadButton.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {if(picPath!=null){handler.sendEmptyMessage(TO_UPLOAD_FILE);}else{Toast.makeText(ImageUploadActivity.this, "上传的文件路径出错", Toast.LENGTH_LONG).show();}}});progressDialog = new ProgressDialog(this);list_gv = (GridView) findViewById(R.id.gv_image);adapter = new MyAdapter();list_gv.setAdapter(adapter);list_gv.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id) {Intent intent;switch (position) {case 0:intent = new Intent(ImageUploadActivity.this,SelectPicActivity.class);startActivityForResult(intent, 0);break;case 1:intent = new Intent(ImageUploadActivity.this,SelectPicActivity.class);startActivityForResult(intent, 1);break;case 2:intent = new Intent(ImageUploadActivity.this,SelectPicActivity.class);startActivityForResult(intent, 2);break;case 3:intent = new Intent(ImageUploadActivity.this,SelectPicActivity.class);startActivityForResult(intent, 3);break;case 4:intent = new Intent(ImageUploadActivity.this,SelectPicActivity.class);startActivityForResult(intent, 4);break;}}});}private class MyAdapter extends BaseAdapter {@Overridepublic int getCount() {return 5;}@Overridepublic Object getItem(int position) {return null;}@Overridepublic long getItemId(int position) {return 0;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {View view = View.inflate(ImageUploadActivity.this, R.layout.item_grid, null);ImageView image = (ImageView) view.findViewById(R.id.item_grida_image);Iterator<Integer> iterator = imageMap.keySet().iterator();while(iterator.hasNext()) {Integer next = iterator.next();if(next.intValue() == position) {image.setImageBitmap(imageMap.get(next));}}TextView textView = (TextView) view.findViewById(R.id.tv_explain);switch (position) {case 0:textView.setText("照片1");break;case 1:textView.setText("<span style="font-family: 'Microsoft YaHei';">照片2</span>");break;case 2:textView.setText("照片3");break;case 3:textView.setText("照片4");break;case 4:textView.setText("照片5");break;default:break;}return view;}}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {if(resultCode==Activity.RESULT_OK) {picPath = data.getStringExtra(SelectPicActivity.KEY_PHOTO_PATH);Log.i(TAG, "最终选择的图片="+picPath);// Toast.makeText(getApplicationContext(), "最终选择的图片="+picPath, 0).show();Bitmap bm = BitmapFactory.decodeFile(picPath);//Bitmap tempBitmap = BitmapUtil.createImageThumbnail(picPath,128);//压缩图片//Bitmap saveBitmap = BitmapUtil.createImageThumbnail(picPath,2048);Bitmap tempBitmap = PictureUtil.getSmallBitmap(picPath, 128, 128);//压缩图片Bitmap saveBitmap = PictureUtil.getSmallBitmap(picPath,1280,720);//上传服务器的bitmap 手机横着拍照String path = Environment.getExternalStorageDirectory()+ "/pos/"+requestCode+".JPEG";FileUtils.saveBitmap(saveBitmap, requestCode+"");filePathMap.put(requestCode, path);imageMap.put(requestCode, tempBitmap);//Toast.makeText(ImageUploadActivity.this, "第"+requestCode+"张图片", 0).show();//  System.out.println("imageMap"+imageMap.size());//    System.out.println("filePathMap"+filePathMap.size());adapter.notifyDataSetChanged();}super.onActivityResult(requestCode, resultCode, data);}private void toUploadFile(){progressDialog.setMessage("正在上传文件...");progressDialog.show();final String fileKey = "upload";final UploadUtil uploadUtil = UploadUtil.getInstance();;uploadUtil.setOnUploadProcessListener(new OnUploadProcessListener() {@Overridepublic void onUploadProcess(int uploadSize) {Message msg = Message.obtain();msg.what = UPLOAD_IN_PROCESS;msg.arg1 = uploadSize; handler.sendMessage(msg);}@Overridepublic void onUploadDone(int responseCode, String message) {progressDialog.dismiss();Message msg = Message.obtain();msg.what = UPLOAD_FILE_DONE;msg.arg1 = responseCode;msg.obj = message;handler.sendMessage(msg);}@Overridepublic void initUpload(int fileSize) {Message msg = Message.obtain();msg.what = UPLOAD_INIT_PROCESS;msg.arg1 = fileSize;handler.sendMessage(msg);}});  //设置监听器监听上传状态final Map<String, String> params = new HashMap<String, String>();params.put("dpnumber", "13800001111");System.out.println(filePathMap.size());new Thread(new Runnable() {int i = 0;@Overridepublic void run() {final boolean uploadFile = uploadUtil.uploadFile(filePathMap,fileKey,requestURL,params);runOnUiThread(new Runnable() {@Overridepublic void run() {if(uploadFile) {Toast.makeText(ImageUploadActivity.this, "上传成功", 0).show();progressDialog.dismiss();} else {Toast.makeText(ImageUploadActivity.this, "上传失败", 0).show();progressDialog.dismiss();}}});}   }).start();}}

选择照片界面程序:

package com.qian.pos;import com.qian.servletasynchttp.R;import android.app.Activity;
import android.content.ContentValues;
import android.content.Intent;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.LinearLayout;
import android.widget.Toast;public class SelectPicActivity extends Activity implements OnClickListener{public static final int SELECT_PIC_BY_TACK_PHOTO = 1;public static final int SELECT_PIC_BY_PICK_PHOTO = 2;public static final String KEY_PHOTO_PATH = "photo_path";private static final String TAG = "SelectPicActivity";private LinearLayout dialogLayout;private Button takePhotoBtn,pickPhotoBtn,cancelBtn;/**获取到的图片路径*/private String picPath;private Intent lastIntent ;private Uri photoUri;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.select_pic_layout);initView();}/*** 初始化加载View*/private void initView() {dialogLayout = (LinearLayout) findViewById(R.id.dialog_layout);dialogLayout.setOnClickListener(this);takePhotoBtn = (Button) findViewById(R.id.btn_take_photo);takePhotoBtn.setOnClickListener(this);pickPhotoBtn = (Button) findViewById(R.id.btn_pick_photo);pickPhotoBtn.setOnClickListener(this);cancelBtn = (Button) findViewById(R.id.btn_cancel);cancelBtn.setOnClickListener(this);lastIntent = getIntent();}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.dialog_layout:finish();break;case R.id.btn_take_photo:takePhoto();break;case R.id.btn_pick_photo:pickPhoto();break;default:finish();break;}}/*** 拍照获取图片*/private void takePhoto() {//执行拍照前,应该先判断SD卡是否存在String SDState = Environment.getExternalStorageState();if(SDState.equals(Environment.MEDIA_MOUNTED)){Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//"android.media.action.IMAGE_CAPTURE"/**** 需要说明一下,以下操作使用照相机拍照,拍照后的图片会存放在相册中的* 这里使用的这种方式有一个好处就是获取的图片是拍照后的原图* 如果不实用ContentValues存放照片路径的话,拍照后获取的图片为缩略图不清晰*/ContentValues values = new ContentValues();  photoUri = this.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);  intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, photoUri);/**-----------------*/startActivityForResult(intent, SELECT_PIC_BY_TACK_PHOTO);}else{Toast.makeText(this,"内存卡不存在", Toast.LENGTH_LONG).show();}}/**** 从相册中取图片*/private void pickPhoto() {Intent intent = new Intent();intent.setType("image/*");intent.setAction(Intent.ACTION_GET_CONTENT);startActivityForResult(intent, SELECT_PIC_BY_PICK_PHOTO);}@Overridepublic boolean onTouchEvent(MotionEvent event) {finish();return super.onTouchEvent(event);}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {if(resultCode == Activity.RESULT_OK){doPhoto(requestCode,data);}super.onActivityResult(requestCode, resultCode, data);}/*** 选择图片后,获取图片的路径* @param requestCode* @param data*/private void doPhoto(int requestCode,Intent data){if(requestCode == SELECT_PIC_BY_PICK_PHOTO )  //从相册取图片,有些手机有异常情况,请注意{if(data == null){Toast.makeText(this, "选择图片文件出错", Toast.LENGTH_LONG).show();return;}photoUri = data.getData();if(photoUri == null ){Toast.makeText(this, "选择图片文件出错", Toast.LENGTH_LONG).show();return;}}String[] pojo = {MediaStore.Images.Media.DATA};Cursor cursor = managedQuery(photoUri, pojo, null, null,null);   if(cursor != null ){int columnIndex = cursor.getColumnIndexOrThrow(pojo[0]);cursor.moveToFirst();picPath = cursor.getString(columnIndex);cursor.close();}Log.i(TAG, "imagePath = "+picPath);if (picPath != null/*&& (picPath.endsWith(".png") || picPath.endsWith(".PNG")|| picPath.endsWith(".jpg") || picPath.endsWith(".JPG")|| picPath.endsWith(".JPG") || picPath.endsWith(".JPEG"))*/){lastIntent.putExtra(KEY_PHOTO_PATH, picPath);setResult(Activity.RESULT_OK, lastIntent);finish();}else{//Toast.makeText(this, picPath, Toast.LENGTH_LONG).show();Toast.makeText(this, "选择图片文件不正确", Toast.LENGTH_LONG).show();}}
}

接下来是程序的核心部分UploadUtil.java

主要实现了图片的上传,上传过程的初始化监听和上传完成的监听,还有上传耗时的计算,设置的回调接口监听器在主界面可以设置,获得必要的信息。

实现的批量的过程也在其中,使用HashMap将各个图片的路径保存在其中,然后HashMap的迭代器实现循环上传,批量上传的过程中,如果图片过大会抛出异常,加载bitmap手机太多内存,没来得及释放又要加载另一张图片,单张原图上传可以实现,但不符合市场需求,一般都是批量上传。

程序的具体执行流程就不详细说了,关键部分注释已经写好了。

package com.qian.pos.util;import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.UUID;import android.content.Context;
import android.util.Log;
import android.widget.Toast;public class UploadUtil {private static UploadUtil uploadUtil;private static final String BOUNDARY =  UUID.randomUUID().toString(); // 边界标识 随机生成private static final String PREFIX = "--";private static final String LINE_END = "\r\n";private static final String CONTENT_TYPE = "multipart/form-data"; // 内容类型private UploadUtil() {}/*** 单例模式获取上传工具类* @return*/public static UploadUtil getInstance() {if (null == uploadUtil) {uploadUtil = new UploadUtil();}return uploadUtil;}private static final String TAG = "UploadUtil";private int readTimeOut = 10 * 10000; // 读取超时private int connectTimeout = 10 * 10000; // 超时时间/**** 请求使用多长时间*/private static int requestTime = 0;private static final String CHARSET = "utf-8"; // 设置编码/**** 上传成功*/public static final int UPLOAD_SUCCESS_CODE = 1;/*** 文件不存在*/public static final int UPLOAD_FILE_NOT_EXISTS_CODE = 2;/*** 服务器出错*/public static final int UPLOAD_SERVER_ERROR_CODE = 3;protected static final int WHAT_TO_UPLOAD = 1;protected static final int WHAT_UPLOAD_DONE = 2;public boolean uploadFile(HashMap<Integer,String> filePathMap, String fileKey, String RequestURL,Map<String, String> param) {String result = null;requestTime= 0;String fileName = "";long requestTime = System.currentTimeMillis();long responseTime = 0;try {//          conn.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");Iterator<Integer> iterator = filePathMap.keySet().iterator();boolean isOk = true;System.out.println("filePathMap.size()"+filePathMap.size());while(iterator.hasNext()) {URL url = new URL(RequestURL);HttpURLConnection conn = (HttpURLConnection) url.openConnection();conn.setReadTimeout(readTimeOut);conn.setConnectTimeout(connectTimeout);conn.setDoInput(true); // 允许输入流conn.setDoOutput(true); // 允许输出流conn.setUseCaches(false); // 不允许使用缓存conn.setRequestMethod("POST"); // 请求方式conn.setRequestProperty("Charset", CHARSET); // 设置编码conn.setRequestProperty("connection", "keep-alive");conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary=" + BOUNDARY);Integer next = iterator.next();System.out.println(next+"");String filepath = filePathMap.get(next);switch (next.intValue()) {case 0:fileName = "IDPositive";break;case 1:fileName = "IDNative";break;case 2:fileName = "BusinessLicense";break;case 3:fileName = "Outdoor";break;case 4:fileName = "Indoor";break;default:break;}DataOutputStream dos = new DataOutputStream(conn.getOutputStream());StringBuffer sb = null;String params = "";if (param != null && param.size() > 0) {Iterator<String> it = param.keySet().iterator();while (it.hasNext()) {sb = null;sb = new StringBuffer();String key = it.next();String value = param.get(key);sb.append(PREFIX).append(BOUNDARY).append(LINE_END);sb.append("Content-Disposition: form-data; name=\"").append(key).append("\"").append(LINE_END).append(LINE_END);sb.append(value).append(LINE_END);params = sb.toString();Log.i(TAG, key+"="+params+"##");dos.write(params.getBytes());
//                      dos.flush();}}sb = null;params = null;sb = new StringBuffer();/*** 这里重点注意: name里面的值为服务器端需要key 只有这个key 才可以得到对应的文件* filename是文件的名字,包含后缀名的 比如:abc.png*/sb.append(PREFIX).append(BOUNDARY).append(LINE_END);sb.append("Content-Disposition:form-data; name=\"" + fileKey+ "\"; filename=\"" + fileName + "\"" + LINE_END);//+ "\"; filename=\"" + file.getName() + "\"" + LINE_END);sb.append("Content-Type:image/pjpeg" + LINE_END); // 这里配置的Content-type很重要的 ,用于服务器端辨别文件的类型的sb.append(LINE_END);params = sb.toString();sb = null;//Log.i(TAG, filepath.getName()+"=" + params+"##");//Log.i(TAG, );dos.write(params.getBytes());/**上传文件*/InputStream is = new FileInputStream(filepath);onUploadProcessListener.initUpload((int)filepath.length());byte[] bytes = new byte[1024];int len = 0;int curLen = 0;while ((len = is.read(bytes)) != -1) {curLen += len;dos.write(bytes, 0, len);onUploadProcessListener.onUploadProcess(curLen);}is.close();is = null;dos.write(LINE_END.getBytes());byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END).getBytes();dos.write(end_data);dos.flush();dos.close();dos = null;
//              dos.write(tempOutputStream.toByteArray());/*** 获取响应码 200=成功 当响应成功,获取响应的流*/int res = conn.getResponseCode();responseTime = System.currentTimeMillis();this.requestTime = (int) ((responseTime-requestTime)/1000);Log.e(TAG, "response code:" + res);if (res == 200) {Log.e(TAG, "request success");InputStream input = conn.getInputStream();StringBuffer sb1 = new StringBuffer();int ss;while ((ss = input.read()) != -1) {sb1.append((char) ss);}result = sb1.toString();Log.e(TAG, "result : " + result);sb1 = null;if(fileName == "Indoor") {sendMessage(UPLOAD_SUCCESS_CODE, "上传结果:"+ result);}conn.disconnect();//return true;} else {Log.e(TAG, "request error");sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:code=" + res);conn.disconnect();isOk = false;}}return isOk;} catch (MalformedURLException e) {sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage());e.printStackTrace();return false;} catch (IOException e) {sendMessage(UPLOAD_SERVER_ERROR_CODE,"上传失败:error=" + e.getMessage());e.printStackTrace();return false;}}private void sendMessage(int responseCode,String responseMessage){onUploadProcessListener.onUploadDone(responseCode, responseMessage);}/*** 下面是一个自定义的回调函数,用到回调上传文件是否完成* * @author shimingzheng* */public static interface OnUploadProcessListener {void onUploadDone(int responseCode, String message);void onUploadProcess(int uploadSize);void initUpload(int fileSize);}private OnUploadProcessListener onUploadProcessListener;public void setOnUploadProcessListener(OnUploadProcessListener onUploadProcessListener) {this.onUploadProcessListener = onUploadProcessListener;}public int getReadTimeOut() {return readTimeOut;}public void setReadTimeOut(int readTimeOut) {this.readTimeOut = readTimeOut;}public int getConnectTimeout() {return connectTimeout;}public void setConnectTimeout(int connectTimeout) {this.connectTimeout = connectTimeout;}/*** 获取上传使用的时间* @return*/public static int getRequestTime() {return requestTime;}public static interface uploadProcessListener{}}

Android实现批量照片上传至服务器,拍照或者从相册选择相关推荐

  1. android照相及照片上传

    android照相及照片上传 Java代码   package com.android.cist.camera.view; import java.io.ByteArrayOutputStream; ...

  2. Android拍照及从相册选择图片传详解(终极版)

    Android 拍照及从相册选择图片传详解 先上图 新知识点速览 URI(统一资源标识符)是标识逻辑或物理资源的字符序列,与URL类似,也是一串字符.通过使用位置,名称或两者来标识Internet上的 ...

  3. Android 拍照、从相册选择图片

    在做Android图片上传功能的时候,获取图片的途径一般都有两种:拍照.从相册选择. 一.拍照 调用相机拍照有两种方法: 直接返回图片. 在调用相机的时候,传入uri,拍照后通过该uri来获取图片. ...

  4. Android 点击换头像(拍照和从相册选择)

    Android 点击换头像(拍照和从相册选择) 首先是layout里的代码,但这之前需要在build.gradle里加两行代码,主要是圆形图片和别的按钮啥的,你也可以改成自己的,不加圆形图片用系统原来 ...

  5. 移动端h5实现手机拍照上传图片,相册选择上传图片input type=file

    移动端h5实现手机拍照上传图片,相册选择上传图片input type=file 有不足或者不对的在下方留言,有不明白的也请在下方留言,虽然我不一定能看到,话不多说上代码,注释都写好了 <!doc ...

  6. Android 图片 批量上传,移动端图片批量上传问题

    一.操作系统 1.ios ios的操作系统: 实现图片批量上传,可以直接使用input  type=file 加上multiple 就可以实现,至于有些图片显示有问题可以通过 new FileRead ...

  7. android 监测bug上传到服务器,基于Android 错误信息捕获发送至服务器的详解

    程序员最头疼的事情就是bug和debug.这次debug长达20天,搞的我心力交瘁.累,因为Android兼容性,不同手机会有不同的bug出来,而且很难复现,所以就上网找了下类似保存错误log到文件再 ...

  8. MUI调用照片以及裁剪和图库照片上传到服务器【MUI前端部分】

    文章目录 一.效果图展示 二. 前端样式以及JS控制转码 2.1 H5+中的调用相机模块和图库 2.2 拍照上传和从相册里上传的js控制如下 2.3 使用cropper框架实现前端裁剪 2.4 前端文 ...

  9. android 自定义相册选择,Android图片选择器,支持拍照、从相册选择、裁剪、自定义主题...

    在很多项目中都会用到图片选择器,比如在选择头像的时候,还会要求选择图片进行裁剪后再进行上传:而有的项目要支持拍照后进行图片裁剪再进行上传.由于Android系统兼容性问题,我们不得不考虑自己实现图片裁 ...

最新文章

  1. python文件操作总结
  2. OPENCV图像变换-1
  3. Ubuntu 12.04 LTS安装VMware Tools:无法找到kernel header path的问题
  4. mysql share mode_mysql锁:mysql lock in share mode 和 select for update
  5. Spring项目使用Junit4测试配置
  6. Vue学习笔记之07-v-for循环遍历
  7. Quick BI 支持多种数据源进行多维分析 1
  8. atitit.javascript调用java in swt attilax 总结
  9. Windows界面编程第十二篇 位图显示特效 飞入效果与伸展效果
  10. 利用Word2Vec在语料中构建种子词集同类词
  11. python:IndentationError:expected an indented block
  12. google vr 简介
  13. (超详细)XSS和SQL注入-网络渗透测试实验三
  14. 重新认识 D 编程语言 —— 基础篇
  15. 大数据在高校的应用场景_大数据在高校教育信息化的应用
  16. 电脑中病毒了怎么修复?电脑中病毒了怎么办?
  17. TM影像各波段及波段组合情况
  18. Javascript对日期的进行加、减操作
  19. STM32入门笔记(02):定时器之定时器中断、输入捕获和PWM输出(SPL库函数版)
  20. couchbase的使用(springboot连接couchbase,node.js连接couchbase)

热门文章

  1. Microsoft IIS tilde directory enumeration 可能的解决方法之一
  2. 在html中加一个超链接,怎么在html加超链接
  3. 招商证券有色金属行业数据库周报目录
  4. js 延迟加载的方式有哪些
  5. Java --- JUC之JMM
  6. python回车键退出_python退出程序
  7. 【JavaScript设计模式】观察者模式
  8. Python中 redis StrictRedis对象操作string类型
  9. 导出PDF文件中文显示不完整
  10. Qt 通讯模块(短信、彩信发送)