按列名java POI读取excel
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/35905135/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
read excel by column name java POI
提问by user2091725
Good afternoon experts have a problem and I need to read the entire excel file by column name instead of its index,example:
下午好专家有一个问题,我需要按列名而不是索引读取整个 excel 文件,例如:
Column1 | Column2 | Column3
data1 data 2 data 3
POI allows me to read the column index by the method getColumnIndex()
returning for the Column1 = 0 , Column2= 1 etc,
but I need to read it by column name Column1
, Column2
etc ,
there any way to do this??
POI让我用该方法来读取列索引getColumnIndex()
返回了Column1 = 0 , Column2= 1 etc,
,但我需要按列名读它Column1
,Column2
等等,有没有办法做到这一点?
I need to read all fields of rows and columns by column name. attach the code which I read my file:
我需要按列名读取行和列的所有字段。附上我阅读文件的代码:
updated code:
更新代码:
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.DataFormatter;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Iterator;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class example {
DataFormatter fmt = new DataFormatter();
/**
* @param args the command line arguments
*/
public static void main(String[] args) throws FileNotFoundException, IOException {
example softMarti = new example();
FileInputStream file = new FileInputStream(new File("C:archive.xlsx"));
XSSFWorkbook workbook = new XSSFWorkbook(file);
XSSFSheet sheet = workbook.getSheetAt(0);
Iterator<Row> rowIterator = sheet.iterator();
while (rowIterator.hasNext()) {
Row row = rowIterator.next();
int rowIndex = row.getRowNum();
if (rowIndex < 1) {
continue;
}
Iterator<Cell> cellIterator = row.cellIterator();
while (cellIterator.hasNext()) {
Cell cell = cellIterator.next();
int columnIndex = cell.getColumnIndex();
if (columnIndex != 0 && columnIndex != 1 && columnIndex != 4) {
continue;
}
String columnName = "";
switch (columnIndex) {
case 0:
columnName = "column1";
break;
case 1:
columnName = "column2";
break;
case 4:
columnName = "column 4";
break;
}
String value = example.getValue(cell);
boolean valid = example.isValid(columnIndex, value);
if (valid) {
continue;
}
System.out.print(columnName + rowIndex);
System.out.println(" -> " + value);
}
}
// TODO code application logic here
}
private String getValue(Cell cell) {
switch (cell.getCellType()) {
case Cell.CELL_TYPE_BLANK:
return null;
case Cell.CELL_TYPE_BOOLEAN:
return "CELL_TYPE_BOOLEAN";
case Cell.CELL_TYPE_ERROR:
return "CELL_TYPE_ERROR";
case Cell.CELL_TYPE_FORMULA:
return "CELL_TYPE_FORMULA";
case Cell.CELL_TYPE_NUMERIC:
return fmt.formatCellValue(cell);
case Cell.CELL_TYPE_STRING:
return cell.getStringCellValue();
default:
return "none";
}
}
boolean isValid(int column, String value) {
if (value == null) {
return false;
}
String pattern = "";
switch (column) {
case 0:
pattern = "[A-Za-z0-9_\- ]{1,20}";
break;
case 1:
pattern = "[A-Za-z0-9_\- ]{1,80}";
break;
case 4:
pattern = "[0-9]{1,8}";
break;
}
Pattern pat = Pattern.compile(pattern);
Matcher mat = pat.matcher(value);
return mat.matches();
}
}
This code works but I need to validate column name because for my project columns may change position, that's my goal
此代码有效,但我需要验证列名,因为对于我的项目,列可能会改变位置,这是我的目标
采纳答案by neal
Why don't you read the first row(0) cell values (0-n) (aka column names) and put(columnName,columnIndex) into a map of String/int. Then you can reference the column index by name.
为什么不读取第一行(0)单元格值(0-n)(又名列名)并将(列名,列索引)放入字符串/整数映射中。然后您可以按名称引用列索引。
Here's an example:
下面是一个例子:
Map<String, Integer> map = new HashMap<String,Integer>(); //Create map
HSSFRow row = sheet.getRow(0); //Get first row
//following is boilerplate from the java doc
short minColIx = row.getFirstCellNum(); //get the first column index for a row
short maxColIx = row.getLastCellNum(); //get the last column index for a row
for(short colIx=minColIx; colIx<maxColIx; colIx++) { //loop from first to last index
HSSFCell cell = row.getCell(colIx); //get the cell
map.put(cell.getStringCellValue(),cell.getColumnIndex()) //add the cell contents (name of column) and cell index to the map
}
After this you'll have the map from columnName ---> index. Then you can do:
在此之后,您将获得 columnName ---> 索引中的映射。然后你可以这样做:
int idx = map.get("ColumnName");
....and you can use this in row.getCell(idx) to get the cells in all the other rows.
....并且您可以在 row.getCell(idx) 中使用它来获取所有其他行中的单元格。
Read the comments below in the code. I can't help you other than this. You need to read the documentation and figure out how to do it.
阅读下面代码中的注释。除了这个,我帮不了你。您需要阅读文档并弄清楚如何去做。
Workbook workbook = WorkbookFactory.create(new FileInputStream("C:\file.xlsx"));
Sheet sheet = workbook.getSheetAt(0);
totalRows = sheet.getPhysicalNumberOfRows();
Map<String, Integer> map = new HashMap<String,Integer>(); //Create map
HSSFRow row = sheet.getRow(0); //Get first row
//following is boilerplate from the java doc
short minColIx = row.getFirstCellNum(); //get the first column index for a row
short maxColIx = row.getLastCellNum(); //get the last column index for a row
for(short colIx=minColIx; colIx<maxColIx; colIx++) { //loop from first to last index
HSSFCell cell = row.getCell(colIx); //get the cell
map.put(cell.getStringCellValue(),cell.getColumnIndex()) //add the cell contents (name of column) and cell index to the map
}
List<ReportRow> listOfDataFromReport = new ArrayList<ReportRow>();
for(int x = 1; x<=totalRows; x++){
ReportRow rr = new ReportRow(); //Data structure to hold the data from the xls file.
HSSFRow dataRow = sheet.getRow(x); //get row 1 to row n (rows containing data)
int idxForColumn1 = map.get("Column1"); //get the column index for the column with header name = "Column1"
int idxForColumn2 = map.get("Column2"); //get the column index for the column with header name = "Column2"
int idxForColumn3 = map.get("Column3"); //get the column index for the column with header name = "Column3"
HSSFCell cell1 = dataRow.getCell(idxForColumn1) //Get the cells for each of the indexes
HSSFCell cell2 = dataRow.getCell(idxForColumn2)
HSSFCell cell3 = dataRow.getCell(idxForColumn3)
//NOTE THAT YOU HAVE TO KNOW THE DATA TYPES OF THE DATA YOU'RE EXTRACTING.
//FOR EXAMPLE I DON'T THINK YOU CAN USE cell.getStringCellValue IF YOU'RE TRYING TO GET A NUMBER
rr.setColumn1(cell1.getStringCellValue()); //Get the values out of those cells and put them into the report row object
rr.setColumn2(cell2.getStringCellValue());
rr.setColumn3(cell3.getStringCellValue());
listOfDataFromReport.add(rr);
}
//Now you have a list of report rows
for(int j = 0; j< listOfDataFromReport.size();j++){
System.out.println("Column 1 Value: " + listOfDataFromReport.get(j).getColumn1())
//etc...
}
//This class holds the values from the xls file. You may not need it
// I have no idea what you're doing with the data. If you simply wanted to
//print the data to console you wouldn't need it.
public static class ReportRow{
private String column1;
private String column2;
private String column3;
public String getColumn1(){
return this.column1;
}
public void setColumn1(String column1){
this.column1 = column1;
}
public String getColumn2(){
return this.column2;
}
public void setColumn2(String column2){
this.column2 = column2;
}
public String getColumn3(){
return this.column3;
}
public void setColumn3(String column3){
this.column3 = column3;
}
}
回答by Kostiantyn
I wrote a method
我写了一个方法
public static int columnName(String a) throws EncryptedDocumentException, InvalidFormatException, IOException {
int coefficient = 0;
String excelFilePath = ConfigurationReader.getProperty("pathToYourFile"); // or specify the path directly
FileInputStream inputStream = new FileInputStream(new File(excelFilePath));
Workbook wb = WorkbookFactory.create(inputStream);
Sheet sh = wb.getSheet("Sheet1");
Row row = sh.getRow(0);
int cellNum = row.getPhysicalNumberOfCells();
for (int i = 0; i < cellNum; i++) {
if ((row.getCell(i).toString()).equals(a)) {
coefficient = i;
}
}
return coefficient;
}
and then just call it in my code:
然后在我的代码中调用它:
Cell anyCellName = row.getCell(columnName("NameOfColumnInMyExcell"));
Cell anyCellName = row.getCell(columnName("NameOfColumnInMyExcell"));
And like this with any column names. Now I'm able to move my columns in any order and code works.
像这样的任何列名。现在我可以按任何顺序移动我的列并且代码有效。