php + js 多个商品照片的上传处理?

发布于 2022-09-11 15:26:20 字数 744 浏览 13 评论 0

先说我目前的做法
是在前台新增商品照片时可以多个上传
每个图片都会有一个 hidden input,里面放的是 base64

<input type="hidden" name="icon[]" value="...">

然后在后端 php 用 file_put_contents 将其下载到指定目录

foreach ($_POST['icon'] as $value) {
    preg_match('/^(data:\s*image\/(\w+);base64,)/', $value, $result);
    $typeThis = '.' . $result[2];
          
    file_put_contents('../../images/product/cover-XXX'.$typeThis, base64_decode(str_replace($result[1], '', $value)));

只是这个字串长度真的让人呵呵
然后如果上传十几MB的图片,就直接不给过(应该是base64字串太长的关系,好像跟SIZE有关)
有没有其他的方式有一样的效果,但更乾淨的做法,可以不用那么长,上传多大的图片都可以?可以用同样的方式在后端抓取?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

还不是爱你 2022-09-18 15:26:20

首先谢邀,
对后端不够熟悉,我给一些偏前端的点吧:

  1. 在文件存储的技术选型上,优先推荐使用对象存储,各家云服务平台应该都有这个产品线。对象存储的好处,其一它存储是以文件为单位的,不依赖具体的服务器或路径(当然一些分类或处理用的路径还是可以自己新建的),这样你做负载均衡时也可以共享着用,而且如果突然有一天想单独管理,像我用过的阿里云OSS,有单独的客户端可以用,登陆上去完全可视化,和操作Windows差不太多;其二它不怕容量撑爆,我记得几年前做过个营销类型的页面,小游戏,需要用户传头像上去,上线当天直接把硬盘堆满了,整台临时下线去扩容……用对象存储的话,Money给够一切好说。。?
  2. 当然如果说服务器空间管够+单主机,选择直接存到主机上也没什么不对,但是非常不建议用Base64,因为用文件存是因为文件是二进制的,转成Base64后,二进制变成ASCII码,体积起码膨胀3倍(字符串占用二进制位数),不做压缩的话,这个数字非常之恐怖,进而会直接导致用户的图片卡在前端传不过来或者页面上的资源没法显示(其实就算传过来了,几十兆字符串存进数据库貌似也不太合适……当然我说的基础型的),所以不要为了表面上“能转成字符串存数据库”就贸然用这个方案,除非你的图片都是平均几k的小东西。
  3. 前端做法我个人建议压缩一下再传给后台。这里压缩的目的一是直观控制图片大小,二是做规格化预处理,把EXIF之类的东西先清一清,需要高质量的话可以输出原大小的PNG。当然如果用对象存储的话,可能还有机会用到一些附加工具在后台做二次处理。
  4. 不建议后端直接下载的方式。从用户体验角度来说,还是建议给用户一个更直观的反馈,同时也是提醒用户需要“主动”去完成操作,这样用户会按照习惯把图片上传完,再去做别的。如果是放在后台静默加载,没加载完用户关闭了页面,那前边不就白传了么?
燕归巢 2022-09-18 15:26:20

从前端上传图片到后端,然后php进行处理,完全不需要传base64的字符串,base64可以在上传到后端前,给用户预览,减少不必要的上传,等用户真正确认后,才开始上传到服务器。

这是目前我自己博客后台用的一个简单的图片上传功能,可以供你参考一下:

前端,我这里用到了jQuery

$('.upfile').on('change', function(){ // 点击input[type=file]控件选择图片
    if( !$(this).val() ){
        return false;
    }
    var formData = new FormData(); // 创建一个formData对象,把图片封装进去
    formData.append('file', $(this)[0].files[0]);

    $.ajax({ // 提交数据
        url: '/admin/upimg',
        type: 'POST',
        cache: false,
        data: formData,
        processData: false,
        contentType: false
    }).done(function(res) {
        // console.log(res);
        // console.log(typeof res);
        // console.log(res.code==0);
        // console.log(res.data.url);
        res = JSON.parse(res);
        if( res.code==0 ){
            $('.showpath').text( 'success:   '+res.data.url );
        }else{

        }
    }).fail(function(res) {});
})

后端使用$_FILES获取到相关的图片列表

$file_name = iconv("UTF-8","gb2312", $_FILES['file']['name']); //文件名称
$filenames= explode(".",$file_name);

$file = $_FILES['file'];
$tempfile = $file['tmp_name'];
$targetFolder = '/upload/' .time().rand(1000, 9999).".".$filenames[count($filenames)-1]; // 重命名
$targetPath = $_SERVER['DOCUMENT_ROOT'] . $targetFolder; //图片存放目录
$targetFile = rtrim($targetPath,'/'); //图片完整路徑

// echo $targetFile;

// Validate the file type
$fileTypes = array('jpg', 'jpeg', 'png', 'gif', 'svg'); // File extensions
$fileParts = pathinfo($_FILES['file']['name']);
// print_r( $file );

// 判断当前上传的文件是否为图片格式
if (in_array($fileParts['extension'],$fileTypes)) {
    $s = move_uploaded_file($tempfile, $targetFile); // 移动到服务器相关的目录
}
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文