淺談Java讀取Csv實(shí)踐
在CSV中,數(shù)據(jù)的字段由逗號(hào)分開,程序通過讀取文件重新創(chuàng)建正確的字段,方法是每次遇到逗號(hào)時(shí)開始新一段數(shù)據(jù)。CSV文件是一個(gè)計(jì)算機(jī)數(shù)據(jù)文件用于執(zhí)行審判和真正的組織工具,逗號(hào)分隔的清單。CSV文件是用于數(shù)字存儲(chǔ)的數(shù)據(jù)結(jié)構(gòu)表的形式列出,每個(gè)相關(guān)的項(xiàng)目(會(huì)員)的一組是與他人也由逗號(hào)隔開的規(guī)定。中的每一行都CSV文件對(duì)應(yīng)表中的行。在一條線,字段用逗號(hào)分隔,每一個(gè)領(lǐng)域?qū)儆谝粋€(gè)表列。CSV文件常常被用于移動(dòng)表格數(shù)據(jù)之間的兩個(gè)不同的計(jì)算機(jī)程序,例如關(guān)系數(shù)據(jù)庫程序和電子表格程序。讓我們一起來探討Java是怎樣讀取CSV的??
看項(xiàng)目中以前的處理方式是直接用用java IO類庫讀取Csv文件,實(shí)際處理中發(fā)現(xiàn)Csv文件本身包含了對(duì)各種特殊字符的處理信息。最常見的比如:
1. 對(duì)包含特殊字符的字符串?dāng)?shù)據(jù)首尾加雙引號(hào)
2. 對(duì)數(shù)據(jù)中的單個(gè)雙引號(hào)前加單個(gè)雙引號(hào)
其它...
所以用Java IO讀到的字符串全是經(jīng)過處理后的字符串,在某些場(chǎng)景下是不符合預(yù)期需求的。比如我需要的是不做任何處理的原始內(nèi)容。
項(xiàng)目中另一種常見的文件格式Excel用了POI來處理,但是POI不支持Csv格式,于是找到了javacsv。
代碼很簡(jiǎn)單:
Java代碼
- public List importCsv(String file)
- {
- List list = new ArrayList();
- CsvReader reader = null;
- try
- {
- //初始化CsvReader并指定列分隔符和字符編碼
- reader = new CsvReader(file, ',', Charset.forName("GBK"));
- while (reader.readRecord())
- {
- //讀取每行數(shù)據(jù)以數(shù)組形式返回
- String[] str = reader.getValues();
- if (str != null && str.length > 0)
- {
- if (str[0] != null && !"".equals(str[0].trim()))
- {
- list.add(str);
- }
- }
- }
- }
- catch (FileNotFoundException e) {
- log.error("Error reading csv file.",e);
- }
- catch (IOException e)
- {
- log.error("",e);
- }
- finally
- {
- if(reader != null)
- //關(guān)閉CsvReader
- reader.close();
- }
- return list;
- }
以上代碼有幾個(gè)要點(diǎn):
1 初始化CsvReader時(shí)指定分隔符和字符編碼,如果不指定,默認(rèn)分別為逗號(hào)和ISO-8859-1,我用了GBK,具體使用時(shí)要看當(dāng)時(shí)的字符編碼而定。
2 讀取每行數(shù)據(jù),返回字符串?dāng)?shù)組,數(shù)組內(nèi)的順序即文件數(shù)據(jù)列的順序
3 ***記得關(guān)閉CsvReader
是不是很簡(jiǎn)單,返回的數(shù)組格式也正好是我想要的,而且拿到是原始的數(shù)據(jù),沒有經(jīng)過特殊字符處理。
有些同學(xué)質(zhì)疑特殊字符未經(jīng)處理,插到數(shù)據(jù)庫會(huì)出錯(cuò),其實(shí)大可不必我們手工處理,一些基礎(chǔ)組件比如JDBC的preparedstatement已經(jīng)包含了對(duì)特殊字符的處理,我們只要以綁定參數(shù)的形式來傳送這些包含特殊字符的數(shù)據(jù)就可以。常用的持久化框架底層也封裝了JDBC,自然也對(duì)特殊字符做了處理。
【編輯推薦】