>[danger] 先來看最基本的文件上傳表單,以及服務器接收到的數據結構
a.php
~~~
<?php
if ($_POST) {
echo '<pre>';
print_r($_POST);
echo '</pre>';
echo '<hr>';
echo '<pre>';
print_r($_FILES);
echo '</pre>';
exit;
}
?>
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<meta http-equiv="Content-Language" content="zh-cn">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<meta name="keywords" content="keywords">
<meta name="description" content="description">
<link href="favicon.ico" rel="icon" type="image/x-icon">
<link href="" type="text/css" rel="stylesheet">
<script src=""></script>
<title>Document</title>
</head>
<body>
<form action="" method="post" enctype="multipart/form-data">
<input type="text" name="val" value="val"><br />
<input type="file" name="logo"><br />
<input type="file" name="pic"><br />
<input type="file" name="list[]"><br />
<input type="file" name="list[]"><br />
<input type="file" name="img-list[]"><br />
<input type="file" name="img-list[]"><br /><br />
<!-- 如果有name有值,則就會有一個$_POST[s] = 提交的值 -->
<input type="submit" name="s" value="提交"><br />
<!-- 沒有提交效果 -->
<input type="button" name="b" value="提交按鈕"><br />
<!-- 在form中有提交效果,type="submit" 也有提交效果,但是 type="button" 就沒有提交效果了 (注意所有的提交按鈕必須在form中才能提交除非用js控制) -->
<button>按鈕</button>
</form>
</body>
</html>
~~~

