OC一般类(简单类,组合类,继承关系的子类)的拷贝

2024-08-28 16:38

本文主要是介绍OC一般类(简单类,组合类,继承关系的子类)的拷贝,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  1、对于简单的类而言,如果想要实现拷贝操作,就需要先将这个类遵守NSCopying协议,然后在它的实现文件即.m文件中实现-(id)copyWithZone:(NSZone *)zone方法,在方法内以allocWithZone的方式分配内存,若本类实现有初始化成员方法则直接调用初始化方法并用自身成员变量作为实参对其相应的形参进行赋值 ,然后生成新的对象指针,把新的对象指针返回即可;若本身没有实现初始化成员的方法则直接调用init方法生成对象指针,然后用新的对象指针调用本类中成员的set方法并用相应的自身成员对其赋值(即“新的对象指针.成员名=成员名”),如果没有为有的成员设置set方法,就要用移动指针(—>)的方式为其赋值 即(新的对象指针->成员名=成员名)的格式为其赋值,最后返回新的成员指针即可。

2、对于组合类而言,一定要先让成员对象所属的类(被包含类)遵守NSCopying协议并实现copyWIthZone方法,然后再让自身类(包含类)遵守NSCopying协议并实现copyWithZone方法即可。在方法内调用方法上同只不过在赋值的时候多了一层,新的对象指针 .(或->) 成员对象指针.(或->) 成员类的实例变量=XXX(自身的值,调用类似)。

3、对于继承关系中的子类而言,只要它直接继承根类的父类遵守了NSCopying协议,它就也遵守了此协议。然后重写copyWithZone方法即可,详细步骤上同。

注意:(1)根据 谁创建,谁释放的内存管理原则,组合类一定要为包含类重写dealloc方法并在方法内让自己的成员对象调用一次release操作进行释放,最后再调用父类的dealloc方法([super dealloc];)。

(2)如果将同一个类的初始化过的对象二赋给已经初始化的对象一,在拷贝之前要对对象一进行一个release操作,不然会造成对象一原先的内存泄露。即在 对象一=[对象二];之前要加[对象一  release];

代码验证实例如下:

新建XYPoint类

编辑XYPoint.h如下:

//
//  XYPoint.h
//  aa
//
//  Created by apple on 15/8/8.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import <Foundation/Foundation.h>@interface XYPoint : NSObject<NSCopying>
{int x;int y;
}
@property int  x,y;
-(id)initWithX:(int) xVal andY:(int) yVal;
-(void)print;
@end
编辑XYPoint.m如下:

//
//  XYPoint.m
//  aa
//
//  Created by apple on 15/8/8.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import "XYPoint.h"@implementation XYPoint
@synthesize x,y;
-(id)initWithX:(int) xVal andY:(int) yVal
{if (self=[super init]) {x = xVal;y = yVal;}return  self;
}
-(id)copyWithZone:(NSZone *)zone
{XYPoint *newPoint=[[XYPoint allocWithZone:zone] initWithX:x andY:y];return  newPoint;
}
-(void)print{NSLog(@"x=%d  y=%d",x,y);
}
@end
新建组合类Circle

编辑Circl.h如下:

//
//  Circle.h
//  aa
//
//  Created by apple on 15/8/8.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import <Foundation/Foundation.h>
#import "XYPoint.h"
@interface Circle : NSObject<NSCopying>
{int radius;XYPoint *origin;// 定义类XYpoint的对象,作为类Circle的实例变量
}
@property int radius;
@property (nonatomic,retain) XYPoint *origin;
-(id)initWithRadius:(int)_r  andOrigin:(XYPoint *) _origin;
-(id) initWithX:(int)_x andY:(int) _y andRadius:(int) _r;
-(int) area; //计算面积方法
-(int) perimeter; //计算周长方法
-(void)print;
@end
编辑Circle.m如下:

//
//  Circle.m
//  aa
//
//  Created by apple on 15/8/8.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import "Circle.h"
@implementation Circle
@synthesize radius,origin;
-(id)initWithRadius:(int)_r  andOrigin:(XYPoint *) _origin
{if (self=[super init]) {self.origin=_origin;radius=_r;}return self;
}
-(id) initWithX:(int)_x andY:(int) _y andRadius:(int) _r
{if (self=[super init]) {self.radius=_r;XYPoint *_origin=[[[XYPoint alloc] initWithX:_x andY:_y] autorelease];self.origin=_origin;}return self;
}
-(int)area
{return radius*radius*3.14;
}
-(int)perimeter
{return radius*3.14*2;
}
-(void)print{[origin print];NSLog(@"radius=%d area=%d perimeter=%d",radius,self.area,self.perimeter);
}
-(id)copyWithZone:(NSZone *)zone
{Circle *newCircle=[[Circle allocWithZone:zone] init];newCircle->radius=radius;newCircle->origin=[origin copy];return newCircle;
}
-(void)dealloc
{[origin release];[super dealloc];
}
@end
新建Circle类的子类Cylinder

