溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何使用koa2完成Excel導(dǎo)入導(dǎo)出

發(fā)布時(shí)間:2023-03-31 11:09:27 來源:億速云 閱讀:110 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹了如何使用koa2完成Excel導(dǎo)入導(dǎo)出的相關(guān)知識(shí),內(nèi)容詳細(xì)易懂,操作簡(jiǎn)單快捷,具有一定借鑒價(jià)值,相信大家閱讀完這篇如何使用koa2完成Excel導(dǎo)入導(dǎo)出文章都會(huì)有所收獲,下面我們一起來看看吧。

一、安裝 node-xlsx

npm i node-xlsx

二、導(dǎo)出excel表

首先是查詢出數(shù)據(jù)庫(kù)內(nèi)所有的表的信息,然后傳至下一個(gè)下劃線轉(zhuǎn)大小寫的公用函數(shù),進(jìn)行key的轉(zhuǎn)換,然后就進(jìn)行數(shù)據(jù)的加工;

大小寫轉(zhuǎn)換函數(shù)封裝可看此篇文章

因?yàn)閷?dǎo)出except表的宮功能會(huì)在業(yè)務(wù)中頻繁使用,所以我們需要給它封裝成一個(gè)公用函數(shù),其他的業(yè)務(wù)使用直接傳參調(diào)用即可!

1、封裝公用函數(shù)

/**
 * excel 導(dǎo)出
 * list:[{}]
 * headers:表頭中文名
 * headerKeys:與表頭中文名一一對(duì)應(yīng)的數(shù)據(jù)區(qū)key
 * tableName:導(dǎo)出的表名稱以什么開頭
 */
export const excelExport = (list, headers, headerKeys, tableName = 'excel') => {
  // excel 通用樣式
  const sheetOptions = { '!cols': [] }
  headers.forEach(() => {
    sheetOptions['!cols'].push({
      wch: 20
    })
  })
  const data = []
  list.forEach((item) => {
    let arr = []
    const item2 = flatten(item)
    headerKeys.forEach((key) => {
      if (excelMap.changDictExport[key]) {
        arr.push(excelMap.changDictExport[key][item[key]])
      } else {
        arr.push(item2[key])
      }
    })
    data.push(arr)
  })
  data.unshift(headers)
  const buffer = xlsx.build(
    [{ options: {}, name: `${tableName}_${new Date().valueOf()}`, data: data }],
    { sheetOptions }
  )
  return buffer
}

2、headers例子:

  userHeader: [
    '用戶序號(hào)',
    '登錄名稱',
    '用戶郵箱',
    '手機(jī)號(hào)碼',
    '用戶性別',
    '帳號(hào)狀態(tài)',
    '最后登錄IP',
    '最后登錄時(shí)間',
    '部門名稱',
    '部門負(fù)責(zé)人'
  ],

3、headerKeys例子:

  userHeaderKeys: [
    'userId',
    'userName',
    'email',
    'phonenumber',
    'sex',
    'status',
    'loginIp',
    'loginDate',
    'dept.deptName',
    'dept.leader'
  ]

4、因?yàn)閿?shù)據(jù)可能存在跨表查詢,會(huì)出現(xiàn)對(duì)象嵌套結(jié)構(gòu),故需要封裝對(duì)象扁平化函數(shù)flatten,將多層結(jié)構(gòu)的key變成dept.deptName這種格式

/**
 * 對(duì)象扁平化
 */
export const flatten = (obj) => {
  let result = {}
  let process = (key, value) => {
    // 首先判斷是基礎(chǔ)數(shù)據(jù)類型還是引用數(shù)據(jù)類型
    if (Object(value) !== value) {
      // 基礎(chǔ)數(shù)據(jù)類型
      if (key) {
        result[key] = value
      }
    } else if (Array.isArray(value)) {
      for (let i = 0; i < value.length; i++) {
        process(`${key}[${i}]`, value[i])
      }
      if (value.length === 0) {
        result[key] = []
      }
    } else {
      let objArr = Object.keys(value)
      objArr.forEach((item) => {
        process(key ? `${key}.${item}` : `${item}`, value[item])
      })
      if (objArr.length === 0 && key) {
        result[key] = {}
      }
    }
  }
  process('', obj)
  return result
}

5、node-xlsx接收的數(shù)據(jù)格式

 [
  [
    '用戶序號(hào)',
    '登錄名稱',
    '用戶郵箱',
    '手機(jī)號(hào)碼',
    '用戶性別',
    '帳號(hào)狀態(tài)',
    '最后登錄IP',
    '最后登錄時(shí)間',
    '部門名稱',
    '部門負(fù)責(zé)人'
  ],
  [
    1,
    'admin',
    '12012311715@163.com',
    '12012311715',
    '男',
    '正常',
    '',
    '00:00:00',
    '深圳總公司',
    'wen'
  ],
  [
    2,          'password',
    null,       null,
    '未知',     '正常',
    null,       null,
    '研發(fā)部門', 'wen'
  ]
 ]

