how to handle tiling of images on the fly

前端 未结 1 1263
情话喂你
情话喂你 2021-01-28 04:31

I am writing an app which would tile the images 256 * 256 and write those tile files back in the directory. I am updating my URL each time if there are any updates and tile thos

1条回答
  •  时光取名叫无心
    2021-01-28 05:04

    I have implemented solution for the similar problem(the difference is, I was not saving them in directory, those were for display purpose only.), with different approach.

    In my problem, I have 84 images of 250x250 dimension with size 8KB each( I added them on scrollView and on scrolling I load them, a bit similar to google maps, but more smooth). At first I was using the same approach as yours, but performance was problem. So, I used asynchornous loading concept. I wrote an UIImageView subclass with connectiond delegates, so the UIImageView subclass was responsible for loading it's image. And as loading is asynchronous so performance is far better.

    As you asked

    1) Memory consumption -- will the memory consumption for 5 images of size 200 KB a lot?

    Ans : 5x200KB = 1MB ~ 1.2MB or so(so you will need that much memory for displaying, if you have that much amount of memory then you should not worry.).. in my case 84x8KB = 672 ~ 900KB(as I was using some additional things like activity indicator for each imageview).

    2) How fast I can process my app if I have to tile 5 different URL with images at the same time ?

    Ans : As you are loading it in viewDidLoad ... or in main thread then performance will be an issue(blocking may happen, as I am not completely sure whether you are using threads or not).

    Quick suggestion:

    1. write an UIImageView subclass which has connection delegate methods.
    2. have some method that you can call from outside to message this imageView to start loading.(give the url)
    3. do proper deallocation of resources like responseData and connection object, once the downloading is complete.
    4. when you move from this view to other view do proper deallocation and removal of all these imageviews.
    5. use intruments to look for the allocations by this.
    

    CODE :

    TileImageView.h

    @interface TileImageView : UIImageView 
    {
        NSURLConnection *serverConnection;
        BOOL isImageRequested;
        NSMutableData *responseData;
    
    }
    -(void) startImageDownloading:(NSString *)pRequestURL
    -(void) deallocateResources;
    -(BOOL) isImageRequested;
    -(void)cancelConnectionRequest;
    
    -(void) addActivityIndicator;
    -(void) removeActivityIndicator;
    @end
    

    TileImageView.m

    @implementation TileImageView
    
    
    - (id)initWithFrame:(CGRect)frame {
    
        self = [super initWithFrame:frame];
        if (self) 
        {
            // Initialization code.
            isImageRequested = NO;
    
        }
        return self;
    }
    
    -(BOOL) isImageRequested
    {
        return isImageRequested;
    }
    
    -(void) startImageDownloading:(NSString *)pRequestURL
    {
    
    
        if (!isImageRequested)
        {
    
    
    
            NSURL *pServerURL = [[NSURL alloc] initWithString:pRequestURL];
            if (pServerURL != nil) 
            {
                isImageRequested = YES;
                [self addActivityIndicator];
                [self setBackgroundColor:[UIColor lightGrayColor]];
                NSURLRequest *pServerRequest = [[NSURLRequest alloc]initWithURL:pServerURL];
                serverConnection = [[NSURLConnection alloc] initWithRequest:pServerRequest delegate:self];
                if(serverConnection)
                {
                    responseData = [[NSMutableData alloc] init];
                }
                [pServerURL release];
                [pServerRequest release];
            }
    
    
        }
    
    }
    
    -(void) addActivityIndicator
    {
        UIActivityIndicatorView *tempActivityIndicator = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleWhite];
        CGFloat size = self.frame.size.width*0.12;
        [tempActivityIndicator setFrame:CGRectMake(0, 0, size, size)];
        [tempActivityIndicator setCenter:CGPointMake(self.frame.size.width/2, self.frame.size.height/2)];
        [tempActivityIndicator setTag:1000];
        [tempActivityIndicator setHidesWhenStopped:YES];
        [tempActivityIndicator startAnimating];
        [self addSubview:tempActivityIndicator];
        [tempActivityIndicator release];
    }
    
    -(void) removeActivityIndicator
    {
        UIActivityIndicatorView *tempActivityIndicator = (UIActivityIndicatorView *)[self viewWithTag:1000];
        if (tempActivityIndicator != nil)
        {
            [tempActivityIndicator stopAnimating];
            [tempActivityIndicator removeFromSuperview];
        }
    }
    
    
    -(void)cancelConnectionRequest
    {
        if (isImageRequested && serverConnection != nil)
        {
            [serverConnection cancel];
            [self removeActivityIndicator];
            [self deallocateResources];
            isImageRequested = NO;
        }
    
    }
    
    
    
    // Name         :   connection: didReceiveAuthenticationChallenge:
    // Description  :   NSURLConnectionDelegate method. Method that gets called when server sends an authentication challenge.
    - (void)connection:(NSURLConnection *)connection didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge 
    {
        if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust])
        {
            [challenge.sender useCredential:[NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust] forAuthenticationChallenge:challenge];
        }
        [challenge.sender continueWithoutCredentialForAuthenticationChallenge:challenge];
    }
    
    // Name         :   connection: didReceiveResponse:
    // Description  :   NSURLConnectionDelegate method. Method that gets called when response for the launched URL is received..
    -(void) connection:(NSURLConnection *) connection didReceiveResponse:(NSURLResponse *) response 
    {
        [responseData setLength:0]; 
    
    }
    
    // Name         :   connection: didReceiveData:
    // Description  :   NSURLConnectionDelegate method. Method that gets called when data for the launched URL is received..
    -(void) connection:(NSURLConnection *) connection didReceiveData:(NSData *) data 
    {
        [responseData appendData:data];
    }
    
    // Name         :   connection: didFailWithError:
    // Description  :   NSURLConnectionDelegate method. Method that gets called when an error for the launched URL is received..
    -(void) connection:(NSURLConnection *) connection didFailWithError:(NSError *) error 
    {
        NSLog(@"Error occured while loading image : %@",error);
        [self removeActivityIndicator];
        [self deallocateResources];
    
    
        UILabel *tempLabel = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 150, 30)];
        [tempLabel setBackgroundColor:[UIColor clearColor]];
        [tempLabel setFont:[UIFont systemFontOfSize:11.0f]];
        [tempLabel setCenter:CGPointMake(self.frame.size.width/2, self.frame.size.height/2)];
        [tempLabel setText:@"Image not available."];
        [self addSubview:tempLabel];
        [tempLabel release];
    
    }
    
    // Name         :   connectionDidFinishLoading
    // Description  :   NSURLConnectionDelegate method. Method that gets called when connection loading gets finished.
    -(void) connectionDidFinishLoading:(NSURLConnection *) connection 
    {
        [self removeActivityIndicator];
        UIImage *tempImage = [[UIImage alloc] initWithData:responseData];
        self.image = tempImage;
        [tempImage release];
        [self deallocateResources];
    }
    
    -(void) deallocateResources
    {
    
        if (serverConnection != nil) 
        {
            [serverConnection release];
            serverConnection = nil;
        }
        if (responseData != nil)
        {
            [responseData release];
            responseData = nil;
        }
    }
    
    - (void)dealloc {
        [super dealloc];
    }
    
    
    @end
    

    So, If you use above code then only thing you have to do is to add the object of TileImageView and just call method -(void) startImageDownloading:(NSString *)pRequestURL.

    Please use instruments to track allocations.
    

    Update :

    **How do I add TileImageView on scrollView ? :**
    

    //like this I add 84 images in a 2D shape( 12 x 7) grid ... and once Images are added I set scrollView's contentSize as per complete grid size.

    TileImageView *tileImageView  = [[TileImageView alloc]initWithFrame:];
    [tileImageView setTag:];
    [myImageScrollView addSubView:tileImageView];
    [tileImageView release];
    

    ..later in code when user scroll's and other imageviews come in visibility.I use following code...

    TileImageView *loadableImageView = (TileImageView *)[myImageScrollView viewWithTag:]; [loadableImageView startImageDownloading:];

    I do not need to do anything in drawRect: , as I have no need to do custome drawing.

    For Image names you can use tag property from imageView, but if you need some different name that are more like string then you can put another property in imageView for image name and set it while adding the image view. for saving data you can call your method once the image is downloaded in didFinishLoading method of TileImageView, where you can use that name.

    SECODN UPDATE

    How I add TileImageView on ScrollView

    gridCount = 0;
    rows = 7;
    columns = 12;
    totalGrids = rows*columns;
    //*above : all are NSInteger type variable declared at class level
    
    chunkWidth = 250;
    chunkHeight = 250;
    contentWidth = 0.0;
    contentHeight = 0.0;
    //*above : all are CGFloat type variable declared at class level
    for (int i=0; i

    And in ScrollViewDelegate method.

    - (void) scrollViewDidScroll:(UIScrollView *)scrollView
    {
        if (isZoomed)
        {
            xOffset = scrollView.contentOffset.x;
            yOffset = scrollView.contentOffset.y;
            //*above : both are CGFloat type variable declared at class level
    
            visibleColumn = xOffset/chunkWidth+1;
            visibleRow = yOffset/chunkHeight+1;
            gridNumber = (visibleRow-1)*columns+visibleColumn;
            adjGrid1 = gridNumber+1;
            adjGrid2 = gridNumber+columns;
            adjGrid3 = adjGrid2+1;
            //*above : all are NSInteger type variable declared at class level
            if (gridNumber ==1)
            {
                [self createAndSendScrollRequest:gridNumber];
            }
            if (adjGrid1 > 0 && adjGrid1 <= totalGrids) 
            {
                [self createAndSendScrollRequest:adjGrid1];
            }
            if (adjGrid2 > 0 && adjGrid2 <= totalGrids) 
            {
                [self createAndSendScrollRequest:adjGrid2];
            }
            if (adjGrid3 > 0 && adjGrid3 <= totalGrids) 
            {
                [self createAndSendScrollRequest:adjGrid3];
            }
        }
    
    
    }
    

    And this is how createAndSendScrollRequest is implemented.

    - (void) createAndSendScrollRequest:(NSInteger)chunkId
    {
    
        TileImageView *loadingImageView = (TileImageView *)[imageScrollView viewWithTag:chunkId];
        if ([loadingImageView image]==nil) 
        {
            [loadingImageView startImageDownloading:];
        }
    
    }
    

    Thanks,

    0 讨论(0)
提交回复
热议问题