SpringBoot 檔案上傳與下載

<!DOCTYPE html>
<html lang="zh" xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>文件上传和下载</title>
</head>
<body>
<form th:action="@{/upload}" method="post" enctype="multipart/form-data">
<p>
<label for="file1" >请选择要上传的文件</label>
<input type="file" id="file1" name="file" multiple/>
</p>
<p>
<input type="submit" value="上传">
</p>
</form>
<ul>
<li th:each="file : ${files}">
<a th:href="@{/download(fileName=${file})}" th:text="${file}"></a>
</li>
</ul>
</body>
</html>
 package com.sun.ch04.controller;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.MultipartHttpServletRequest;

import javax.servlet.http.HttpServletRequest;
import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.List;
import org.apache.commons.io.FileUtils;

@Controller
public class FileController {
@GetMapping("/upload")
public String doUpload(Model model) {
// 将上传文件目录下的所有文件名列出来,保存到模型对象中
String uploadPath = "C:" + File.separator + "SpringBootUpload";
File dir = new File(uploadPath);
model.addAttribute("files", dir.list());
return "upload";
}

@PostMapping("/upload")
@ResponseBody
public String upload(HttpServletRequest request) {
// 得到所有文件的列表
List<MultipartFile> files = ((MultipartHttpServletRequest) request).getFiles("file");
String uploadPath = "C:" + File.separator + "SpringBootUpload";
File dir = new File(uploadPath);

// 如果保存上传文件的目录不存在,则创建它
if (!dir.exists()) {
dir.mkdirs();
}
for (MultipartFile f : files) {
if (f.isEmpty()) {
continue;
}
File target = new File(uploadPath + File.separator + f.getOriginalFilename());
try {
f.transferTo(target);
} catch (IllegalStateException | IOException e) {
e.printStackTrace();
}
}
return "文件上传成功!";
}

@GetMapping("/download")
public ResponseEntity<byte[]> download(@RequestParam String fileName) throws IOException {
String dir = "C:" + File.separator + "SpringBootUpload";
String fileFullPath = dir + File.separator + fileName;
File file = new File(fileFullPath);

ResponseEntity.BodyBuilder builder = ResponseEntity.ok();
builder.contentLength(file.length());
builder.contentType(MediaType.APPLICATION_OCTET_STREAM);
fileName = new String(fileName.getBytes(StandardCharsets.UTF_8),StandardCharsets.ISO_8859_1);
builder.header("Content-Disposition", "attachment; filename=" + fileName);
return builder.body(FileUtils.readFileToByteArray(file));
}

}

SpringBoot Maven pom基本介紹2


<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<!--專案識別-->
<!--版本-->
<modelVersion>4.0.0</modelVersion>
<!--.公司.組織-->
<groupId>org.imi</groupId>
<!--專案名-->
<artifactId>myDemo</artifactId>
<!--版本 SNAPSHOT為不穩定版本-->
<version>1.0.0</version>
<packaging>war</packaging>
<name>myDemo</name>
<description>這是一個demo用的專案</description>

<!--變數設置-->
<properties>
<java.version>1.8</java.version>
<maven-jar-plugin.version>3.1.1</maven-jar-plugin.version>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>

<!--繼承springframework-->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.6.7</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>

<!--多環境建置-->
<profiles>
<!--開發環境-->
<profile>
<id>dev</id>
<properties>
<activatedProperties>dev</activatedProperties>
</properties>
<activation>
<!--默認環境-->
<activeByDefault>true</activeByDefault>
</activation>
</profile>
<!--正式環境-->
<profile>
<id>prod</id>
<properties>
<activatedProperties>prod</activatedProperties>
</properties>
</profile>
</profiles>

<!--設定引用函式庫-->
<dependencies>
<!--spring-boot Web服务-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>

<!--spring-boot thymeleaf模版引擎-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>

<!--spring-boot security安全守護-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-security</artifactId>
</dependency>

<!-- &lt;!&ndash;spring-boot jpa 資料庫&ndash;&gt;-->
<!-- <dependency>-->
<!-- <groupId>org.springframework.boot</groupId>-->
<!-- <artifactId>spring-boot-starter-data-jpa</artifactId>-->
<!-- </dependency>-->
<!-- <dependency>-->
<!-- <groupId>mysql</groupId>-->
<!-- <artifactId>mysql-connector-java</artifactId>-->
<!-- <scope>runtime</scope>-->
<!-- </dependency>-->

