CS50棕褐色问题,将图像从正常状态转换为棕褐色

如何解决CS50棕褐色问题,将图像从正常状态转换为棕褐色

我正在研究一个名为filter(不舒服,第4周)的CS50程序,它必须将图像从正常图像传输到棕褐色。除非必须转移白色,否则它工作正常。尝试转移白色时,它只是将其转换为蓝色和绿色。像这样:

原始

ORIGINAL

SEPIA

enter image description here

如您所见,除了白色或接近白色外,它可以将所有内容都转换为精细。 这是我的代码(仅适用于棕褐色部分):


void sepia(int height,int width,RGBTRIPLE image[height][width])
{
    for(int j = 0; j < width; j++)
    {
       for (int i = 0; i < height; i++)
       {
           int sepiared =  image[i][j].rgbtRed *.393  +   image[i][j].rgbtGreen *.769 +  image[i][j].rgbtBlue *.189;
           int sepiagreen =  image[i][j].rgbtRed *.349  +  image[i][j].rgbtGreen *.686 +  image[i][j].rgbtBlue *.168;
            int sepiablue =  image[i][j].rgbtRed *.272  +   image[i][j].rgbtGreen *.534 +  image[i][j].rgbtBlue *.131;
           image[i][j].rgbtRed = sepiared;
           image[i][j].rgbtGreen = sepiagreen;
           image[i][j].rgbtBlue = sepiablue;
       }
    }

    return;
}

请帮助我理解为什么会这样。 Clang不打印任何错误消息。

您的确是, 代码丢失:)

解决方法

您需要使用“饱和度数学”。

对于近白色,您的中间值(例如sepiared)可以超过255。

255(0xFF)是unsigned char可以容纳的最大值

例如,如果sepiared为256(0x100),则将其放入rgbtRed时,将仅保留最右边的8位,并且该值将被截断 设为0。因此,您将得到一个非常暗的值[接近黑色],而不是一个非常明亮的值[接近白色]。

要解决此问题,请添加:

if (sepiared > 255)
    sepiared = 255;

另外,请注意for循环的顺序,它的缓存效率非常低。

而且,在任何地方使用image[i][j].whatever是浪费[并且可能很慢]。最好使用指向当前像素的指针。

无论如何,这是您的代码的更新版本,其中包含以下更改:

void
sepia(int height,int width,RGBTRIPLE image[height][width])
{
    RGBTRIPLE *pix;

    for (int i = 0; i < height; i++) {
        pix = &image[i][0];
        for (int j = 0; j < width; j++,pix++) {
            int sepiared = pix->rgbtRed * .393 +
                pix->rgbtGreen * .769 +
                pix->rgbtBlue * .189;
            int sepiagreen = pix->rgbtRed * .349 +
                pix->rgbtGreen * .686 +
                pix->rgbtBlue * .168;
            int sepiablue = pix->rgbtRed * .272 +
                pix->rgbtGreen * .534 +
                pix->rgbtBlue * .131;

            if (sepiared > 255)
                sepiared = 255;
            if (sepiagreen > 255)
                sepiagreen = 255;
            if (sepiablue > 255)
                sepiablue = 255;

            pix->rgbtRed = sepiared;
            pix->rgbtGreen = sepiagreen;
            pix->rgbtBlue = sepiablue;
        }
    }
}

此外,请注意在像素图像上使用浮点数学运算可能会有点慢。在这种情况下,使用缩放整数数学会更快/更好。

这是执行此操作的版本:

void
sepia(int height,pix++) {
            int sepiared = pix->rgbtRed * 393 +
                pix->rgbtGreen * 769 +
                pix->rgbtBlue * 189;
            int sepiagreen = pix->rgbtRed * 349 +
                pix->rgbtGreen * 686 +
                pix->rgbtBlue * 168;
            int sepiablue = pix->rgbtRed * 272 +
                pix->rgbtGreen * 534 +
                pix->rgbtBlue * 131;

            sepiared /= 1000;
            sepiagreen /= 1000;
            sepiablue /= 1000;

            if (sepiared > 255)
                sepiared = 255;
            if (sepiagreen > 255)
                sepiagreen = 255;
            if (sepiablue > 255)
                sepiablue = 255;

            pix->rgbtRed = sepiared;
            pix->rgbtGreen = sepiagreen;
            pix->rgbtBlue = sepiablue;
        }
    }
}
,

所以您需要认真编辑代码

