程序员取悦女朋友的正确姿势---Tips(iOS美容篇)

前言

女孩子都喜欢用美图工具进行图片美容,近来无事时,特意为某人写了个自定义图片滤镜生成器,安装到手机即可完成自定义滤镜渲染照片。app独一无二,虽简亦繁。

JH定律:
魔镜:最漂亮的女人是你老婆
魔镜:程序员不是木头人

核心技术

图片滤镜核心技术的基本思路如下:

核心技术流程

具体流程

1、创建一个图像处理工具类

注:该类实例包括一个图像处理方法,该方法在传入原始图像和一个颜色矩阵后生成一个处理好的图像。

 @interface JHFeilterManager : NSObject

@property (nonatomic,copy)imageBlock imageBLOCK;

- (UIImage *)createImageWithImage:(UIImage *)inImage colorMatrix:(const float *)f;

2、获取图像的每个像素点的RGBA值数组

注:该c方法返回一个指针,该指针指向一个数组,数组中的每四个元素都是图像上的一个像素点的RGBA的数值(0-255),用无符号的char是因为它正好的取值范围就是0-255

static unsigned char *RequestImagePixelData(UIImage * inImage){  

    CGImageRef img = [inImage CGImage];
    CGSize size = [inImage size];
    //使用上面的函数创建上下文
    CGContextRef cgctx = CreateRGBABitmapContex(img);
    CGRect rect = {{0,0},{size.width,size.height}};
    //将目标图像绘制到指定的上下文,实际为上下文内的bitmapData。
    CGContextDrawImage(cgctx, rect, img);
    unsigned char *data = CGBitmapContextGetData(cgctx);
    //释放上面的函数创建的上下文
    CGContextRelease(cgctx);
    cgctx = NULL;
    return data;
}

3、调整一个像素点的RGBA值

注:如下方法传入参数为数值指针和一个颜色矩阵,通过颜色矩阵调整指针指向地址存储的数值

static void changeRGB(int *red,int* green,int*blue,int*alpha ,const float *f){
    int redV = *red;
   int greenV = *green;
   int blueV = *blue;
   int alphaV = *alpha;
   //色值重新计算 计算规则如下
   *red = f[0] * redV + f[1]*greenV + f[2]*blueV + f[3] * alphaV + f[4];
   *green = f[5] * redV + f[6]*greenV + f[7]*blueV + f[8] * alphaV+ f[9];
   *blue = f[10] * redV + f[11]*greenV + f[12]*blueV + f[11] * alphaV+ f[14];
   *alpha = f[15] * redV + f[16]*greenV + f[17]*blueV + f[18] * alphaV+ f[19];
   *red < 0 ? (*red = 0):(0);
   *red > 255 ? (*red = 255):(0);
   *green < 0 ? (*green = 0):(0);
   *green > 255 ? (*green = 255):(0);
   *blue < 0 ? (*blue = 0):(0);
   *blue > 255 ? (*blue = 255):(0);
   *alpha < 0 ? (*alpha = 0):(0);
   *alpha > 255 ? (*alpha = 255):(0);
}

4、遍历每个像素,调整色值

注意!!!

在以下方法中,不要立刻释放malloc方法生成的bitmapData内存空间指针,(可能有的朋友觉得已经把内存空间地址给了位图上下文就可以立马释放掉了,但是实际上,由于位图上下文在后来的图像渲染时,仍然需要这一块内存,因此不能在此处立马释放掉内存,之前拜读的几篇博客索性就不释放内存了,因此会导致内存泄漏,处理一些高清图像时,手机内存会轻易飙升到1G以上,而导致程序挂掉)不然会导致位图上下文的内容数据不能正常存在而导致图片生成失败,在这里需要一个全局内存指针来指向它,并且在合适的时候释放内存。

该方法中需要传入一个原始图片信息和一个颜色矩阵,颜色矩阵决定了图像的渲染效果,因此不同的滤镜效果可以通过设置不同的颜色矩阵进行转换,如果您不了解颜色矩阵,点击这里进行了解

