在现在的网络开发中,上传图片类的需求实在是太普通不过了,但是对于怎么样做到上传图片,对于刚开始建立项目的时候,还是有点不知所措的。也许有幸,我们做的项目是之前已经有人写过类似的用例了,那么我们只需要依葫芦画瓢就行了。

  好好了解下图片上传(文件上传)的方式,对于认知的提升还是有好处的。而且说不定哪天你就有个这样的需求呢,这里是一条龙上传。

  本文就一个从app到php层,再到java层的流程,演译下整个上传图片的流程吧。

一、app端获取用户选择的图片,转化为输入流,上传至php前端接口:

package com.dia.ration;

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.Map;
import java.util.UUID; /**
* 上传文件到服务器类
*/
public class UploadUtil {
private static final String TAG = "uploadFile";
private static final int TIME_OUT = 10 * 1000; // 超时时间
private static final String CHARSET = "utf-8"; // 设置编码
/**
* Android上传文件到服务端
*
* @param file 需要上传的文件
* @param RequestURL 请求的rul
* @return 返回响应的内容
*/
public static String uploadFile(File file, String RequestURL) {
String result = null;
String BOUNDARY = UUID.randomUUID().toString(); // 边界标识 随机生成
String PREFIX = "--", LINE_END = "\r\n";
String CONTENT_TYPE = "multipart/form-data"; // 内容类型
try {
URL url = new URL(RequestURL);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setReadTimeout(TIME_OUT);
conn.setConnectTimeout(TIME_OUT);
conn.setDoInput(true); // 允许输入流
conn.setDoOutput(true); // 允许输出流
conn.setUseCaches(false); // 不允许使用缓存
conn.setRequestMethod("POST"); // 请求方式
conn.setRequestProperty("Charset", CHARSET); // 设置编码
conn.setRequestProperty("connection", "keep-alive");
conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary=" + BOUNDARY);
if (file != null) {
DataOutputStream dos = new DataOutputStream(conn.getOutputStream());
StringBuffer sb = new StringBuffer();
sb.append(PREFIX);
sb.append(BOUNDARY);
sb.append(LINE_END);
/**
* 这里重点注意: name里面的值为服务端需要key 只有这个key 才可以得到对应的文件
* filename是文件的名字,包含后缀名的 比如:abc.png
*/
sb.append("Content-Disposition: form-data; name=\"uploadfile\"; filename=\""
+ file.getName() + "\"" + LINE_END);
sb.append("Content-Type: application/octet-stream; charset=" + CHARSET + LINE_END);
sb.append(LINE_END);
dos.write(sb.toString().getBytes());
InputStream is = new FileInputStream(file);
byte[] bytes = new byte[1024];
int len = 0;
while ((len = is.read(bytes)) != -1) {
dos.write(bytes, 0, len);
}
is.close();
dos.write(LINE_END.getBytes());
byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END).getBytes();
dos.write(end_data);
dos.flush();
InputStream input = conn.getInputStream();
StringBuffer sb1 = new StringBuffer();
int ss;
while ((ss = input.read()) != -1) {
sb1.append((char) ss);
}
result = sb1.toString();
}
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return result;
}
/**
* 通过拼接的方式构造请求内容,实现参数传输以及文件传输
*
* @param url Service net address
* @param params text content
* @param files pictures
* @return String result of Service response
* @throws IOException
*/
public static String post(String url, Map<String, String> params, Map<String, File> files)
throws IOException {
String BOUNDARY = UUID.randomUUID().toString();
String PREFIX = "--", LINEND = "\r\n";
String MULTIPART_FROM_DATA = "multipart/form-data";
String CHARSET = "UTF-8";
URL uri = new URL(url);
HttpURLConnection conn = (HttpURLConnection) uri.openConnection();
conn.setReadTimeout(10 * 1000); // 缓存的最长时间
conn.setDoInput(true); // 允许输入
conn.setDoOutput(true); // 允许输出
conn.setUseCaches(false); // 不允许使用缓存
conn.setRequestMethod("POST");
conn.setRequestProperty("connection", "keep-alive");
conn.setRequestProperty("Charsert", "UTF-8");
conn.setRequestProperty("Content-Type", MULTIPART_FROM_DATA + ";boundary=" + BOUNDARY);
// 首先组拼文本类型的参数
StringBuilder sb = new StringBuilder();
for (Map.Entry<String, String> entry : params.entrySet()) {
sb.append(PREFIX);
sb.append(BOUNDARY);
sb.append(LINEND);
sb.append("Content-Disposition: form-data; name=\"" + entry.getKey() + "\"" + LINEND);
sb.append("Content-Type: text/plain; charset=" + CHARSET + LINEND);
sb.append("Content-Transfer-Encoding: 8bit" + LINEND);
sb.append(LINEND);
sb.append(entry.getValue());
sb.append(LINEND);
}
DataOutputStream outStream = new DataOutputStream(conn.getOutputStream());
outStream.write(sb.toString().getBytes());
// 发送文件数据
if (files != null)
for (Map.Entry<String, File> file : files.entrySet()) {
StringBuilder sb1 = new StringBuilder();
sb1.append(PREFIX);
sb1.append(BOUNDARY);
sb1.append(LINEND);
sb1.append("Content-Disposition: form-data; name=\"uploadfile\"; filename=\""
+ file.getValue().getName() + "\"" + LINEND);
sb1.append("Content-Type: application/octet-stream; charset=" + CHARSET + LINEND);
sb1.append(LINEND);
outStream.write(sb1.toString().getBytes());
InputStream is = new FileInputStream(file.getValue());
byte[] buffer = new byte[1024];
int len = 0;
while ((len = is.read(buffer)) != -1) {
outStream.write(buffer, 0, len);
}
is.close();
outStream.write(LINEND.getBytes());
}
byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINEND).getBytes();
outStream.write(end_data);
outStream.flush();
int res = conn.getResponseCode();
InputStream in = conn.getInputStream();
StringBuilder sb2 = new StringBuilder();
if (res == 200) {
int ch;
while ((ch = in.read()) != -1) {
sb2.append((char) ch);
}
}
outStream.close();
conn.disconnect();
return sb2.toString();
}
// 测试
public static void main(String[] args) throws IOException {
String requestURL = "sss";
final Map<String, String> params = new HashMap<String, String>();
params.put("send_userId", String.valueOf(1));
params.put("send_email", "ss@ss.com");
final Map<String, File> files = new HashMap<String, File>();
files.put("uploadfile", new File("/var/data/de.jpg"));
final String result = UploadUtil.post(requestURL, params, files);
System.out.println("result is: " + result);
}
}