编辑Cylinder.h如下:

//
//  Cylinder.h
//  aa
//
//  Created by apple on 15/8/20.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import "Circle.h"
#import "Circle.h"
@interface Cylinder : Circle
@property int height;
-(id)initWithX:(int)_x andY:(int)_y andRadius:(int)_r andHeight:(int) _h;
-(void) print;
@end
编辑Cylinder.m如下:

//
//  Cylinder.m
//  aa
//
//  Created by apple on 15/8/20.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import "Cylinder.h"
@implementation Cylinder
-(id)initWithX:(int)_x andY:(int)_y andRadius:(int)_r andHeight:(int) _h
{if (self=[super initWithX:_x andY:_y andRadius:_r]) {_height=_h;}return self;
}
-(void) print
{NSLog(@"height= %d",self.height);[super print];NSLog(@"volume= %d",self.area*_height);
}
-(id)copyWithZone:(NSZone *)zone
{Cylinder *newCy=[[Cylinder allocWithZone:zone] initWithX:origin.x andY:origin.y andRadius:radius andHeight:_height];return newCy;
}
@end
在main.m中调用如下:

//
//  main.m
//  aa
//
//  Created by apple on 15/8/8.
//  Copyright (c) 2015年 liu. All rights reserved.
//#import <Foundation/Foundation.h>
#import "Cylinder.h"
void Test1()
{//验证单个类XYPoint类的复制操作NSLog(@"简单类的拷贝过程如下:");XYPoint * p=[[XYPoint alloc] initWithX:1 andY:2];XYPoint *p1=[[XYPoint alloc] initWithX:3 andY:4];[p print];[p1 print];[p release];p=[p1 copy];[p print];[p1 print];[p release];[p1 release];NSLog(@"---------------------------------");
}
void Test2()
{ //验证组合类Circle类的复制操作NSLog(@"组合类的拷贝过程如下:");XYPoint *pp1 = [[[XYPoint alloc] initWithX:5 andY:6] autorelease];Circle *c1= [[Circle alloc] initWithRadius:7 andOrigin:pp1] ;XYPoint *pp2 = [[[XYPoint alloc] initWithX:8 andY:9] autorelease];Circle *c2 = [[[Circle alloc] initWithRadius:10 andOrigin:pp2] autorelease];[c1 print];[c2 print];[c1 release];c1=[c2 copy];[c1 print];[c2 print];[c1 release];NSLog(@"---------------------------------");
}
void Test3()
{NSLog(@"继承中子类的拷贝过程如下:");Cylinder *cy1=[[Cylinder alloc] initWithX:7 andY:8 andRadius:9 andHeight:10] ;Cylinder *cy2=[[[Cylinder alloc] initWithX:11 andY:12 andRadius:13 andHeight:14] autorelease];[cy1 print];[cy2 print];[cy1 release];cy1=[cy2 copy];[cy1 print];[cy2 print];[cy1 release];NSLog(@"---------------------------------");
}
int main(int argc, const char * argv[])
{@autoreleasepool {Test1();Test2();Test3();}return 0;
}

运行结果如下:








这篇关于OC一般类(简单类,组合类,继承关系的子类)的拷贝的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115312

相关文章

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

hdu4869(逆元+求组合数)

//输入n,m,n表示翻牌的次数,m表示牌的数目,求经过n次操作后共有几种状态#include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdlib.h>#includ

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu

uva 10130 简单背包

题意: 背包和 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <queue>#include <map>

POJ1269 判断2条直线的位置关系

题目大意:给两个点能够确定一条直线,题目给出两条直线(由4个点确定),要求判断出这两条直线的关系:平行,同线,相交。如果相交还要求出交点坐标。 解题思路: 先判断两条直线p1p2, q1q2是否共线, 如果不是,再判断 直线 是否平行, 如果还不是, 则两直线相交。  判断共线:  p1p2q1 共线 且 p1p2q2 共线 ,共线用叉乘为 0  来判断,  判断 平行:  p1p

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

JavaSE——封装、继承和多态

1. 封装 1.1 概念      面向对象程序三大特性:封装、继承、多态 。而类和对象阶段,主要研究的就是封装特性。何为封装呢?简单来说就是套壳屏蔽细节 。     比如:对于电脑这样一个复杂的设备,提供给用户的就只是:开关机、通过键盘输入,显示器, USB 插孔等,让用户来和计算机进行交互,完成日常事务。但实际上:电脑真正工作的却是CPU 、显卡、内存等一些硬件元件。