- (UIImage *)createImageWithImage:(UIImage *)inImage colorMatrix:(const float *)f{
   /* 图片位图像素值数组 */
   unsigned char *imgPixel = RequestImagePixelData(inImage);
   CGImageRef inImageRef = [inImage CGImage];
   /* 获取像素的横向和纵向个数 */
   long w = CGImageGetWidth(inImageRef);
   long h = CGImageGetHeight(inImageRef);

   int wOff = 0;
   int pixOff = 0;
   /* 遍历修改位图像素值 */
   for (long y = 0; y<h; y++) {
       pixOff = wOff;
       for (long x = 0; x<w; x++) {
       int red = (unsigned char)imgPixel[pixOff];
       int green = (unsigned char)imgPixel[pixOff+1];
       int blue = (unsigned char)imgPixel[pixOff +2];
       int alpha = (unsigned char)imgPixel[pixOff +3];
       changeRGB(&red, &green, &blue, &alpha,f);
       imgPixel[pixOff] = red;
       imgPixel[pixOff + 1] = green;
       imgPixel[pixOff + 2] = blue;
       imgPixel[pixOff + 3] = alpha;
       pixOff += 4;
  }
wOff += w * 4 ;
}
  NSInteger dataLength = w * h * 4;
  //创建要输出的图像的相关参数
  CGDataProviderRef provider = CGDataProviderCreateWithData(NULL, imgPixel, dataLength, NULL);
  if (!provider) {
    NSLog(@"创建输出图像相关参数失败!");
  }else{
    //每个像素点每个颜色的空间大小 单位为bit
    int bitsPerComponent = 8;
    //每个像素的空间大小 单位为bit
    int bitsPerPixel = 32;
    //每一行的颜色点的个数 每个像素有4个颜色点 每行有2个点
    ItemCount bytesPerRow = 4 * w;
    //创建RBGA色彩空间
    CGColorSpaceRef colorSpaceRef = CGColorSpaceCreateDeviceRGB();
    //位图的组成部分信息
    CGBitmapInfo bitmapInfo = kCGBitmapByteOrderDefault;
    //图像渲染参数 使用默认值
    CGColorRenderingIntent rederingIntent = kCGRenderingIntentDefault;
    //创建要输出的图像,参数依次为 宽、高、每个像素点的大小、每行占用空间大小、颜色空间、位图信息、相关输出参数
    CGImageRef imageRef = CGImageCreate(w, h,bitsPerComponent, bitsPerPixel, bytesPerRow, colorSpaceRef, bitmapInfo, provider,NULL, NO, rederingIntent);
    if (!imageRef) {
      NSLog(@"创建输出图像失败");
    }else{
      UIImage *my_image = [UIImage imageWithCGImage:imageRef];
      CFRelease(imageRef);
      CGColorSpaceRelease(colorSpaceRef);
      CGDataProviderRelease(provider);
  if (_imageBLOCK) {
    _imageBLOCK(my_image);
  }
  NSData *data = UIImageJPEGRepresentation(my_image, 1.0);
//在此释放位图空间
  free(bitmap);
  return [UIImage imageWithData:data];
}
}
  return nil;
}

5、UI搭建

UI界面非常简单,就是一张大图、20个颜色矩阵的信息录入框和两个按钮组成,代码就不过多赘述了,有需要的话可以到这里下载demo,喜欢的话不防给个star哦。

成果展示

成型的demo就是这个样子喽!

作者:Gabriella ,转载请自 SUN‘S BLOG - 专注互联网知识,分享互联网精神!
原文地址《程序员取悦女朋友的正确姿势---Tips(iOS美容篇)

时间: 2024-10-11 23:56:18

程序员取悦女朋友的正确姿势---Tips(iOS美容篇)的相关文章

CSDN日报20170224——《程序员该用哪种姿势来理财》

[程序人生] 程序员该用哪种姿势来理财 作者:纯洁的虫子 其实一直想写一篇文章名字都想好了,叫做"程序员该不该理财?".后来想了想,该不该这个就不用想了,必须要理财! 那么市面上那么多理财的方式对于我们屌丝的程序员该如何选择呢? 其实我也是那种土的掉渣的那种类型,以前几乎没有想过神马理财的,一来呢毕业的时候工资全都不够花的还理个毛线,二来总是感觉理财好像都是有钱人搞的东西:后来偶然进入了互联网金融行业,呆了几年,慢慢也接触了很多理财方式,但也还是一个门外汉,此文就是和大家一起聊聊我们程

关于程序员交女朋友的那点事儿

A与B(程序员)聊天 A:"我的老婆对我很好,每天都陪我熬夜完成工作!" B:"我的老婆也一样." A:"我的老婆在我累了的时候可以给我讲笑话!" B:"我的老婆也会给我讲笑话." A:"我的老婆在我有需求的时候可以让我很销魂!" B:"我老婆也可以帮我." A:"哦?这么说来你的老婆也很温柔很漂亮咯?" B:"--" 夜晚,B怀抱着自己的笔记本躺

程序员该用哪种姿势来理财

