After BLOCK Completed only Return

后端 未结 3 1274
别跟我提以往
别跟我提以往 2020-12-22 15:28

May I know is what are the solution to use in order to make the following code working in order.

- (CGFloat)getRowImageHeight
{
    CGFloat defaultHeight   =         


        
相关标签:
3条回答
  • 2020-12-22 15:50

    Things that I am trying to do is at view load, I preload the image before it appears on the view. Then get image block to return the image height and I wanted to return back the height to

    - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
    

    But it seems that answers above told me that the approach is impossible. So I tried to use delegate and return the height to main view and update the cell height by using the code below after the image has been successfully fetch at cell level.

    - (void)reloadItemImageTVCellHeight:(CGFloat)height
    {
        if( !self.hasUpdatedImageCellHeight ){
            self.hasUpdatedImageCellHeight   = YES;
            self.itemImageCellHeight        = height;
            [self.mTableView reloadRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:ROW_IMAGE inSection:0]]
                             withRowAnimation:UITableViewRowAnimationNone];
        }
    }
    

    In viewDidLoad, I've set self.hasUpdatedImageCellHeight to NO and self.itemImageCellHeight to 300.f. self.hasUpdatedImageCellHeight is to control whether the delegate has already return me any value, if YES then I will update it to YES in order to prevent the delegate codes keep calling again & again.

    This is my approach for it, but still it is best if there is an approach that can know the image height natively before the table view being completely loaded.

    0 讨论(0)
  • 2020-12-22 15:58

    the good approach would be something like this:

    - (void)getRowImageHeightWithCompletionBlock:(void(^)(CGFloat height))completion {
        CGFloat defaultHeight = 300.f;
        [self configureImageTVCell:self.itemImageTVCell block:^(UIImage *image, BOOL succeeded) {
            if(succeeded && completion) {
                completion(image.size.height);
            }
        }];
    }
    
    0 讨论(0)
  • 2020-12-22 16:09

    Basically, what you are trying to do is impossible. On the one hand, you have a method from which you need to return a value immediately:

    - (CGFloat)getRowImageHeight
    {
        // ... do some stuff ...
        return height;
    }
    

    On the other hand, in the middle of that code ("do some stuff"), you are performing an asynchronous operation:

    [self configureImageTVCell:self.itemImageTVCell
                         block:^(UIImage *image, BOOL succeeded) {
                             // THIS IS ASYNCHRONOUS
                             if( succeeded )
                                 height     = image.size.height;
                         }];
    

    That means, by definition, that the code in the block will run at some future, unknown time. Meanwhile, your outer method getRowImageHeight has finished and returned its value long ago.

    That is the nature of asynchronous code execution.

    You need to rearchitect your entire approach so that it works in conjunction with your asynchronous code.

    Of course, you have not revealed what you are really trying to do, so I can't tell you how to rearchitect it. But let us pretend for a moment that you are trying to populate a table view's cells with images to be downloaded from the Internet. Well, that is a well-established problem with answers all over the place, including many good explanations on Stack Overflow. So you would read those answers.

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