<!-- &lt;!&ndash;spring-boot test單元測試&ndash;&gt;-->
<!-- <dependency>-->
<!-- <groupId>org.springframework.boot</groupId>-->
<!-- <artifactId>spring-boot-starter-test</artifactId>-->
<!-- <scope>test</scope>-->
<!-- </dependency>-->
<!-- <dependency>-->
<!-- <groupId>junit</groupId>-->
<!-- <artifactId>junit</artifactId>-->
<!-- <scope>test</scope>-->
<!-- </dependency>-->


<!-- &lt;!&ndash;logback&ndash;&gt;-->
<!-- <dependency>-->
<!-- <groupId>ch.qos.logback</groupId>-->
<!-- <artifactId>logback-classic</artifactId>-->
<!-- <version>1.2.3</version>-->
<!-- </dependency>-->
</dependencies>

<!--建置專案所需要資訊-->
<build>
<finalName>${project.artifactId}</finalName>
<!--專案使用的程式列表-->
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>2.6.7</version>
</plugin>
</plugins>

<resources>
<!-- <resource>-->
<!-- <directory>src/main/java</directory>-->
<!-- <includes>-->
<!-- <include>**/*.xml</include>-->
<!-- </includes>-->
<!-- <filtering>false</filtering>-->
<!-- </resource>-->
<resource>
<directory>src/main/resources</directory>
<filtering>false</filtering>
<includes>
<include>ajax/</include>
<include>static/</include>
<!-- <include>**/*.xml</include>-->
</includes>
</resource>
<!--多環境配置-->
<resource>
<directory>src/main/resources</directory>
<!-- < filtering >true</ filtering >其含义是扫描src/main/resources/下的所有propertiesxml文件将其中的${}引用在打包时换成直接引用。-->
<filtering>true</filtering>
<includes>
<!-- 项目打包完成的包中只包含当前环境文件 -->
<include>application.properties</include>
<include>application-${activatedProperties}.properties</include>
</includes>
<!-- <excludes>-->
<!-- <exclude>static/font/**</exclude>-->
<!-- </excludes>-->
</resource>
</resources>
</build>
</project>

SpringBoot Maven pom基本介紹

 

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<!--專案識別-->
<!--版本-->
<modelVersion>4.0.0</modelVersion>
<!--.公司.組織-->
<groupId>org.imi</groupId>
<!--專案名-->
<artifactId>myDemo</artifactId>
<!--版本 SNAPSHOT為不穩定版本-->
<version>1.0.0</version>
<packaging>war</packaging>
<name>myDemo</name>
<description>這是一個demo用的專案</description>


<!--變數設置-->
<properties>
<java.version>1.8</java.version>
<maven-jar-plugin.version>3.1.1</maven-jar-plugin.version>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>

<!--繼承springframework-->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.6.7</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>

<!--設定引用函式庫-->
<dependencies>
<!--spring-boot Web服务-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>

<!--建置專案所需要資訊-->
<build>
<!--專案使用的程式列表-->
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>

</project>

Java NIO逐行读文件并写文件


/**
 * Created by scott on 2019/7/19.
 */
public class test {
    public static void main(String args[]) throws Exception {
        int bufSize = 100;
        File fin = new File("C:\\Users\\User\\Desktop\\localhost_access_log.2019-07-16.txt");
        File fout = new File("C:\\Users\\User\\Desktop\\localhost_access_log.2019-07-16_AAAA.txt");

        FileChannel fcin = new RandomAccessFile(fin, "r").getChannel();
        ByteBuffer rBuffer = ByteBuffer.allocate(bufSize);

        FileChannel fcout = new RandomAccessFile(fout, "rws").getChannel();
        ByteBuffer wBuffer = ByteBuffer.allocateDirect(bufSize);

        readFileByLine(bufSize, fcin, rBuffer, fcout, wBuffer);

        System.out.print("OK!!!");
    }

