Android記憶體優化之封裝九宮格
隨著市場上越來越多的APP上線,好多軟體對手機的記憶體要求也是很大,所以我們在開發的時候一定要掌握如何去優化記憶體,將自己的APP儘可能優化。今天我們就一起看一下九宮格的優化。下面是軟體的截圖
1、為了達到更好的效果我們不用UITableView,首先我們要通過XIB自定義一個圖片和文字如圖:
2、自定義主檢視JRMainScrollView,通過協議代理來實現功能,做法和UITableView類似,大家可以參考一下UITableView
首先:我們要定義資料來源協議
//資料來源協議 @protocol JRMainScrollDataSource <NSObject> //獲取總的數量 - (NSInteger) numberOfItems; //獲取列的數量 - (NSInteger) numberOfColumsOfRow; //獲取item - (UIView *) mainScrollView:(JRMainScrollView *)mainScrollView itemAtIndex:(NSInteger) index; @end
其次:我們要定義屬性協議
//屬性協議 @protocol JRMainScrollDelegate <NSObject> @optional //獲取高度 - (CGFloat) heightForItemAtView:(JRMainScrollView *) mainScrollView; //獲取寬度 - (CGFloat) widthForItemAtView:(JRMainScrollView *) mainScrollView; //獲取間距 - (CGFloat) mainScrollView:(JRMainScrollView *)mainScrollView spaceForItemWithType:(kJRMainScrollItemSpace)type; @end
注意獲取間距包括到左右的間距和上下的間距通過定義一個列舉實現
typedef enum{ kJRMainScrollItemLeftSpace, kJRMainScrollItemTopSpace } kJRMainScrollItemSpace;
3、內部佈局實現,計算出當前所有的frame,並且放入陣列在此期間,用的的屬性引數都需要從代理來獲取,程式碼如下
//載入子檢視 - (void)_loadSubViews{ //獲取總個數和列數 NSInteger totalItems=[self.jrDataSource numberOfItems]; NSInteger colum=[self.jrDataSource numberOfColumsOfRow]; //獲取寬度和高度 CGFloat itemWidth=[self.jrDelegate widthForItemAtView:self]; CGFloat itemHeigt=[self.jrDelegate heightForItemAtView:self]; //獲取上下間距 CGFloat leftSpace=[self.jrDelegate mainScrollView:self spaceForItemWithType:kJRMainScrollItemLeftSpace]; CGFloat topSpace=[self.jrDelegate mainScrollView:self spaceForItemWithType:kJRMainScrollItemTopSpace]; CGFloat space=(kWidth-2*leftSpace-colum*itemWidth)/(colum-1)+itemWidth; for (int i=0;i<totalItems;i++) { int clo=i%colum; int row=i/colum; CGRect frame=CGRectMake(leftSpace+clo*space, 20+row*(itemHeigt+topSpace), itemWidth, itemHeigt); [self.array addObject:[NSValue valueWithCGRect:frame]]; } self.contentSize=CGSizeMake(0, CGRectGetMaxY([[self.array lastObject] CGRectValue])); self.showsVerticalScrollIndicator=NO; }
4、判斷當前的frame是不是在當前的螢幕可視範圍之內,如果要是在的進行檢視的渲染,如果不在不予理睬。
-(void)layoutSubviews{ [super layoutSubviews]; //迴圈便利獲取在螢幕中的frame for (int i=0;i<self.array.count;i++) { UIView * tempView=(UIView *)self.current[@(i)]; CGRect rect=[self.array[i] CGRectValue]; if ([self isInScreenWith:rect]) { if(!tempView){//字典裡沒有的才需要重重新載入 UIView *view=[self.jrDataSource mainScrollView:self itemAtIndex:i]; view.frame=rect; [self.current setObject:view forKey:@(i)]; [self addSubview:view]; } }else if(tempView){//如果存在字典而且不在視線內部的則移除 [self.current removeObjectForKey:@(i)]; [tempView removeFromSuperview]; [self.pool addObject:tempView]; } } //判斷是不是在視野內部,其中有兩種情況,Y值在螢幕內部,或者MAXY值在螢幕內部 - (BOOL) isInScreenWith:(CGRect) frame{ CGFloat setMiny=self.contentOffset.y; CGFloat setMaxy=self.contentOffset.y+kHeight; BOOL condition1=frame.origin.y>=setMiny&&frame.origin.y<=setMaxy; BOOL condition2=CGRectGetMaxY(frame)>=setMiny&&CGRectGetMaxY(frame)<=setMaxy; if(condition1||condition2){ return YES; } return NO; }
5、操作緩衝池重複利用物件
/** 存放frame*/ @property(nonatomic,strong) NSMutableArray * array; /** 存放當前顯示的物件*/ @property(nonatomic,strong) NSMutableDictionary * current; /** 存放緩衝池物件*/ @property(nonatomic,strong) NSMutableSet * pool; /** * 獲取重複利用的物件 * * @param identy <#identy description#> * * @return <#return value description#> */ - (JRRectView *) dequeueReusedItemWithIdenty:(NSString *) identy{ JRRectView * view=[self.pool anyObject]; if (view!=nil) { [self.pool removeObject:view]; } return view; }
6、在主控制器載入檢視並實現代理方法即可
//載入所有資料 - (void) _loadSubviews{ //1 增加滾動檢視 JRMainScrollView * mainScroll=[[JRMainScrollView alloc] initWithFrame:self.view.bounds]; mainScroll.jrDataSource=self; mainScroll.jrDelegate=self; [mainScroll reloadViews]; [self.view addSubview:mainScroll]; } #pragma mark - 資料來源方法 -(NSInteger)numberOfItems{ return 132; } -(NSInteger) numberOfColumsOfRow{ return 3; } -(UIView *) mainScrollView:(JRMainScrollView *)mainScrollView itemAtIndex:(NSInteger)index{ JRRectView *cell=[mainScrollView dequeueReusedItemWithIdenty:@"test"]; if (cell==nil) { cell=[[[NSBundle mainBundle] loadNibNamed:@"rect" owner:nil options:nil] lastObject]; } cell.titleLabel.text=[NSString stringWithFormat:@"下載"]; NSString * imageName=[NSString stringWithFormat:@"%d",arc4random_uniform(20)+256]; UIImage *image=[UIImage imageNamed:imageName]; cell.image.image=image; return cell; } #pragma mark - 代理方法 //獲取高度 - (CGFloat) heightForItemAtView:(JRMainScrollView *) mainScrollView{ return 100; } //獲取寬度 - (CGFloat) widthForItemAtView:(JRMainScrollView *) mainScrollView{ return 90; } //獲取間距 - (CGFloat) mainScrollView:(JRMainScrollView *)mainScrollView spaceForItemWithType:(kJRMainScrollItemSpace)type{ if (type==kJRMainScrollItemLeftSpace) { return 20; }else if (type==kJRMainScrollItemTopSpace){ return 20; } return 20; }
相關文章
- Android 效能優化之記憶體優化Android優化記憶體
- Android記憶體優化之記憶體快取Android記憶體優化快取
- Android效能優化篇之記憶體優化--記憶體洩漏Android優化記憶體
- Android記憶體優化之圖片優化Android記憶體優化
- Android APP 記憶體優化之圖片優化AndroidAPP記憶體優化
- Android效能優化之記憶體篇Android優化記憶體
- Android 效能優化之記憶體洩漏檢測以及記憶體優化(上)Android優化記憶體
- Android 效能優化之記憶體洩漏檢測以及記憶體優化(下)Android優化記憶體
- Android 效能優化之記憶體洩漏檢測以及記憶體優化(中)Android優化記憶體
- Android 效能優化(四)之記憶體優化實戰Android優化記憶體
- Android記憶體優化Android記憶體優化
- Android 記憶體優化Android記憶體優化
- Android效能優化 - 記憶體優化Android優化記憶體
- Android效能優化(三)之記憶體管理Android優化記憶體
- Android效能優化之記憶體洩漏Android優化記憶體
- Android應用優化之記憶體概念Android優化記憶體
- Android記憶體優化之static使用篇Android記憶體優化
- Android開發優化之——對Bitmap的記憶體優化Android優化記憶體
- css 九宮格CSS
- 遊戲九宮格遊戲
- Android Note - 記憶體優化Android記憶體優化
- android 記憶體優化篇Android記憶體優化
- Android記憶體優化(一):Java記憶體區域Android記憶體優化Java
- Android學習之 記憶體管理機制與應用記憶體優化Android記憶體優化
- 淺談Android記憶體優化Android記憶體優化
- Android記憶體優化全解析Android記憶體優化
- Android記憶體優化雜談Android記憶體優化
- android,記憶體優化詳解Android記憶體優化
- Android效能優化之巧用軟引用與弱引用優化記憶體使用Android優化記憶體
- Android效能優化之常見的記憶體洩漏Android優化記憶體
- android 應用記憶體優化之OnLowMemory&OnTrimMemoryAndroid記憶體優化
- Android應用記憶體優化方式Android記憶體優化
- Android記憶體優化(三)避免可控的記憶體洩漏Android記憶體優化
- Android記憶體優化(五)詳解記憶體分析工具MATAndroid記憶體優化
- Android記憶體優化——記憶體洩露檢測分析方法Android優化記憶體洩露
- 九、Android效能優化之網路優化Android優化
- Canvas 基礎系列(零)之大轉盤九宮格刮刮卡抽獎外掛封裝Canvas封裝
- 數獨遊戲九宮格遊戲