點擊 提交 :
~~~
Array
(
[val] => val
[s] => 提交
)
Array
(
[logo] => Array
(
[name] => QQ圖片20161106223133.jpg
[type] => image/jpeg
[tmp_name] => D:\wamp64\tmp\php92D0.tmp
[error] => 0
[size] => 217991
)
[pic] => Array
(
[name] => QQ圖片20161106223133.jpg
[type] => image/jpeg
[tmp_name] => D:\wamp64\tmp\php931F.tmp
[error] => 0
[size] => 217991
)
[list] => Array
(
[name] => Array
(
[0] => QQ圖片20161106223133.jpg
[1] => fdc20716fe1372b0859ec0019d7e6194.jpg
)
[type] => Array
(
[0] => image/jpeg
[1] => image/jpeg
)
[tmp_name] => Array
(
[0] => D:\wamp64\tmp\php932F.tmp
[1] => D:\wamp64\tmp\php9350.tmp
)
[error] => Array
(
[0] => 0
[1] => 0
)
[size] => Array
(
[0] => 217991
[1] => 6768
)
)
[img-list] => Array
(
[name] => Array
(
[0] => QQ圖片20161106223133.jpg
[1] => bg-01.jpg
)
[type] => Array
(
[0] => image/jpeg
[1] => image/jpeg
)
[tmp_name] => Array
(
[0] => D:\wamp64\tmp\php9351.tmp
[1] => D:\wamp64\tmp\php9361.tmp
)
[error] => Array
(
[0] => 0
[1] => 0
)
[size] => Array
(
[0] => 217991
[1] => 471588
)
)
)
~~~
**上面就是所有最基本的表單上傳,最基本的上傳數據結構了。**
文件上傳根據類型,可分為兩種:
- **單文件上傳**
- 單一字段
- **多文件上傳**
- 多字段的形式多文件
- 單一字段數組(img[])的形式多文件上傳**(這種最復雜,注意這種形式的數據結構)**
- 組合多文件(上面幾種情況的組合)
參考:
[修復Request->file()判斷臨時文件不嚴謹導致多文件上傳失敗的情況](https://github.com/top-think/framework/pull/348)
[文件上傳那些事兒](https://mp.weixin.qq.com/s/ni_5rZDCOypxtHrVXq1Aww)
### 擴展
~~~html
<input type="file" multiple name="pic[]">
~~~
`multiple`是HTML5新增的屬性,這個屬性可以使上傳表單上傳時選擇多個文件,點擊“選擇文件”,在彈出系統選擇文件窗口中按住`Ctrl`鍵可以連續選擇多個文件,**但是此屬性必須配合pic[]數組字段的形式才有作用(注意有些上傳插件比如uploadify,這些可以選擇多文件上傳,此時可能也有這個問題,注意檢查上傳字段名的設置)**,否則即使選擇多個文件,提交到服務器也只有最后一個選擇的文件。(所以這個屬性可以用來避免以前上傳多文件需要多表單,僅一個表單就搞定了)
* * * * *
### 其他
對于前端來說,圖片上傳有以下知識點:
- 普通上傳
- ajax無刷新文件上傳
- 大文件分塊上傳
- MD5提前校驗去重,存在不上傳(傳到服務端再校驗用處不大,只是節省空間,但是并沒有節省不必要上傳的流量)
- 圖像壓縮上傳
- 圖像裁剪(放大,縮小,旋轉等手勢操作)
這些操作以前只有flash才能全部做到,現在html5的新API也能夠做到了,不過大部分接口對瀏覽器的版本要求比較高。
服務端收到文件的方式無非兩種:
1. 普通的$\_FIELS
2. base64編碼的字符圖片$\_POST(非常規Coentent-Type時用`file_get_contents('php://input', 'r')`)
**參考:**
[php://input - 簡單--生活 - 博客園](http://www.cnblogs.com/xiangxiaodong/archive/2012/11/07/2758685.html)
[PHP: php:// - Manual](http://php.net/manual/zh/wrappers.php.php)
* * * * *
[Http 歷險記(上)](https://mp.weixin.qq.com/s?__biz=MzAxOTc0NzExNg==&mid=2665513069&idx=1&sn=548c497c46c7c076145064a120c7c101&scene=21#wechat_redirect)
[Http歷險記(下)-- Struts的秘密](https://mp.weixin.qq.com/s?__biz=MzAxOTc0NzExNg==&mid=2665513080&idx=1&sn=d24a4cdfc71412c581393d584fd91326&scene=21#wechat_redirect)
* * * * *
last update:2017-8-17 09:45:38
- 開始
- 公益
- 更好的使用看云
- 推薦書單
- 優秀資源整理
- 技術文章寫作規范
- SublimeText - 編碼利器
- PSR-0/PSR-4命名標準
- php的多進程實驗分析
- 高級PHP
- 進程
- 信號
- 事件
- IO模型
- 同步、異步
- socket
- Swoole
- PHP擴展
- Composer
- easyswoole
- php多線程
- 守護程序
- 文件鎖
- s-socket
- aphp
- 隊列&并發
- 隊列
- 講個故事
- 如何最大效率的問題
- 訪問式的web服務(一)
- 訪問式的web服務(二)
- 請求
- 瀏覽器訪問阻塞問題
- Swoole
- 你必須理解的計算機核心概念 - 碼農翻身
- CPU阿甘 - 碼農翻身
- 異步通知,那我要怎么通知你啊?
- 實時操作系統
- 深入實時 Linux
- Redis 實現隊列
- redis與隊列
- 定時-時鐘-阻塞
- 計算機的生命
- 多進程/多線程
- 進程通信
- 拜占庭將軍問題深入探討
- JAVA CAS原理深度分析
- 隊列的思考
- 走進并發的世界
- 鎖
- 事務筆記
- 并發問題帶來的后果
- 為什么說樂觀鎖是安全的
- 內存鎖與內存事務 - 劉小兵2014
- 加鎖還是不加鎖,這是一個問題 - 碼農翻身
- 編程世界的那把鎖 - 碼農翻身
- 如何保證萬無一失
- 傳統事務與柔性事務
- 大白話搞懂什么是同步/異步/阻塞/非阻塞
- redis實現鎖
- 淺談mysql事務
- PHP異常
- php錯誤
- 文件加載
- 路由與偽靜態
- URL模式之分析
- 字符串處理
- 正則表達式
- 數組合并與+
- 文件上傳
- 常用驗證與過濾
- 記錄
- 趣圖
- foreach需要注意的問題
- Discuz!筆記
- 程序設計思維
- 抽象與具體
- 配置
- 關于如何學習的思考
- 編程思維
- 談編程
- 如何安全的修改對象
- 臨時
- 臨時筆記
- 透過問題看本質
- 程序后門
- 邊界檢查
- session
- 安全
- 王垠
- 第三方數據接口
- 驗證碼問題
- 還是少不了虛擬機
- 程序員如何談戀愛
- 程序員為什么要一直改BUG,為什么不能一次性把代碼寫好?
- 碎碎念
- 算法
- 實用代碼
- 相對私密與絕對私密
- 學習目標
- 隨記
- 編程小知識
- foo
- 落盤
- URL編碼的思考
- 字符編碼
- Elasticsearch
- TCP-IP協議
- 碎碎念2
- Grafana
- EFK、ELK
- RPC
- 依賴注入
- 科目一
- 開發筆記
- 經緯度格式轉換
- php時區問題
- 解決本地開發時調用遠程AIP跨域問題
- 后期靜態綁定
- 談tp的跳轉提示頁面
- 無限分類問題
- 生成微縮圖
- MVC名詞
- MVC架構
- 也許模塊不是唯一的答案
- 哈希算法
- 開發后臺
- 軟件設計架構
- mysql表字段設計
- 上傳表如何設計
- 二開心得
- awesomes-tables
- 安全的代碼部署
- 微信開發筆記
- 賬戶授權相關
- 小程序獲取是否關注其公眾號
- 支付相關
- 提交訂單
- 微信支付筆記
- 支付接口筆記
- 支付中心開發
- 下單與支付
- 支付流程設計
- 訂單與支付設計
- 敏感操作驗證
- 排序設計
- 代碼的運行環境
- 搜索關鍵字的顯示處理
- 接口異步更新ip信息
- 圖片處理
- 項目搭建
- 閱讀文檔的新方式
- mysql_insert_id并發問題思考
- 行鎖注意事項
- 細節注意
- 如何處理用戶的輸入
- 不可見的字符
- 抽獎
- 時間處理
- 應用開發實戰
- python 學習記錄
- Scrapy 教程
- Playwright 教程
- stealth.min.js
- Selenium 教程
- requests 教程
- pyautogui 教程
- Flask 教程
- PyInstaller 教程
- 蜘蛛
- python 文檔相似度驗證
- thinkphp5.0數據庫與模型的研究
- workerman進程管理
- workerman網絡分析
- java學習記錄
- docker
- 筆記
- kubernetes
- Kubernetes
- PaddlePaddle
- composer
- oneinstack
- 人工智能 AI
- 京東
- pc_detailpage_wareBusiness
- doc
- 電商網站設計
- iwebshop
- 商品規格分析
- 商品屬性分析
- tpshop
- 商品規格分析
- 商品屬性分析
- 電商表設計
- 設計記錄
- 優惠券
- 生成唯一訂單號
- 購物車技術
- 分類與類型
- 微信登錄與綁定
- 京東到家庫存系統架構設計
- crmeb
- 命名規范
- Nginx https配置
- 關于人工智能
- 從人的思考方式到二叉樹
- 架構
- 今日有感
- 文章保存
- 安全背后: 瀏覽器是如何校驗證書的
- 避不開的分布式事務
- devops自動化運維、部署、測試的最后一公里 —— ApiFox 云時代的接口管理工具
- 找到自己今生要做的事
- 自動化生活
- 開源與漿果
- Apifox: API 接口自動化測試指南