二、php服务端接收文件,临时保存并继续上传至java后端:

  1. 接收文件类

<?php
namespace App\Controller; use Action\RestAction;
use Api\UploadApi; class UserController extends RestAction
{
/**
* 用户头像上传
*/
public function set_avatar_post($code)
{
$uploadApi = new UploadApi();
$res = $uploadApi->uploads('avatar');
$filename = $res['data']; $result = $uploadApi->uploadAvatar($code, $filename);
@unlink($filename); //删除图片
if (!$result['status']) {
$this->response($result);
}
$avatar = A("Personal", "Api")->getAvatar($code);
$this->response($avatar);
}
}

  2. 上传类

<?php
namespace Api\Action; class UploadApi
{
public function __construct()
{
//...
} public function curlGet($url, $param = array(), $timeout = 30, $ajaxResponseImmediately = true)
{
$opts = array(
CURLOPT_TIMEOUT => $timeout,
CURLOPT_RETURNTRANSFER => 1,
CURLOPT_SSL_VERIFYPEER => false,
CURLOPT_SSL_VERIFYHOST => false,
CURLOPT_HTTPHEADER => $header
);
switch (strtoupper($method)) {
// case 'POST':
// $opts[CURLOPT_URL] = $url;
// $opts[CURLOPT_POST] = 1;
// $opts[CURLOPT_POSTFIELDS] = $param;
// break;
default:
$opts[CURLOPT_URL] = $url . '?' . http_build_query($param);
break;
} $ch = curl_init();
curl_setopt_array($ch, $opts);
$result = curl_exec($ch); //记录请求日志
curl_close($ch);
return $result;
} public function curlPost($url, $param = array(), $timeout = 30, $ajaxResponseImmediately = true)
{
$opts = array(
CURLOPT_TIMEOUT => $timeout,
CURLOPT_RETURNTRANSFER => 1,
CURLOPT_SSL_VERIFYPEER => false,
CURLOPT_SSL_VERIFYHOST => false,
CURLOPT_HTTPHEADER => $header
);
switch (strtoupper($method)) {
case 'POST':
default:
$opts[CURLOPT_URL] = $url;
$opts[CURLOPT_POST] = 1;
$opts[CURLOPT_POSTFIELDS] = $param;
break;
// $opts[CURLOPT_URL] = $url . '?' . http_build_query($param);
// break;
} $ch = curl_init();
curl_setopt_array($ch, $opts);
$result = curl_exec($ch); $log_data['result'] = $result;
if (!empty($param)) $log_data['param'] = $param;
curl_close($ch);
return $result;
} public function uploads($param = '')
{
if ($param == '') {
$param = 'imgFile';
}
// 文件保存目录路径
$save_url = dirname(dirname(dirname(__FILE__))) . DIRECTORY_SEPARATOR . "Uploads" . DIRECTORY_SEPARATOR;
// 定义允许上传的文件扩展名
$ext_arr = array(
'image' => array('gif', 'jpg', 'jpeg', 'png', 'bmp'),
);
// 最大文件大小
$max_size = 20 * 1024;
// PHP上传失败
if (!empty ($_FILES [$param] ['error'])) {
switch ($_FILES [$param] ['error']) {
case '1' :
$error = '超过php.ini允许的大小。';
break;
case '2' :
$error = '超过表单允许的大小。';
break;
case '3' :
$error = '图片只有部分被上传。';
break;
case '4' :
$error = '请选择图片。';
break;
case '6' :
$error = '找不到临时目录。';
break;
case '7' :
$error = '写文件到硬盘出错。';
break;
case '8' :
$error = 'File upload stopped by extension。';
break;
case '999' :
default :
$error = '未知错误。';
}
$result = array('status' => '0', 'error' => '111111', 'msg' => $error); }
// 有上传文件时
if (empty ($_FILES) === false) {
$file_name = $_FILES [$param] ['name'];// 原文件名
$tmp_name = $_FILES [$param] ['tmp_name'];// 服务器上临时文件名
$file_size = $_FILES [$param] ['size'];// 文件大小
// 检查文件名
if (!$file_name) {
$result = array('status' => '0', 'error' => '111111', 'msg' => '请选择文件');
}
// 检查是否已上传
if (@is_uploaded_file($tmp_name) === false) {
$result = array('status' => '0', 'error' => '111111', 'msg' => '上传失败');
}
// 检查文件大小
if ($file_size > $max_size) {
$result = array('status' => '0', 'error' => '111111', 'msg' => '');
}
// 检查目录名
$dir_name = empty ($_GET ['dir']) ? 'image' : trim($_GET ['dir']);
if (empty ($ext_arr [$dir_name])) {
$result = array('status' => '0', 'error' => '111111', 'msg' => '目录名不正确');
}
// 获得文件扩展名
$temp_arr = explode('.', $file_name);
$file_ext = array_pop($temp_arr);
$file_ext = trim($file_ext);
$file_ext = strtolower($file_ext);
// 检查扩展名
if (in_array($file_ext, $ext_arr [$dir_name]) === false) {
$result = array('status' => '0', 'error' => '111111', 'msg' => '上传文件扩展名是不允许的扩展名');
}
// 创建文件夹
if ($dir_name !== '') {
if (!file_exists($save_url)) {
mkdir($save_url);
}
}
$new_file_name = date('YmdHis') . '_' . rand(10000, 99999) . '.' . $file_ext;
$file_path = $save_url . $new_file_name;
if (move_uploaded_file($tmp_name, $file_path) === false) {
$result['msg'] = '上传文件失败';
$result = array('status' => '0', 'error' => '111111', 'msg' => '上传文件失败');
} else {
$result = array('status' => '1', 'error' => '000000', 'data' => $file_path);
}
@chmod($file_path, 0644);
return $result;
}
} public function uploadAvatar($code, $avatarImageName) {
$url = $this->getApiUrl(__METHOD__);
$data = array(
"code" => $code,
"ip" => $this->params['ip'],
"avatar" => !empty($avatarImageName) ? '@' . $avatarImageName : '',
);
$result = $this->curlPost($url, $data);
return $result;
}
}

  这样,php就已经接收到了来自客户端的 图片上传了,并且已经上传到java后端服务器。

  注意:这里有个坑,即php版本大于5.6以后,直接使用 @ 符号无法上传文件了,需要 加上一个安全选项:CURLOPT_SAFE_UPLOAD => false 才可以,或者使用5.6以的高级上传类上传文件:

curl_setopt(ch, CURLOPT_POSTFIELDS, [
'file' => new CURLFile(realpath('image.png')),
]);

三、java后端接收php上传的图片

package com.xx.c.action;

import com.xx.core.pojo.Constants;
import com.xx.core.pojo.MicroException;
import com.xx.core.pojo.ResponseEntity;
import com.xx.core.web.spring.bind.annotation.ClientIP;
import com.xx.core.web.spring.bind.annotation.SessionUserId;
import com.xx.c.pojo.user.UpFileUrlBean;
import com.xx.c.service.user.UserService;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ModelAttribute;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.MultipartHttpServletRequest; import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import java.util.Iterator; @Controller
@RequestMapping(value = "upload")
public class UploadAction { @Resource(name = "userService")
private UserService userService; @RequestMapping(value = "/uploadAvatar", method = RequestMethod.POST, produces = "application/json")
@ResponseBody
public Object uploadAvatar(@RequestParam String code, @ClientIP String addIp, @SessionUserId Long userId,
@ModelAttribute UpFileUrlBean bean, HttpServletRequest request) throws MicroException {
bean.setAddIp(addIp);
bean.setUserId(userId); try {
// 转型为MultipartHttpRequest:
MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
// 从其中取出一个文件 后续可使用spring 上传文件方法:file.transferTo(destFile);
MultipartFile file = null;
for (Iterator<String> it = multipartRequest.getFileNames(); it.hasNext();) {
file = multipartRequest.getFile((String) it.next());
}
userService.uploadAvatar(file, bean);
} catch (Exception e) {
throw new MicroException(Constants.ErrCode.UPLOAD_AVATAR_TO_SERVER_FAILED, Constants.ErrMsg.UPLOAD_AVATAR_TO_SERVER_FAILED, e);
} ResponseEntity ret = new ResponseEntity(Constants.System.OK);
return ret;
} }

  至此,上传流程已经完成了。(当然,后续还可能使用其他上传,比如dubbo调用文件系统上传文件,调用第三方sdk上传到文件服务器。。。, 原理大抵一样,使用字节流进行传输,然后读取出来存储到文件)

  一般为app写的接口中,都会涉及到加解密问题,此时,文件不应该算作加密的范畴,而应单独给一个字段。