将原始图像暂时存储一段时间


            originalBlue = image[i][j].rgbtBlue;
            originalRed = image[i][j].rgbtRed;
            originalGreen = image[i][j].rgbtGreen;

每个公式的结果都可能不是整数,因此请使用float并将其四舍五入到最接近的整数

            sepiaRed = round(.393 * originalRed + .769 * originalGreen + .189 * originalBlue);
            sepiaGreen = round(.349 * originalRed + .686 * originalGreen + .168 * originalBlue);
            sepiaBlue = round(.272 * originalRed + .534 * originalGreen + .131 * originalBlue);

            if (sepiaRed > 255)
            {
                sepiaRed = 255;
            }

            if (sepiaGreen > 255)
            {
                sepiaGreen = 255;
            }

            if (sepiaBlue > 255)
            {
                sepiaBlue = 255;
            }

现在将值存储为原始值

            image[i][j].rgbtBlue = sepiaBlue;
            image[i][j].rgbtRed = sepiaRed;
            image[i][j].rgbtGreen = sepiaGreen;

在循环外声明所有变量

    float sepiaRed;
    float sepiaBlue;
    float sepiaGreen;
    int originalRed;
    int originalBlue;
    int originalGreen;

希望对您有帮助

,

首先,顺便说一句,当您遇到CS50问题时,请先在StackOverflow中搜索关键字。答案可能在那里。如果您搜索sepia RGBTRIPLE cs50,您将获得很多匹配。

经过大量像素处理后,您将磨练一些有用的调试直观知识。其中:

  • 如果看到对角线偏移,则图像的每行字节数可能大于宽度乘以像素大小。 (尤其是在YCbCr图像中或在图像缓冲区与128位向量大小对齐的平台上。)
  • 2倍或0.5倍的图像显示可能意味着您没有注意视网膜显示屏上的比例值。
  • 某些色彩空间错误将立即向您指出BGR与RGB字节排序问题。根本没有蓝色通道或全部没有蓝色通道?可能将ARGB与BGRA混合。

更重要的是:

  • 如果您在明亮或色彩饱和的区域看到不整洁的图像,则您的像素分量值可能会过饱和(超过最大值,并丢掉高位)。

每次(1)将颜色分量乘以大于1的数字或(2)每次将多个颜色分量相加时,都需要考虑如果超过最大值会发生什么。如果您的中级数学将添加两个值然后除以2,请确保您的编译操作将使用足够大的变量大小来容纳该额外位。

因此,在您的内部循环中,当操作白色像素时,几乎每个颜色分量都将超过255(即红色和绿色将超过,但蓝色不会超过,因为棕褐色的蓝色较低):

int sepiared =  image[i][j].rgbtRed *.393  +   image[i][j].rgbtGreen *.769 +  image[i][j].rgbtBlue *.189;
int sepiagreen =  image[i][j].rgbtRed *.349  +  image[i][j].rgbtGreen *.686 +  image[i][j].rgbtBlue *.168;
int sepiablue =  image[i][j].rgbtRed *.272  +   image[i][j].rgbtGreen *.534 +  image[i][j].rgbtBlue *.131;

结果值为{255,255,255} x {.393 + .769 + .189,.349 + .686 + .168,.272 + .534 + .131}或{344.5,306.8 ,238.9}。

但是因为在RGBTRIPLE结构的BYTE组件中没有足够的位容纳那些值,所以您的值将是错误的。因此,您可以执行以下操作:

int sepiared =  (int) image[i][j].rgbtRed *.393  +   image[i][j].rgbtGreen *.769 +  image[i][j].rgbtBlue *.189;
int sepiagreen =  (int) image[i][j].rgbtRed *.349  +  image[i][j].rgbtGreen *.686 +  image[i][j].rgbtBlue *.168;
int sepiablue =  (int) image[i][j].rgbtRed *.272  +   image[i][j].rgbtGreen *.534 +  image[i][j].rgbtBlue *.131;
sepiared = min(sepiared,255);
sepiagreen = min(sepiagreen,255);
sepiablue = min(sepiablue,255);

请注意,我进行了两项更改:

  1. 将每个表达式中的第一个值转换为(int);否则,计算将以字节为单位,您将丢失高位。
  2. 在每个像素组件上强制使用最大值255。

在考虑其他答案时,请添加我的第一个修正。如果您已经放弃高位,则最多检查255个值将无济于事!

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-