如何从CGImage获取原始像素数据

14

如何在缩放和移动后获取UIimage的像素数据,我想获取CGImage像素数据,然后从CGImage像素数据中获取UIImage。我们该如何做?


你能在 Pastebin 上提供一些代码吗? - Nikhil Bansal
2个回答

26

通过调用此函数可以获取原始数据

CFDataRef rawData = CGDataProviderCopyData(CGImageGetDataProvider(aCGImageRef));

你可以像这样逐个像素地遍历:

UInt8 * buf = (UInt8 *) CFDataGetBytePtr(rawData); 
CFIndex length = CFDataGetLength(rawData);

for(unsigned long i=0; i<length; i+=4)
{
    int r = buf[i];
    int g = buf[i+1];
    int b = buf[i+2];
     }
CFRelease(rawData);

1
如何从此缓冲区中获取UIImage? - user1362790
我有这样的代码: CFDataRef imageData = CGDataProviderCopyData(CGImageGetDataProvider(collisionImage.CGImage)); const UInt32 pixels = (const UInt32)CFDataGetBytePtr(imageData); 请问如何使用pixels再次创建图像? - user1362790
如果有人理解我的问题,请回复我,我想解释一下。m_cAppDelegatePtr.myFinalImageData = (NSData *) CGDataProviderCopyData(CGImageGetDataProvider(image)); UInt8 * buf = (UInt8 *) CFDataGetBytePtr(data); int length = CFDataGetLength(data);然后我像这样使用imageEditPtr = [[UIImage alloc] initWithData: m_cAppDelegatePtr.myFinalImageData];但是我没有得到任何图像到imageEditPtr。 - user1362790

0

试试这个。它可以给你所选像素的偏移量、Alpha、RGB颜色等信息。

源代码:

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event 
{
UITouch *touch = [[event allTouches] anyObject]; 

CGPoint startLocation   = [touch locationInView:self.view];

    if ([touch view] == YourImageView)
    {
        CGPoint location = [touch YourImageView];
        [self getPixelColorAtLocation:location];
        if((alpha==255)||(alpha==0))
        {
            //perform Your task
        }
    }
} 


- (UIColor*) getPixelColorAtLocation:(CGPoint)point 
{

UIColor* color = nil;

CGImageRef inImage;

inImage = YourImageView.image.CGImage;


// Create off screen bitmap context to draw the image into. Format ARGB is 4 bytes for each pixel: Alpa, Red, Green, Blue
CGContextRef cgctx = [self createARGBBitmapContextFromImage:inImage];
if (cgctx == NULL) { return nil; /* error */ }

size_t w = CGImageGetWidth(inImage);
size_t h = CGImageGetHeight(inImage);
CGRect rect = {{0,0},{w,h}};


// Draw the image to the bitmap context. Once we draw, the memory 
// allocated for the context for rendering will then contain the 
// raw image data in the specified color space.
CGContextDrawImage(cgctx, rect, inImage); 

// Now we can get a pointer to the image data associated with the bitmap
// context.
unsigned char* data = CGBitmapContextGetData (cgctx);
if (data != NULL) {
    //offset locates the pixel in the data from x,y. 
    //4 for 4 bytes of data per pixel, w is width of one row of data.
    int offset = 4*((w*round(point.y))+round(point.x));
    alpha =  data[offset]; 
    int red = data[offset+1]; 
    int green = data[offset+2]; 
    int blue = data[offset+3]; 
    NSLog(@"offset: %i colors: RGB A %i %i %i  %i",offset,red,green,blue,alpha);
    color = [UIColor colorWithRed:(red/255.0f) green:(green/255.0f) blue:(blue/255.0f) alpha:(alpha/255.0f)];
}

// When finished, release the context
//CGContextRelease(cgctx); 
// Free image data memory for the context
if (data) { free(data); }

return color;
}

- (CGContextRef) createARGBBitmapContextFromImage:(CGImageRef)inImage 
{
CGContextRef    context = NULL;
CGColorSpaceRef colorSpace;
void *          bitmapData;
int             bitmapByteCount;
int             bitmapBytesPerRow;

// Get image width, height. We'll use the entire image.
size_t pixelsWide = CGImageGetWidth(inImage);
size_t pixelsHigh = CGImageGetHeight(inImage);

// Declare the number of bytes per row. Each pixel in the bitmap in this
// example is represented by 4 bytes; 8 bits each of red, green, blue, and
// alpha.
bitmapBytesPerRow   = (pixelsWide * 4);
bitmapByteCount     = (bitmapBytesPerRow * pixelsHigh);

// Use the generic RGB color space.
colorSpace = CGColorSpaceCreateDeviceRGB();

if (colorSpace == NULL)
{
    fprintf(stderr, "Error allocating color space\n");
    return NULL;
}

// Allocate memory for image data. This is the destination in memory
// where any drawing to the bitmap context will be rendered.
bitmapData = malloc( bitmapByteCount );
if (bitmapData == NULL) 
{
    fprintf (stderr, "Memory not allocated!");
    CGColorSpaceRelease( colorSpace );
    return NULL;
}

// Create the bitmap context. We want pre-multiplied ARGB, 8-bits 
// per component. Regardless of what the source image format is 
// (CMYK, Grayscale, and so on) it will be converted over to the format
// specified here by CGBitmapContextCreate.
context = CGBitmapContextCreate (bitmapData,
                                 pixelsWide,
                                 pixelsHigh,
                                 8,      // bits per component
                                 bitmapBytesPerRow,
                                 colorSpace,
                                 kCGImageAlphaPremultipliedFirst);
if (context == NULL)
{
    free (bitmapData);
    fprintf (stderr, "Context not created!");
}

// Make sure and release colorspace before returning
CGColorSpaceRelease( colorSpace );

return context;
}

希望能有所帮助。谢谢 :)

实际上我需要整个图像的像素,并且我需要将图像保存到NSData或CFDataRef中,然后在另一个控制器中从NSData和CFDataRef获取图像。如果您知道,请帮助我。 - user1362790
你可以使用以下代码从 NSData 获取图像:NSData *imageData = UIImagePNGRepresentation(yourUIImage); uiImage = [UIImage imageWithData:imageData]; - Nikhil Bansal
但是我们无法将缩放后的图像保存到NSData中,也无法使用NSData获取缩放后的图像。如果我们存储了原始数据,那么也会得到原始数据,我只想要缩放后的数据。 - user1362790
为什么不呢??我们可以将缩放后的图像保存到NSData中!!! - Nikhil Bansal
看这里伙计,你正在缩放一张图片,然后肯定要把那个缩放后的图片保存在某个地方!!!这样你才能在项目的其他地方使用那个缩放后的图片。从该路径获取缩放后的图片。你明白我的意思吗??? - Nikhil Bansal
显示剩余5条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接