最近帮学长做一个需求,总结一下在实现过程中遇到的问题,以及一些新知识的学习。

  1. 点击上传文件,并展示。
  2. 点击按钮进行测试,执行Python文件,并将指定生成文件夹中的图片展示在页面上

1. 文件的上传预览

方法一:FileReader

参考文档:MDN FileReader

<body>
  <!-- multiple 属性规定输入字段可选择多个值。如果使用该属性,则字段可接受多个值。 -->
  <!-- webkitdirectory 属性可以以文件夹形式上传文件 -->
  <input type="file" id="selectFiles" onchange="dealSelectFiles()" multiple webkitdirectory>
  <div id="showImg"></div>

  <script type="text/javascript">
    function dealSelectFiles() {
      let selectFiles = document.getElementById("selectFiles").files; //获得一个对象,是上传的文件信息(可以是多个)

      for (let file of selectFiles) {
        // console.log(file.webkitRelativePath);//包含文件路径,相对于用户所选的祖先目录。调用webkitRelativePath返回的结果都不会是你希望得到的路径

        let reader = new FileReader(); //返回一个新构造的FileReader。
        reader.readAsDataURL(file); // 开始读取指定的Blob中的内容。一旦完成,result属性中将包含一个data: URL格式的Base64字符串以表示所读取文件的内容。

        reader.onloadend = function() { // 在读取结束的时候触发
          let img = new Image();
          img.src = this.result; //读取完成后result属性中将包含一个data: URL格式的Base64字符串以表示所读取文件的内容。
          img.onload = function() { // 在图片加载完成后立即执行
            let showImg = document.getElementById("showImg");
            showImg.appendChild(img)
          }
        }
      }
    }
  </script>
</body>
方法二:URL.createObjectURL()

参考文档:MDN URL.createObjectURL()

<script type="text/javascript">
    function dealSelectFiles() {
      let selectFiles = document.getElementById("selectFiles").files; //获得一个对象,是上传的文件信息(可以是多个)

      for (let file of selectFiles) {
        let img = new Image();
        let url = window.URL.createObjectURL(file); //获取到获取图片的地址,但是是经过浏览器处理不是真实的地址
        img.src = url;
        img.onload = function() { // 在图片加载完成后立即执行
          let showImg = document.getElementById("showImg");
          showImg.appendChild(img);
          window.URL.revokeObjectURL(file);
        }
      }
    }
  </script>

2. 将上传的文件传给后台处理(我这里使用node.js+express框架)

A. 实现将上传的图片放到固定的文件夹中存储

实现这个功能我们需要使用:

1. fs模块

npm install --save fs

2. multer模块

github中multer中文文档

npm install --save multer

  1. Multer 会添加一个 body 对象 以及 filefiles 对象 到 express 的 request 对象中。
  2. body 对象包含表单的文本域信息
  3. file 或 files 对象包含 对象表单上传的文件信息。

以下代码实现了将上传的图片放到固定的文件夹中存储

let express = require('express')
let { join } = require('path')
let multer = require('multer')
let fs = require('fs');

let app = express();
app.use(express.static(join(__dirname, 'public')))

// 1. 打开或创建一个名字叫folder文件夹
let createFolder = function(folder) {
  try {
    fs.accessSync(folder); // 打开文件夹
  } catch (e) {
    fs.mkdirSync(folder); // 创建文件夹
  } 
};
let uploadFolder = './upload/'; // 设定存储文件夹为当前目录下的 /upload 文件夹
createFolder(uploadFolder);

// 2. 设置磁盘存贮
let storage = multer.diskStorage({
  destination: function(req, file, cb) {
    cb(null, uploadFolder); // 他会放在当前目录下的 /upload 文件夹下(没有该文件夹,就新建一个)
  },
  filename: function(req, file, cb) { // 在这里设定文件名
    cb(null, file.originalname); // file.originalname是将文件名设置为上传时的文件名,file中携带的
    // cb(null, Date.now() + '-' + file.originalname) // 加上Date.now()可以避免命名重复
  }
})

let upload = multer({ storage: storage });

app.post('/upload', upload.single('upload'), function(req, res, next) {
  res.send("success");
  next();
});

B. 在发送请求后执行Python文件

方法一: 使用了jsexecpy模块执行Python文件

app.get('/test', (req, res) => {
		let jsexecpy = require("jsexecpy");
		jsexecpy.runpath("./public/python/star.py");
	})

但是需要传递参数,并且还要执行cmd命令,所以又有了

方法二: 使用child_process(执行cmd命令)

child_process官方文档

app.get('/test', (req, res) => {
	  let cmd = 'python D:/DeskTop/star.py --source';
	  // Sync同步执行
	  process.execSync(cmd, function(error, stdout, stderr) {
	    console.log("error:" + error);
	    console.log("stdout:" + stdout);
	    console.log("stderr:" + stderr);
	  });
	})

C: 获取某一个文件夹下的所有的图片
需要用到的模块

let fs = require('fs'); //引用文件系统模块
let image = require("imageinfo"); //引用imageinfo模块
const { join } = require('path');

递归获取文件夹下的文件(封装成函数,并开放出去)

// 定义遍历目录的方法
function readFileList(path, filesList) {
  var files = fs.readdirSync(path);
  files.forEach(function(item, index) {
    // 获取文件信息
    var stat = fs.statSync(path + item); //用于递归
    // 是否为目录(isDirectory)
    if (stat.isDirectory()) { //递归读取文件
      readFileList(path + item + "/", filesList)
    } else {
      var obj = {}; //定义一个对象存放文件的路径和名字
      obj.path = path; //路径
      obj.filename = item //名字
      filesList.push(obj);
    }
  })
}

module.exports = getFiles = {
  //获取文件夹下的所有文件
  getFileList: function(path) {
    var filesList = [];
    readFileList(path, filesList);
    return filesList;
  },
  //获取文件夹下的所有图片的路径
  getImageFiles: function(path) {
    var imageList = [];

    this.getFileList(path).forEach((item) => {
      var ms = image(fs.readFileSync(item.path + item.filename));
      // console.log(ms);
      ms.mimeType && (imageList.push(join(item.path, item.filename)))//拼接上public路径
    });
    return imageList;
  }
};

// 获取文件夹下的所有图片的
// console.log(getFiles.getImageFiles("./public/images/"));

以上能够获取public文件夹下图片的名称,并且拼接上了public路径,通过public将其开放出去并在页面上展示了。

总结:在这次的的案例实现过程中,看似很简单的操作,实际敲起代码来却并不顺利,对于node基础知识掌握的还不是很牢固,还需要不断的复习与使用。这两天也学到许多新知识,希望在技术的路上越走越深。