6、在業(yè)務(wù)函數(shù)中調(diào)用導(dǎo)出excel表的數(shù)據(jù)

 const buffer = excelExport(
      users,
      excelExportMap.userHeader,
      excelExportMap.userHeaderKeys,
      'user'
    )

7、最終excel導(dǎo)出的效果

如何使用koa2完成Excel導(dǎo)入導(dǎo)出

三、導(dǎo)入excel表

因?yàn)閷?dǎo)入的情況比較復(fù)雜,會(huì)分為多種情況上傳excel文件:

1、單文件單工作表;

2、單文件多工作表;

3、多文件(單)多工作表;

我個(gè)人解決辦法是獲取放置excel文件的文件夾內(nèi)所有的excel文件,然后進(jìn)行數(shù)據(jù)的提取,在提取完數(shù)據(jù)后,就將該次的excel文件刪除掉,當(dāng)然導(dǎo)入excel功能也是需要進(jìn)行公用封裝的;

1、解析本次導(dǎo)入的所有excel文件內(nèi)數(shù)據(jù)

// 導(dǎo)入excel--解析
export const importExcelsMid = (tableMap: string) => {
  return async (ctx: Context, next: () => Promise<void>) => {
    try {
      const fileExistPath = path.resolve() + '\src\upload'
      let fileName = [] // 多個(gè)excel文件保存地
      fs.readdirSync(path.format({ dir: fileExistPath })).forEach((excel) => {
        if (excel.split('.')[excel.split('.').length - 1] === 'xlsx' && 'xls') {
          fileName.push(excel)
        }
      })
      // 拿去多個(gè)excel文件
      const workSheetsFromBuffer = []
      fileName.forEach((item) => {
        const absoluteFilePath = fileExistPath + '\' + item //整個(gè)文件的絕對(duì)路徑
        workSheetsFromBuffer.push(xlsx.parse(fs.readFileSync(absoluteFilePath))) //這種方式是解析buffer
      })
      // 生成默認(rèn)用戶密碼
      const salt = bcrypt.genSaltSync(10)
      const hash = bcrypt.hashSync('123456', salt)
      const arr = [] // 存儲(chǔ)sql批量創(chuàng)建的信息 object[]
      workSheetsFromBuffer.forEach((element) => {
        element.forEach((item: any) => {
          // 此層是遍歷表數(shù)量(單表數(shù)據(jù)提取)
          const data = item.data
          for (let j = 1; j < data.length; j++) {
            // 此層是加入每行數(shù)據(jù)
            const obj = {
              password: hash
            }
            for (let i = 0; i < data[0].length; i++) {
              let key = excelMap[tableMap][data[0][i]]
              if (excelMap.changDict[key]) {
                obj[key] = excelMap.changDict[key][data[j][i]]
              } else {
                obj[key] = data[j][i]
              }
            }
            arr.push(obj)
          }
        })
      })
      // 獲取數(shù)據(jù)后刪除excel文件
      fileName.forEach((path) => {
        removeSpecifyFile(path)
      })
      ctx.state.excelData = arr
    } catch (error) {
      console.error('用戶excel上傳表頭格式不正確!', ctx.request['body'])
      return ctx.app.emit('error', importUserListErr, ctx)
    }
    await next()
  }
}

2、寫入本次導(dǎo)入的所有excel文件內(nèi)數(shù)據(jù)

updates:是控制你更新哪些的key數(shù)組

 [    'dept_id',    'user_name',    'nick_name',    'email',    'phonenumber',    'sex',    'status'  ]

updates案例 ?

// 導(dǎo)入excel--新增修改sql
export const judegImportMid = (table, updates) => {
  return async (ctx: Context, next: () => Promise<void>) => {
    const { updateSupport } = ctx.query
    try {
      if (updateSupport === '1') {
        // 新增 且 修改
        await table.bulkCreate(ctx.state.excelData, {
          updateOnDuplicate: updates
        })
      } else {
        // 不更改 只新增
        await table.bulkCreate(ctx.state.excelData)
      }
      ctx.body = {
        code: 200,
        message: '用戶信息上傳成功!'
      }
    } catch (error) {
      console.error('user excel新增與修改錯(cuò)誤', ctx.request['body'])
      return ctx.app.emit('error', { code: '400', message: error.errors[0].message }, ctx)
    }
  }
}

3、導(dǎo)入的案例excel

如何使用koa2完成Excel導(dǎo)入導(dǎo)出

4、解析后的數(shù)據(jù)

[
  {
    password: '$2a$10$Mp19aHpTTIZXwAYpwAg8QuOUQ6DmBswHFhwR8iRqjduNw9tQU.xRO',
    undefined: 'test',
    user_name: 'test',
    email: 'test',
    phonenumber: 'test',
    sex: '0',
    status: '0'
  }
]

5、已寫入數(shù)據(jù)庫(kù)

如何使用koa2完成Excel導(dǎo)入導(dǎo)出

關(guān)于“如何使用koa2完成Excel導(dǎo)入導(dǎo)出”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對(duì)“如何使用koa2完成Excel導(dǎo)入導(dǎo)出”知識(shí)都有一定的了解,大家如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI