- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试在列中创建渐变图案,每个单元格都有其 rgb 值。我面临的问题是 RGB 颜色在该列的其他单元格中被覆盖。因此,最后生成的 RGB 颜色将被赋予该列中的所有单元格。我尝试在每次迭代的循环中创建新对象,但覆盖仍然存在。
public static void giveGradientToColumn(HSSFWorkbook workbook, HSSFSheet sheet, String yemi, Double minimum, Double maximum) throws IOException {
int columnIndex = 5;
int maxRows = sheet.getPhysicalNumberOfRows();
Random rand = new Random();
int i = maxRows+1;
for(int rowIndex = maxRows-1 ; rowIndex > 0 ; rowIndex--){
Row row = CellUtil.getRow(rowIndex, sheet);
Cell cell = CellUtil.getCell(row, columnIndex);
String cellContent = cell.toString();
String percentvalue = cellContent.split("%")[0];
if(!(percentvalue.equals("NaN")))
{
FileOutputStream fileOut = new FileOutputStream(yemi);
double value;
HSSFWorkbook workbook1 = workbook;
try{
value = Double.parseDouble(percentvalue);
}
catch(Exception e){
continue;
}
double ratio;
if(maximum!=minimum)
ratio = 2 * (value-minimum) / (maximum - minimum);
else
ratio = 1;
int b = (max(0, 255*(1 - ratio)));
int r = (max(0, 255*(ratio - 1)));
int g = 255 - b - r;
r = rand.nextInt(255);
g = rand.nextInt(255);
b = rand.nextInt(255);
System.out.println(r+" "+g+" "+b);
HSSFCellStyle style = workbook1.createCellStyle();
HSSFPalette palette = workbook1.getCustomPalette();
HSSFColor myColor = setColor(workbook1, (byte) r, (byte) g, (byte) b);
short palIndex = myColor.getIndex();
style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
style.setFillForegroundColor(palIndex);
cell.setCellStyle(style);
workbook1.write(fileOut);
fileOut.close();
}
}
}
@SuppressWarnings("deprecation")
public static HSSFColor setColor(HSSFWorkbook workbook, byte r,byte g, byte b){
HSSFPalette palette = workbook.getCustomPalette();
HSSFColor hssfColor = null;
try {
hssfColor= palette.findColor(r, g, b);
if (hssfColor == null)
{
palette.setColorAtIndex(HSSFColor.GOLD.index, r, g, b);
hssfColor = palette.getColor(HSSFColor.GOLD.index);
}
} catch (Exception e) {
System.out.println(e);
}
return hssfColor;
}
我正在打印 rgb 值以进行调试,并且最后打印的颜色值将被赋予该列中的所有单元格。 (RGB目前是随机计算的。)
输出:- Output Image
我哪里出错了?
最佳答案
您的代码存在多个问题,导致多个不同的问题。所以答案会有点长。
首先了解 Excel 工作簿的结构:
工作簿由至少一个但也可能是多个工作表组成,每个工作表具有多行,每行具有多个单元格。每个单元格都可以有样式。但样式设置不存储在单元格中,而是存储在工作簿级别的样式表(样式表)中。因此,样式可以跨单元格、行甚至工作表存储。如果多个单元格具有相同的外观,则它们可以共享相同的存储样式。颜色也是如此。颜色也存储在工作簿级别。不幸的是,在 *.xls
BIFF
格式中,颜色仅限于调色板中的索引颜色。在 BIFF8 中,用户定义的 PALETTE 记录中有 49 个(索引 16 到 64),另外还有一些。这 49 个可以被覆盖,但数量不能增加。
现在看你的代码:
每次更改工作表行中的一个单元格时,您都会写出整个工作簿。你不应该这样做。相反,如果您完成了工作簿,则应该将其写出来。
您正在为每个需要添加样式的单元格创建一个新的单元格样式。你不应该这样做。 Excel *.xls
仅限于大约 4,000 种不同的单元格格式组合。因此,您需要检查工作簿中是否已存在您不需要的样式。这可能非常乏味,但有 CellUtil ,您已经找到并已在代码中使用它。这提供了 setCellStyleProperties
,它“尝试查找与单元格当前样式以及属性中的样式属性相匹配的现有 CellStyle。如果工作簿不包含匹配的样式,则会创建一个新样式。”。
您首先要搜索所需的颜色是否已存在。那挺好的。但如果没有,您将始终覆盖 GOLD
相同的颜色索引。由于颜色也存储在工作簿级别,因此只有最后覆盖的颜色值才会存储为GOLD
。如果工作簿中需要存储不同的颜色,则需要覆盖不同颜色索引。
示例:
源 Excel:
代码:
import java.io.FileInputStream;
import java.io.FileOutputStream;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.hssf.usermodel.*;
import org.apache.poi.hssf.util.HSSFColor;
import org.apache.poi.ss.util.CellUtil;
import java.util.Map;
import java.util.HashMap;
public class ExcelSetHSSFCellStyleColors {
static void giveGradientToColumn(HSSFSheet sheet, int columnIndex, double minimum, double maximum) throws Exception {
DataFormatter formatter = new DataFormatter(java.util.Locale.US);
short colorIndex = 16; //color indexes 0 to 15 should not be overwritten
HSSFPalette palette = sheet.getWorkbook().getCustomPalette();
for (Row row : sheet) {
Cell cell = CellUtil.getCell(row, columnIndex);
String cellContent = formatter.formatCellValue(cell);
System.out.println(cellContent);
String percentValue = cellContent.split("%")[0];
double value = Double.NaN;
try {
value = Double.valueOf(percentValue);
} catch(Exception e){
//percentValue was not numeric
}
if (!Double.isNaN(value) && value >= minimum && value <= maximum){
double ratio = (value - minimum) / (maximum - minimum);
byte r = (byte)Math.round(Math.max(0, 255 * (1 - ratio)));
byte b = 0;
byte g = (byte)Math.round(Math.max(0, 255 - (int)b - (int)r));
System.out.println(ratio + " " + String.format("%02X", r) + ":" + String.format("%02X", g) + ":" + String.format("%02X", b));
HSSFColor hssfColor = palette.findColor(r, g, b);
if (hssfColor == null /*&& colorIndex < 64*/) {
palette.setColorAtIndex(colorIndex, r, g, b);
hssfColor = palette.getColor(colorIndex);
colorIndex++;
}
System.out.println("got color: " + ((hssfColor!=null)?hssfColor.getIndex() + ": " + hssfColor.getHexString():hssfColor)); //if not a index available, hssfColor may be null
if (hssfColor != null) {
Map<String, Object> styleproperties = styleproperties = new HashMap<String, Object>();
styleproperties.put(CellUtil.FILL_FOREGROUND_COLOR, hssfColor.getIndex());
styleproperties.put(CellUtil.FILL_PATTERN, FillPatternType.SOLID_FOREGROUND);
CellUtil.setCellStyleProperties(cell, styleproperties);
}
}
}
}
public static void main(String[] args) throws Exception {
Workbook workbook = WorkbookFactory.create(new FileInputStream("ExcelTest.xls"));
HSSFSheet sheet = (HSSFSheet)workbook.getSheetAt(0);
giveGradientToColumn(sheet, 5, 10, 90);
workbook.write(new FileOutputStream("ExcelTestNew.xls"));
workbook.close();
}
}
结果:
免责声明:
代码经过测试,可使用 apache poi
的最新稳定版本 3.17
进行测试和运行。您的代码使用了旧版本(我知道是因为使用了 HSSFColor.GOLD.index
)。
顺便说一句:
我真正建议的是将 Excel 文件格式升级为现代的 *.xlsx
Office Open XML
文件格式。不仅颜色限制消失了,您还可以简单地使用 Conditional Formatting使用色阶可能会更好地满足您的要求。
关于java - 使用自定义 rgb 对列的单元格进行着色,以在列中创建渐变图案,每个单元格使用 apache poi 都有自己的 rgb,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51171098/
我正在 csv 上使用 hadoop 来分析一些数据。我使用sql/mysql(不确定)来分析数据,现在陷入了僵局。 我花了好几个小时在谷歌上搜索,却没有找到任何相关的东西。我需要一个查询,在该查询中
我正在为 Bootstrap 网格布局的“简单”任务而苦苦挣扎。我希望在大视口(viewport)上有 4 列,然后在中型设备上有 2 列,最后在较小的设备上只有 1 列。 当我测试我的代码片段时,似
对于这个令人困惑的标题,我深表歉意,我想不出这个问题的正确措辞。相反,我只会给你背景信息和目标: 这是在一个表中,一个人可能有也可能没有多行数据,这些行可能包含相同的 activity_id 值,也可
具有 3 列的数据库表 - A int , B int , C int 我的问题是: 如何使用 Sequelize 结果找到 A > B + C const countTasks = await Ta
我在通过以下功能编写此查询时遇到问题: 首先按第 2 列 DESC 排序,然后从“不同的第 1 列”中选择 只有 Column1 是 DISTINCT 此查询没有帮助,因为它首先从第 1 列中进行选择
使用 Bootstrap 非常有趣和有帮助,目前我在创建以下需求时遇到问题。 “使用 bootstrap 在桌面上有 4 列,在平板电脑上有 2 列,在移动设备上有 1 列”谁能告诉我正确的结构 最佳
我是 R 新手,正在问一个非常基本的问题。当然,我在尝试从所提供的示例中获取指导的同时做了功课here和 here ,但无法在我的案例中实现这个想法,即可能是由于我的问题中的比较维度更大。 我的实
通常我会使用 R 并执行 merge.by,但这个文件似乎太大了,部门中的任何一台计算机都无法处理它! (任何从事遗传学工作的人的附加信息)本质上,插补似乎删除了 snp ID 的 rs 数字,我只剩
我有一个 df , delta1 delta2 0 -1 2 0 -1 0 0 0 我想知道如何分配 delt
您好,我想知道是否可以执行以下操作。显然,我已经尝试在 phpMyAdmin 中运行它,但出现错误。也许还有另一种方式来编写此查询。 SELECT * FROM eat_eat_restaurants
我有 2 个列表(标题和数据值)。我想要将数据值列 1 匹配并替换为头文件列 1,以获得与 dataValue 列 1 和标题值列 2 匹配的值 头文件 TotalLoad,M0001001 Hois
我有两个不同长度的文件,file2 是一个很大的引用文件,我从中提取文件 1 的数据。 我有一行 awk,我通常会对其进行调整以在我的文件中进行查找和替换,但它总是在同一列中进行查找和替换。 所以对于
假设我有两个表,如下所示。 create table contract( c_ID number(1) primary key, c_name varchar2(50) not
我有一个带有 varchar 列的 H2 表,其检查约束定义如下: CONSTRAINT my_constraint CHECK (varchar_field <> '') 以下插入语句失败,但当我删
这是最少量的代码,可以清楚地说明我的问题: One Two Three 前 2 个 div 应该是 2 个左列。第三个应该占据页面的其余部分。最后,我将添加选项来隐藏和
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
在 Azure 中的 Log Analytics 中,我为 VM Heartbeat 选择一个预定义查询,我在编辑器中运行查询正常,但当我去创建警报时,我不断收到警报“查询未返回 TimeGenera
今天我开始使用 JexcelApi 并遇到了这个:当您尝试从特定位置获取元素时,不是像您通常期望的那样使用sheet.getCell(row,col),而是使用sheet.getCell(col,ro
我有一个包含 28 列的数据库。第一列是代码,第二列是名称,其余是值。 public void displayData() { con.Open(); MySqlDataAdapter
我很沮丧:每当我缩小这个网页时,一切都变得一团糟。我如何将网页居中,以便我可以缩小并且元素不会被错误定位。 (它应该是 2 列,但所有内容都合并为 1)我试过 但由于某种原因,这不起作用。 www.o
我是一名优秀的程序员,十分优秀!