    public static void readFileByLine(int bufSize, FileChannel fcin, ByteBuffer rBuffer, FileChannel fcout, ByteBuffer wBuffer) {
        String enterStr = "\n";
        try {
            byte[] bs = new byte[bufSize];
            StringBuffer strBuf = new StringBuffer("");
            while (fcin.read(rBuffer) != -1) {
                int rSize = rBuffer.position();
                rBuffer.rewind();
                rBuffer.get(bs);
                rBuffer.clear();
                String tempString = new String(bs, 0, rSize);
                int fromIndex = 0;
                int endIndex = 0;
                //查找换行符符号\n,如果找到了,则写文件,如果没有找到则继续读取
                while ((endIndex = tempString.indexOf(enterStr, fromIndex)) != -1) {
                    String line = tempString.substring(fromIndex, endIndex);
                    line = strBuf.toString() + line;
                    System.out.println(line);
                    ;

                    if (line.contains("1111111111111111")) {
                        writeFileByLine(fcout, line + "\n");
                    }

                    strBuf.delete(0, strBuf.length());
                    fromIndex = endIndex + 1;
                }

                if (rSize > tempString.length()) {
                    strBuf.append(tempString.substring(fromIndex, tempString.length()));
                } else {
                    strBuf.append(tempString.substring(fromIndex, rSize));
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void writeFileByLine(FileChannel fcout, String line) {
        try {
            fcout.write(ByteBuffer.wrap(line.getBytes()), fcout.size());
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

使用 JavaScript 找到物件陣列中某屬性的最大值

這是個小技巧,使用 JavaScript/TypeScript 找到物件陣列中某項屬性的最大值,或者最小值也 OK。
假設我們有個兩個陣列如下:
var numArray = [1, 2, 3, 4];
var objArray = [
    { id: 1, name: 'A1' },
    { id: 2, name: 'A2' },
    { id: 3, name: 'A3' },
    { id: 4, name: 'A4' }
];

ES5

在 JavaScript ES5 中,可以使用這樣的寫法,找出陣列中的最大值
Math.max.apply(null, numArray) //4
首先 Math.max() 並利用 apply 的特性,將 numArray 陣列解構成一個一個的參數給 Math.max 去執行,藉此找到陣列中的最大值。
call 跟 apply 的差別在於 apply 第二個參數是陣列,而 call 是連續指定的參數
如果是要找物件陣列中某個屬性(例如 id)的最大值,可以搭配 Array.map() 改寫成這樣
Math.max.apply(null, objArray.map(function (o) {
    return o.id;
})) //4

ES6

在 JavaScript ES5 的時候,可以透過 apply 的特性來處理這次的問題,而到了 ES6 的時候,可以透過解構子 ... 來更優雅的處理這個問題,讓寫法變得更精簡。
找出陣列中的最大值
Math.max(...numArray) //4
找出物件陣列中的最大值,這裡使用 Arrow function 來改寫 Array.map() 的 CallBack function
Math.max(...objArray.map(p => p.id)) //4
越來越優雅且精簡的寫法,看起來好舒服唷~

Java 8 中的 Streams API

  public static void main(String[] args) {
        List<OrderInfo> orderList = new ArrayList<>();

        OrderInfo orderInfo = new OrderInfo();
        orderInfo.setOrderNo(null);


        OrderInfo orderInfo2 = new OrderInfo();
        orderInfo2.setOrderNo("123");

        OrderInfo orderInfo3 = new OrderInfo();
        orderInfo3.setOrderNo("222");


        orderList.add(orderInfo);
        orderList.add(orderInfo2);
        orderList.add(orderInfo3);

//        ---------------------------------------------------------------

        orderList = orderList.stream()
                .filter(                 //篩選
//                        o -> o.getOrderNo() != null
                        o -> {
                            if (o.getOrderNo() != null) {
                                return true;
                            } else {
                                return false;
                            }
                        }
                )
                .map(o -> {
//                    o.setOrderNo(o.getOrderNo() + "----");
                    return o;
                })
                .sorted((o1,o2) -> {    //設值
                    Integer oo1 = Integer.parseInt(o1.getOrderNo());
                    Integer oo2 = Integer.parseInt(o2.getOrderNo());
                    return oo2.compareTo(oo1);
                })
                .limit(1)               //幾個之前
                .collect(Collectors.toList());



//        走訪
        orderList.stream().forEach(o -> System.out.println(o.getOrderNo()));
        orderList.parallelStream().forEach(o -> System.out.println(o.getOrderNo()));    //有順序的


    }

批次檔BAT最基礎的71個指令 (轉貼)

在DOS統下,有三類檔案是可以使DOS執行而進行某些工作,這三類檔案分別有以下三個不同的副檔名 
1. .com (指令檔,命令檔) 例如 Command.com, Edit.com 等。 
2. .exe (執行檔) 例如 Copy.exe, Mem.exe 等。 
3. .bat (批次檔) 例如 Autoexec.bat 等。 

其中 .com 及 .exe 的檔案是用編譯或組合語言寫出,需要較多工夫來學習編寫,一般終端用者 (end-user) 不會花時間來學習編寫這些程式,但 .bat 的批次檔是一連串的 DOS 內部或外部指令 (批次檔的意思是 指令集) ,或是執行程式的主檔名,因此只要懂得這些指令,加上一些簡單的語法,就可寫出批次檔。終端用者利用自己編寫的批次檔,就可使 DOS 執行自己編定的程序。 

批次檔是由 DOS 指令組成,因此批次檔有錯誤時,就等於我們在提示號 (C)鍵入錯的指令一樣,螢幕會有 Bad command or file name 的回應。 

批次檔必定要是純文字檔案,可使用 DOS 的編輯器(DOS Editor)或用windows內的 記事本 寫出來,這兩個都是簡單的文書處理器,文稿內不含任何控制碼 (如字款和字的大小,文章闊度等等),這就最適合用來編來批次檔。 

指令大多數是為程式所用,對有寫一些簡單的自動化批次檔(Batch File)的人來說非常實用。在Window XP/7中利用cmd或是「所有程式/附屬應用程式/命令提示字元」進入指令模式後,你也可以利用「[help/?」指令查詢在系統提供的文字指令有哪些?

要瞭解特定命令的詳細資訊,請輸入 HELP 命令名稱:(以Windows XP為例)
ASSOC   顯示或修改檔案附檔名關聯。
AT      排定電腦上要執行的命令和程式。
ATTRIB  顯示或變更檔案屬性。
BREAK   設定或清除擴充的 CTRL+C 檢查。
CACLS   顯示或修改檔案的存取控制清單 (ACLs)。
CALL    從另一個批次程式呼叫一個批次程式。
CD      顯示目前目錄的名稱或變更。
CHCP    顯示或設定作用中的字碼編號。
CHDIR   顯示目前目錄的名稱或變更。
CHKDSK  檢查磁碟並顯示狀態報告。
CHKNTFS 顯示或修改開機時的磁碟檢查。
CLS     清除螢幕。
CMD     開始新的 Windows 命令轉譯器。
COLOR   設定預設主控台的前景和背景色彩。
COMP    比較兩個或兩組檔案的內容。
COMPACT 顯示或變更 NTFS 磁碟分割上的檔案壓縮
CONVERT 將 FAT 磁碟區轉換成 NTFS 格式。您不可轉換目前的磁碟機。
COPY    將一個或數個檔案複製到另一個位置。
DATE    顯示或設定日期
DEL     刪除檔案。
DIR     顯示目錄中的檔案和子目錄清單。
DISKCOMP比較兩張磁片的內容。
DISKCOPY將磁片上的內容複製到另一張磁片上。
DOSKEY  編輯命令列、恢復 Windows 命令和建立巨集
ECHO    顯示訊息、開啟或關閉命令回音。
ENDLOCAL結束批次檔環境變更的本土化工作。
ERASE   刪除一個或更多檔案。
EXIT    結束 CMD.EXE 程式 (命令轉譯器)。
FC      比較兩個或兩組檔案,然後顯示兩者之間的相異處。
FIND    在檔案中搜尋文字字串。
FINDSTR 在檔案中搜尋字串。
FOR     在一組檔案中的每個檔案執行一個特定的命令。
FORMAT  將磁碟格式化供 Windows 使用。
FTYPE   顯示或修改用於檔案附檔名關聯中的檔案類型。
GOTO    將 Windows 命令轉譯器指向批次程式中已經加了標籤的列。
GRAFTABL啟用 Windows 在圖形模式下顯示擴充的字集。
HELP    為 Windows 命令提供說明資訊。
IF      在批次程式中執行有條件的處理程序。
LABEL   建立、變更或刪除磁碟的磁碟區標籤。
MD      建立目錄。
MKDIR   建立目錄。
MODE    設定系統裝置。
MORE    一次顯示一個螢幕的輸出。
MOVE    從一個目錄移動一個或數個檔案到另一個目錄。
PATH    顯示或設定執行檔的搜尋路徑。
PAUSE   暫停處理批次檔並顯示訊息。
POPD    還原 PUSHD 儲存的目錄之前的值。
PRINT   列印文字檔案。
PROMPT  變更 Windows 的命令提示。
PUSHD   儲存目前的目錄,然後變更它。
RD      移除目錄。
RECOVER 從損壞或不良的磁碟中修復可讀取的資訊。
REM     在批次檔或 CONFIG.SYS 記錄意見 (註解)。
REN     重新命名檔案。
RENAME  重新命名檔案。
REPLACE 取代檔案。
RMDIR   移除目錄。
SET     顯示、設定或移除 Windows 環境變數
SETLOCAL開始批次檔中環境變更的本土化工作。
SHIFT   變更批次檔中可取代參數的位置。
SORT    將輸入排序。
START   開始另一個視窗來執行指定的程式或命令。
SUBST   將路徑與磁碟機代號相關聯。
TIME    顯示或設定系統時間。
TITLE   設定 CMD.EXE 工作階段的視窗標題。
TREE    以圖形顯示磁碟機或路徑的目錄結構。
TYPE    顯示文字檔的內容。
VER     顯示 Windows 版本。
VERIFY  告訴 Windows 是否要檢查您的檔案寫入磁片時正確與否。
VOL     顯示磁碟區標籤和序號。
XCOPY   複製檔案和樹狀目錄。