在 JTable 中用逗号替换点会导致错误 sorting-sequence
Replace point with comma in a JTable causes a wrong sorting-sequence
我尝试了很多方法来在 "German" 样式的 JTable 中显示逗号而不是点。在我的 JTable-Class 中,我编写了一个循环传递的结果集的方法,用于填充 DefaultTableModel 并直接返回模型或 JTable。
如果我将 BigDecimal 转换为字符串以用逗号替换点,结果将在我的 JTable 中正确显示 - 但是我的 Sorting-Sequence 不正确,因为值将被排序为字符串而不是 BigDecimal...
另一种解决方案是直接通过 SQL-Statement 排序并在我单击 row-header 时动态更改 sorting-sequence - 但这对我来说不是最佳解决方案,因为我的 Table 的内容是通过执行大量需要花费大量时间的查询生成的。
我试图设置我的 JTable 的语言环境,但没有任何效果。
有人知道如何以德语格式显示吗?
非常感谢!
这是我的 Class:
import java.math.BigDecimal;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.util.Locale;
import java.util.Vector;
import javax.swing.JTable;
import javax.swing.RowSorter;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableModel;
import javax.swing.table.TableRowSorter;
public class MyJTable extends JTable {
private static final long serialVersionUID = 1L;
public MyJTable(){
this.setLocale(Locale.GERMANY);
}
public MyJTable(ResultSet rs) throws SQLException{
DefaultTableModel model = this.getTableModel(rs);
this.setModel(model);
RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
this.setRowSorter(sorter);
this.setLocale(Locale.GERMANY);
}
public DefaultTableModel getTableModel(ResultSet rs) throws SQLException{
ResultSetMetaData metaData = rs.getMetaData();
// names of columns
Vector<String> columnNames = new Vector<String>();
int columnCount = metaData.getColumnCount();
for (int column = 1; column <= columnCount; column++) {
columnNames.add(metaData.getColumnName(column));
}
// data of the table
Vector<Vector<Object>> data = new Vector<Vector<Object>>();
String bigdecimal=null;
Object obigdecimal;
while (rs.next()) {
Vector<Object> vector = new Vector<Object>();
for (int columnIndex = 1; columnIndex <= columnCount; columnIndex++) {
if (metaData.getColumnClassName(columnIndex).equals("java.math.BigDecimal")){
bigdecimal=rs.getString(columnIndex);
if (bigdecimal!=null){
bigdecimal=bigdecimal.replace(".",",");
obigdecimal=bigdecimal;
vector.add(obigdecimal);
}
}
else
{
vector.add(rs.getObject(columnIndex));
}
}
data.add(vector);
}
return new DefaultTableModel(data, columnNames){
private static final long serialVersionUID = 1L;
public Class getColumnClass(int column) {
Class returnValue=Object.class;
try{
if ((column >= 0) && (column < getColumnCount())) {
returnValue = getValueAt(0, column).getClass();
} else {
returnValue = Object.class;
}
}
catch(Exception e){}
return returnValue;
}
};
}
}
Replace point with comma in a JTable causes a wrong sorting-sequence
问题的一小部分是您混淆了数据本身应该如何显示或呈现。
然而,恕我直言,主要问题是您没有在 table 模型中覆盖 getColumnClass(int columnIndex):我永远不会相信向模型询问第一行单元格的 class es。如果 getValueAt(0,1)
returns null
呢?答案:您将在 getValueAt(0,1).getClass()
电话中接到 NullPointerException
。
现在鉴于您问题中的上下文,您无法提供适当的 TableCellRenderer
,因为您事先不知道 class 结果集 return 是什么。但是,您可以依赖 Concepts: Editor and Renderers and override getColumnClass(...)
method to return a correct class using the ResultSetMetaData 中描述的默认行为,如下所示:
public TableModel getTableModel(ResultSet resultSet) throws SQLException {
ResultSetMetaData metaData = resultSet.getMetaData();
int columnCount = metaData.getColumnCount(); // columns number
String[] columnNames = new String[columnCount];
final Class[] columnClasses = new Class[columnCount];
for (int i = 1; i <= columnCount; i++) {
columnNames[i-1] = metaData.getColumnName(i); // fill columns names
try {
columnClasses[i-1] = Class.forName(metaData.getColumnClassName(i)); // fill column classes
} catch (ClassNotFoundException ex) {
// Log the exception here
}
}
resultSet.last();
int rowCount = resultSet.getRow(); // get rows number
resultSet.beforeFirst();
Object[][] data = new Object[rowCount][columnCount];
int currentRow = 0;
while (resultSet.next()) {
for (int currentColumn = 1; currentColumn <= columnCount; currentColumn++) {
data[currentRow][currentColumn - 1] = resultSet.getObject(currentColumn); // fill data set
}
currentRow++;
}
TableModel model = new DefaultTableModel(data, columnNames) {
@Override
public Class<?> getColumnClass(int columnIndex) {
return columnClasses[columnIndex];
}
}
return model;
}
注:使用Object[]
而不是Vector
只是因为我有代码somewhere else而且我太懒了修改它。
but then my Sorting-Sequence is not correct because the values will be sorted as a String instead of a BigDecimal...
不要在 TableModel 中存储字符串。存储 BigDecimal。
If I cast a BigDecimal to a String for replacing a point with a comma,
格式化数据是渲染器的工作。查看 Table Format Renderers 了解如何覆盖渲染器的 setValue(...)
方法的简单示例。
您还需要将 getColumnClass(...)
方法重写为 return BigDecimal,以便进行适当的排序。
我尝试了很多方法来在 "German" 样式的 JTable 中显示逗号而不是点。在我的 JTable-Class 中,我编写了一个循环传递的结果集的方法,用于填充 DefaultTableModel 并直接返回模型或 JTable。
如果我将 BigDecimal 转换为字符串以用逗号替换点,结果将在我的 JTable 中正确显示 - 但是我的 Sorting-Sequence 不正确,因为值将被排序为字符串而不是 BigDecimal...
另一种解决方案是直接通过 SQL-Statement 排序并在我单击 row-header 时动态更改 sorting-sequence - 但这对我来说不是最佳解决方案,因为我的 Table 的内容是通过执行大量需要花费大量时间的查询生成的。
我试图设置我的 JTable 的语言环境,但没有任何效果。
有人知道如何以德语格式显示吗?
非常感谢!
这是我的 Class:
import java.math.BigDecimal;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.util.Locale;
import java.util.Vector;
import javax.swing.JTable;
import javax.swing.RowSorter;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableModel;
import javax.swing.table.TableRowSorter;
public class MyJTable extends JTable {
private static final long serialVersionUID = 1L;
public MyJTable(){
this.setLocale(Locale.GERMANY);
}
public MyJTable(ResultSet rs) throws SQLException{
DefaultTableModel model = this.getTableModel(rs);
this.setModel(model);
RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
this.setRowSorter(sorter);
this.setLocale(Locale.GERMANY);
}
public DefaultTableModel getTableModel(ResultSet rs) throws SQLException{
ResultSetMetaData metaData = rs.getMetaData();
// names of columns
Vector<String> columnNames = new Vector<String>();
int columnCount = metaData.getColumnCount();
for (int column = 1; column <= columnCount; column++) {
columnNames.add(metaData.getColumnName(column));
}
// data of the table
Vector<Vector<Object>> data = new Vector<Vector<Object>>();
String bigdecimal=null;
Object obigdecimal;
while (rs.next()) {
Vector<Object> vector = new Vector<Object>();
for (int columnIndex = 1; columnIndex <= columnCount; columnIndex++) {
if (metaData.getColumnClassName(columnIndex).equals("java.math.BigDecimal")){
bigdecimal=rs.getString(columnIndex);
if (bigdecimal!=null){
bigdecimal=bigdecimal.replace(".",",");
obigdecimal=bigdecimal;
vector.add(obigdecimal);
}
}
else
{
vector.add(rs.getObject(columnIndex));
}
}
data.add(vector);
}
return new DefaultTableModel(data, columnNames){
private static final long serialVersionUID = 1L;
public Class getColumnClass(int column) {
Class returnValue=Object.class;
try{
if ((column >= 0) && (column < getColumnCount())) {
returnValue = getValueAt(0, column).getClass();
} else {
returnValue = Object.class;
}
}
catch(Exception e){}
return returnValue;
}
};
}
}
Replace point with comma in a JTable causes a wrong sorting-sequence
问题的一小部分是您混淆了数据本身应该如何显示或呈现。
然而,恕我直言,主要问题是您没有在 table 模型中覆盖 getColumnClass(int columnIndex):我永远不会相信向模型询问第一行单元格的 class es。如果 getValueAt(0,1)
returns null
呢?答案:您将在 getValueAt(0,1).getClass()
电话中接到 NullPointerException
。
现在鉴于您问题中的上下文,您无法提供适当的 TableCellRenderer
,因为您事先不知道 class 结果集 return 是什么。但是,您可以依赖 Concepts: Editor and Renderers and override getColumnClass(...)
method to return a correct class using the ResultSetMetaData 中描述的默认行为,如下所示:
public TableModel getTableModel(ResultSet resultSet) throws SQLException {
ResultSetMetaData metaData = resultSet.getMetaData();
int columnCount = metaData.getColumnCount(); // columns number
String[] columnNames = new String[columnCount];
final Class[] columnClasses = new Class[columnCount];
for (int i = 1; i <= columnCount; i++) {
columnNames[i-1] = metaData.getColumnName(i); // fill columns names
try {
columnClasses[i-1] = Class.forName(metaData.getColumnClassName(i)); // fill column classes
} catch (ClassNotFoundException ex) {
// Log the exception here
}
}
resultSet.last();
int rowCount = resultSet.getRow(); // get rows number
resultSet.beforeFirst();
Object[][] data = new Object[rowCount][columnCount];
int currentRow = 0;
while (resultSet.next()) {
for (int currentColumn = 1; currentColumn <= columnCount; currentColumn++) {
data[currentRow][currentColumn - 1] = resultSet.getObject(currentColumn); // fill data set
}
currentRow++;
}
TableModel model = new DefaultTableModel(data, columnNames) {
@Override
public Class<?> getColumnClass(int columnIndex) {
return columnClasses[columnIndex];
}
}
return model;
}
注:使用Object[]
而不是Vector
只是因为我有代码somewhere else而且我太懒了修改它。
but then my Sorting-Sequence is not correct because the values will be sorted as a String instead of a BigDecimal...
不要在 TableModel 中存储字符串。存储 BigDecimal。
If I cast a BigDecimal to a String for replacing a point with a comma,
格式化数据是渲染器的工作。查看 Table Format Renderers 了解如何覆盖渲染器的 setValue(...)
方法的简单示例。
您还需要将 getColumnClass(...)
方法重写为 return BigDecimal,以便进行适当的排序。