从app上传图片到php,再上传到java后端服务器的方法一条龙服务的更多相关文章

  1. 从app上传图片到php,再上传到java后端服务器的方法一览

    在现在的网络开发中,上传图片类的需求实在是太普通不过了,但是对于怎么样做到上传图片,对于刚开始建立项目的时候,还是有点不知所措的.也许有幸,我们做的项目是之前已经有人写过类似的用例了,那么我们只需要依 ...

  2. 将本地文件上传到指定的服务器(HttpWebRequest方法)

    将本地文件上传到指定的服务器(HttpWebRequest方法),通过文件流,带文件名,同文件一同上传的表单文本域及值. ///<summary> /// 将本地文件上传到指定的服务器(H ...

  3. C# 向服务器上传文件(客服端winform、服务端web)

    转载 首先写客服端,winform模拟一个post提交: /// <summary> /// 将本地文件上传到指定的服务器(HttpWebRequest方法) /// </summa ...

  4. ASP.NET简单实现APP中用户个人头像上传和裁剪

    最近有个微信项目的用户个人中心模块中,客户要求用户头像不仅仅只是上传图片,还需要能对图片进行裁剪.考虑到flash在IOS和Android上的兼容性问题,于是想着能从js这块入手,在网上发现了devo ...

  5. ueditor1.3.6jsp版在struts2应用中上传图片报"未找到上传文件"解决方案

    摘要: ueditor1.3.6jsp版在struts2应用中上传图片报"未找到上传文件"解决方案 在struts2应用中使用ueditor富文本编辑器上传图片或者附件时,即使配置 ...

  6. asp.net core 通过ajax上传图片及wangEditor图片上传

    asp.net core 通过ajax上传图片 .net core前端代码,因为是通过ajax调用,首先要保证ajax能调用后台代码,具体参见上一篇.net core 使用ajax调用后台代码. 前端 ...

  7. 图片上传oss--先拿server端签名再上传oss,返回id值

    目前项目oss阿里云存储图片,图片上传主要步骤是:前端从服务端拿到签名signature,再上传到oss上busket里,上传成功返回图片id (imgId),最后再给server端: 注:官网上有个 ...

  8. 微信多媒体上传图片,创建卡券上传 LOGO

    //*****************************************多媒体上传图片 begin******************************************** ...

  9. [Swift通天遁地]四、网络和线程-(9)上传图片并实时显示上传进度

    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...