其实一直想写一篇文章名字都想好了,叫做“程序员该不该理财?”.后来想了想,该不该这个就不用想了,必须要理财!那么市面上那么多理财的方式对于我们屌丝的程序员该如何选择呢?其实我也是那种土的掉咋的那种类型,以前几乎没有想过神马理财的,一来呢毕业的时候工资全都不够花的还理个毛线,二来总是感觉理财好像都是有钱人搞的东西:后来偶然进入了互联网金融行业,呆了几年,慢慢也接触了很多理财方式,但也还是一个门外汉,此文就是和大家一起聊聊我们程序员该如何去理财?算是抛砖引玉,欢迎拍砖. 我的理财经历 先聊聊我的理财

Python 程序员的 Golang 学习指南(III): 入门篇

基础语法 类型和关键字 类型 // 基础类型 布尔类型: bool 整型: int8,uint8,int16,uint16,int32,uint32,int64,uint64,int,rune,byte,complex128, complex64,其中,byte 是 int8 的别名 浮点类型: float32 . float64 复数类型: complex64 . complex128 字符串: string 字符类型: rune(int32的别名) 错误类型: error // 复合类型 指

【程序员笔试面试复习】之一 网络与通信篇(一) 几大网络模型:OSI、TCP/IP、B/S与C/S、MVC结构

9.1网络模型 9.1.1. OSI七层模型 OSI(Open System Interconnection,开放系统互联)七层网络模型称为开放式网络互联参考模型.其为国际标准组织指定的一个指导信息互联.互通和协作的网络规范. 开放是指只要遵循OSI标准,位于世界上任何地方的任何系统之间都可以进行通信,开放系统是指遵循互联协议的实际系统,如电话系统. 从逻辑上可以将OSI开放系统互联分为七层模型,由下至上分别为物理层.数据链路层.网络层.传输层.会话层.表示层和应用层. 其中,上三层称为高层,用

浅谈月薪3万 iOS程序员 的职业规划与成长!(进阶篇)

前言: 干了这么多年的iOS,虽然接触了许多七七八八的东西.技术,但是感觉本身iOS却没有什么质的飞越,可能跟自己接触的项目深度有关,于是决定在学习其他技术的同时,加强自己在iOS方面的学习,提高自己的竞争力. 程序员薪水有高有低,同样有五年工作经验的程序员,可能一个人每月拿20K,一个拿5K.是什么因素导致了这种差异?我特意总结了容易导致薪水低的九大行为表现. 习惯即刻回报 他不懂得只有春天播种,秋天才会有收获.刚刚付出一点点,甚至还没有付出,就想要得到回报.技术刚刚掌握,能一边百度一边干活了

程序员陪女朋友自拍杆哪个好?自拍杆品牌推荐

出门在外旅游时拍照是个挺不方便的事,想给自己拍照还得请别人帮助,要么就只能拍风景和同伴了,随着雷军.崔永元等名人的自拍秀,自拍杆就一下子火起来了.有了自拍杆,自拍不再是爱美女孩的专属,企业大佬甚至国家元首都纷纷开始自拍秀,而各大旅游景点也不再是长枪短炮,更多的是拉长的各式自拍杆.出门在外,背包里的自拍杆已经和移动电源一样成为手机伴侣了,不管是在热门旅游景点,还是大街小巷,随处都能看见手持自拍杆的人.看似长得差不多的的自拍杆,价格区间从5元还包邮的淘宝爆款到站内壕友晒过的500元MOMAX不等,如

程序员必知的技术官网系列--mysql篇

mysql 官网 https://www.mysql.com/ 官网布局很简单, 其中常用的两块就是下载和文档这两块, 其中下载没什么可讲的, 本次重点依旧是文档. 首页 mysql 文档导航页 https://dev.mysql.com/doc/, 点击官网中的文档就能进入到下面的页面, mysql把内容分成了好几大块,第一块内Mysql 服务应用, 是所有内容里面最核心的.其他几块内容在第一块内容里面都有涉及, 稍后我们来看一下. 文档页导航 点开第一块内容的引用手册, 可以看到下面关于my

程序员必知的技术官网系列--spring篇

进入官网首先看到的是官网的轮播和首页导航,其中首页导航有四个, 下图有详细介绍, 随后也会单独讲解个导航包含的内容,本文只介绍前两个导航栏的内容, 本文只负责介绍官网的页面结构和文章用到的页面的内容翻译来辅助官网内容结构的讲解, 不做具体技术文档的解释, 以后有机会了再做解释. 好了开干. 首页 官网首页导航 看完导航栏后我们先往下滚动可以看到spring全家桶中的三个明星产品:spring boot,spring cloud,spring data flow和一个居中的标题spring:the