随机推荐

  1. jQuery formValidator API

    jQuery formValidator插件的API帮助 目前支持5种大的校验方式,分别是:inputValidator(针对input.textarea.select控件的字符长度.值范围.选择个数 ...

  2. JVM系列3:类加载机制

    了解类加载机制也是深入了解Java的重要一环,它包括加载过程.类加载器.加载机制等内容. 以下是我总结的思维导图. 首先讲讲类加载的时机,以下是会触发类加载的时机: 1.new.get/put/inv ...

  3. Github(远程仓库) 2

    远程仓库之前就添加好了 今天弄了简单的查看远程库,提取远程库,在线修改以及本地更新修改,推送到远程仓库,删除远程仓库,参考http://www.runoob.com/git/git-remote-re ...

  4. easyui - 标签属性顺序要对 否则options 错误

    标签属性顺序要对 否则options 错误

  5. Windows 10同步时间的方法

    今天在安装了Windows 10 1809(October 2018 update)之后发现时间不能同步,以前并没有出现这种情况. 1) 打开控制面板,找到时钟域地区 2) 选择日期和时间 3) 选择 ...

  6. mysql_day03

    MySQL-Day02回顾1.表记录的管理 1.删除表记录 1.delete from 表名 where 条件; ## 不加where条件全部删除 2.更新表记录 1.update 表名 set 字段 ...

  7. mybatis动态排序

    如果我们要传入排序字段作为一个参数到mybatis中,用以实现按照指定字段来排序的功能,那么我们需要使用$,而不是像其他参数一样,使用#.如下所示. <if test="sortnam ...

  8. 加NONCLUSTERED INDEX索引,在ON了之后还要INCLUDE

    之前加了索引,但效果不大 SET STATISTICS TIME ON --执行时间 SET STATISTICS IO ON --IO读取 DBCC DROPCLEANBUFFERS --清除缓冲区 ...

  9. php设计模式-工厂模式(一)

    <?php abstract class Creator{ /* startFactory 返回一个具体的产品 factoryMethod 返回对象 */ protected abstract ...

  10. React-router4 简单总结

    官方文档读到这里,大概明白了React-router是专门为单页面设计的,,我只能说多页面格外的不方便 首先这个是基本的套路 import React from 'react' import Reac ...