Day17.protocol协议和NSString字符串的用法
1,protocol基本概念
#import <Foundation/Foundation.h>
@protocol SportProtocol <NSObject>
// 方法声明列表
- (void)playFootball;
- (void)playBasketball;
- (void)playBaseball;
@end
2,protocol注意点
- SportProtocol.h
#import <Foundation/Foundation.h>
@protocol SportProtocol <NSObject>
// 1.协议只能声明方法, 不能声明属性
//{
// int _age;
//}
// 方法声明列表
// 注意: 如果没有使用任何关键字修饰协议中的方法, 那么该方法默认就是required的
@required
// 如果协议中的方法是@required的, 而遵守协议的类又没有实现该方法, 那么会报一个警告
- (void)playFootball;
@optional // 可选
// 如果协议中的方法是@optional的, 而遵守协议的类又没有实现该方法, 那么不会报警告
- (void)playBasketball;
- (void)playBaseball;
// 注意:@required和@optional仅仅使用程序员之间交流, 并不能严格的控制某一个遵守该协议的类必须要实现该方法, 因为即便不是实现也不会报错, 只会报一个警告
@end
- CrazySportProtocol.h
#import <Foundation/Foundation.h>
#import "SportProtocol.h"
// 4.OC中的协议又可以遵守其它协议, 只要一个协议遵守了其它协议, 那么这个协议中就会自动包含其它协议的声明
@protocol CrazySportProtocol <SportProtocol>
// 跳楼
- (void)jumping;
@end
- StudyProtocol.h
#import <Foundation/Foundation.h>
@protocol StudyProtocol <NSObject>
- (void)study;
@end
- Person.h
#import <Foundation/Foundation.h>
#import "SportProtocol.h"
#import "StudyProtocol.h"
// 3.在OC中一个类可以遵守1个或多个协议
// 注意: OC中的类只能有一个父类, 也就是说OC只有单继承
@interface Person : NSObject<SportProtocol, StudyProtocol>
@end
- Sutdent.h
#import "Person.h"
// 2.父类遵守了某个协议, 那么子类也会自动遵守这个协议
@interface Sutdent : Person
@end
3,协议的应用场景1-类型限定
- 首先定义一个协议WifeCondition.h
#import <Foundation/Foundation.h>
@protocol WifeCondition <NSObject>
// 会做饭
- (void)cooking;
// 会洗衣服
- (void)washing;
// 有一份好工作
- (void)job;
@end
- 定义一个Wife.h类
#import <Foundation/Foundation.h>
#import "WifeCondition.h"
@interface Wife : NSObject <WifeCondition>
@end
#import "Wife.h"
@implementation Wife
// 会做饭
- (void)cooking
{
NSLog(@"%s", __func__);
}
// 会洗衣服
- (void)washing
{
NSLog(@"%s", __func__);
}
// 有一份好工作
- (void)job
{
NSLog(@"%s", __func__);
}
@end
- main.m类
#import <Foundation/Foundation.h>
#import "Person.h"
#import "Wife.h"
#import "WifeCondition.h"
int main(int argc, const char * argv[]) {
Person *p = [Person new];
// 1.协议的第一个应用场景, 可以将协议写在数据类型的右边, 明确的标注如果想给该变量赋值, 那么该对象必须遵守某个协议
// Wife<WifeCondition> *w = [Wife new];
Wife *w = [Wife new];
p.wife = w;
[p show];
return 0;
}
- 定义Person.h类
#import <Foundation/Foundation.h>
#import "Wife.h"
@interface Person : NSObject
// 媳妇
// 注意: 记住一点, 类型限定是写在数据类型的右边的
@property (nonatomic, strong) Wife<WifeCondition> *wife;
- (void)show;
@end
#import "Person.h"
@implementation Person
- (void)show
{
// 注意: 虽然在接受某一个对象的时候, 对这个对象进行了类型限定(限定它必须实现某个协议), 但是并不意味着这个对象就真正的实现了该方法. 所以每次在调用对象的协议方法时应该进行一次验证
if ([self.wife respondsToSelector:@selector(cooking)]) {
[self.wife cooking];
}
if ([self.wife respondsToSelector:@selector(washing)]) {
[self.wife washing];
}
if ([self.wife respondsToSelector:@selector(job)]) {
[self.wife job];
}
}
@end
4,协议的应用场景2-代理设计模式
- 首先定义一个BabyProtocol协议
#import <Foundation/Foundation.h>
@class Baby;
@protocol BabyProtocol <NSObject>
// 喂婴儿吃东西
- (void)feedFood:(Baby *)baby;
// 哄婴儿睡觉
- (void)hongBaby:(Baby *)baby;
@end
- 重点Baby.h类
#import <Foundation/Foundation.h>
#import "BabyProtocol.h"
@class Nanny, Studnet;
/*
什么是代理设计模式
心相印和超市
超市就是心相印的代理, 它可以帮助心相印卖纸巾
也就是说, 代理也帮助另外一个类做一些事情
代理设计模式的应用场景
1.当A对象想监听B对象的一些变化时, 可以使用代理设计模式
保姆想监听婴儿的变化, 那么保姆就可以成为婴儿的代理, 当婴儿发生变化之后保姆就可以监听到
2.当B对象发生一些事情, 想通知A对象的时候, 可以使用代理设计模式
婴儿想通知保姆, 那么就可以 让保姆成为婴儿的代理, 只要保姆成为婴儿的代理, 以后婴儿发生变化就可以通知保姆
3.当对象A无法处理某些行为的时候,想让对象B帮忙处理(让对象B成为对象A的代理对象)
婴儿无法自己吃东西, 也无法自己入睡, 所以可以让保姆帮忙处理. 只要让保姆成为婴儿的代理就可以帮婴儿喂它吃东西和哄他睡觉
*/
@interface Baby : NSObject
// 饥饿程度
@property (nonatomic, assign) int hungry;
// 睡意
@property (nonatomic, assign) int sleep;
// 吃东西
- (void)food;
// 困了
- (void)sleepy;
// 让婴儿拥有一个保姆
//@property (nonatomic, strong) Nanny *nanny;
//@property (nonatomic, strong) Studnet *stu;
// 如果使用id类型来接收保姆, 如果将来换保姆了, 婴儿类不用修改代码
@property (nonatomic, strong) id<BabyProtocol> nanny;
@end
#import "Baby.h"
#import "Nanny.h"
@implementation Baby
- (void)food
{
self.hungry += 5;
NSLog(@"要吃东西 %i", self.hungry);
// 通知保姆
if ([self.nanny respondsToSelector:@selector(feedFood:)]) {
[self.nanny feedFood:self];
}
}
- (void)sleepy
{
self.sleep += 5;
NSLog(@"困了 %i", self.sleep);
// 通知保姆
if ([self.nanny respondsToSelector:@selector(hongBaby:)]) {
[self.nanny hongBaby:self];
}
}
@end
- 代理类 和Nanny.h都一样的实现方式
#import <Foundation/Foundation.h>
#import "BabyProtocol.h"
@class Baby;
@interface Nanny : NSObject <BabyProtocol>
/*
// 喂婴儿吃东西
- (void)feedFood:(Baby *)baby;
// 哄婴儿睡觉
- (void)hongBaby:(Baby *)baby;
*/
@end
#import "Nanny.h"
#import "Baby.h"
@implementation Nanny
- (void)feedFood:(Baby *)baby
{
baby.hungry -= 10;
NSLog(@"喂婴儿吃东西 %i", baby.hungry);
}
- (void)hongBaby:(Baby *)baby
{
baby.sleep += 10;
NSLog(@"哄婴儿睡觉 %i", baby.sleep);
}
@end
- main.m类
int main(int argc, const char * argv[]) {
// 1.创建婴儿
Baby *b = [Baby new];
// 2.创建保姆
// Nanny *n = [Nanny new];
// 3.将保姆赋值给婴儿呢
// b.nanny = n;
// 4.换保姆
// Studnet *stu = [Studnet new];
// b.nanny = stu;
// 5.再次换保姆
// Teacher *tea = [Teacher new];
// b.nanny = tea;
Dog *d = [Dog new];
b.nanny = d;
//4.婴儿发飙
[b food];
[b sleepy];
return 0;
}
5,代理设计模式练习
- main类
#import <Foundation/Foundation.h>
#import "Person.h"
#import "LinkHome.h"
#import "LoveHome.h"
int main(int argc, const char * argv[]) {
/*
用代理实现学生找房子, 不具备找房子的能力
所以学生可以找另一个对象来帮它找房子, 那么另一个对象就是学生的代理
*/
Person *p = [Person new];
LinkHome *lh = [LinkHome new];
p.delegate = lh;
// LoveHome *lh = [LoveHome new];
// p.delegete = lh;
[p findHourse];
return 0;
}
- StudentProtocol.h协议类
#import <Foundation/Foundation.h>
@protocol StudentProtocol <NSObject>
// 帮学生找房子
- (void)studentFindHourse;
@end
- Person.h类
#import <Foundation/Foundation.h>
//#import "StudentProtocol.h"
/*
协议的编写规范:
1.一般情况下, 当前协议属于谁, 我们就将协议定义到谁的头文件中
2.协议的名称一般以它属于的那个类的类名开头, 后面跟上protocol或者delegate
3.协议中的方法名称一般以协议的名称protocol之前的作为开头
4.一般情况下协议中的方法会将触发该协议的对象传递出去
5.一般情况下一个类中的代理属于的名称叫做 delegate
6.当某一个类要成为另外一个类的代理的时候,
一般情况下在.h中用@protocol 协议名称;告诉当前类 这是一个协议.
在.m中用#import真正的导入一个协议的声明
*/
@class Person;
@protocol PersonProtocol <NSObject>
- (void)personFindHourse:(Person *)person;
@end
@interface Person : NSObject
//@property (nonatomic, strong) id<StudentProtocol> delegete;
@property (nonatomic, strong) id<PersonProtocol> delegate;
- (void)findHourse;
@end
#import "Person.h"
@implementation Person
- (void)findHourse
{
NSLog(@"学生想找房子");
// 通知代理帮他找房子
if ([self.delegate respondsToSelector:@selector(personFindHourse:)]) {
[self.delegate personFindHourse:self];
}
}
@end
- LinkHome.h类
#import <Foundation/Foundation.h>
//#import "Person.h"
@protocol PersonProtocol;
@interface LinkHome : NSObject <PersonProtocol>
@end
#import "LinkHome.h"
#import "Person.h"
@implementation LinkHome
- (void)personFindHourse:(Person *)person
{
NSLog(@"%s", __func__);
}
@end
NSString和NSMutableString的用法
1,NSString的基本概念
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
//如何创建字符串对象
/*
通过不同的方式创建字符串,字符串对象储存的位置也不一样
>如果是通过字符串常量创建,那么字符串对象存储在常量区中
>如果是通过alloc initWithFormat/stringWithFormat创建,那么字符串对象存储在堆区中
而且需要注意:
>不同的平台存储的方式也不一样,如果是Mac平台系统会自动对字符串对象进行优化,但是如果是iOS平台就是两个对象
>不同的编译器存储的方式也不一样,如果是Xcode6以下并且是在iOS平台,那么每次alloc都会创建一个新的对象,如果是在Xcode6以上那么alloc多次指向同一块存储空间
*/
//1.通过字符串常量创建
//注意:如果是通过字符串常量创建对象,并且字符串常量的内容一致,那么如果创建多个字符串对象,多个对象指向同一块存储空间
NSString *str1 = @"lnj";
NSString *str11 = @"lnj";
NSLog(@"str1 = %p, str11 = %p", str1 ,str11);
//2.通过alloc init创建
//只要调用alloc就会在堆内存中开辟一块存储空间
NSString *str2 = [[NSString alloc]initWithFormat:@"lmj"];
NSString *str22 = [[NSString alloc]initWithFormat:@"lmj"];
NSLog(@"str2 = %p, str22 = %p", str2, str22);
//3.通过类工厂方法创建/ stringWithFormat
//内部其实就是封装了alloc init
NSString *str3 = [NSString stringWithFormat:@"zs"];
NSString *str33= [NSString stringWithFormat:@"zs"];
/*
注意:一般情况下,只要是通过alloc或者类工厂方法创建的对象,每次都会在堆内存中开辟一块新的存储空间
但是如果是通过alloc的initWithString方法除外,因为这个方法是通过copy返回一个字符串对象给我们
而copy又分为深拷贝和浅拷贝,如果是深拷贝会创建一个新的对象,如果是浅拷贝不会创建一个新的对象,而是直接返回被拷贝的对象的地址给我们
*/
NSString *str4 = [[NSString alloc]initWithString:@"ls"];
NSString *str44 = [[NSString alloc]initWithString:@"ls"];
NSLog(@"str4 = %p, str44 = %p", str4, str44);
return 0;
}
2,字符串的读写
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
/*
file: 文件路径,
encoding: 编码英文 iOS-5988-1 中文 GBK GBK2312 , 一般情况填写UTF-8
error: 如果读取错误, 会将错误信息保存到error中 ,如果读取正确, 就没有error = nil
注意: 以后在OC方法中但凡看到XXXofFile的方法, 传递的一定是全路径(绝对路径)
*/
NSString *path = @"/Users/xiaomage/Desktop/课堂共享/day17/note/lnj.txt";
NSError *error = nil;
// 从文件中读取字符串
/*
NSString *str = [NSString stringWithContentsOfFile:path encoding:NSUTF8StringEncoding error:&error];
if (error == nil) {
NSLog(@"str = %@", str);
}else
{
NSLog(@"error = %@", [error localizedDescription]);
}
*/
/*
// 将字符串写入到文件中
NSString *str = @"iOS0601基础班";
// atomically 如果传入YES, 字符串写入文件的过程中如果没有写完, 那么不会生成文件
// 如果传入NO, 字符串写入文件的过程中如果没有写完, 会生成文件
NSString *path2 = @"/Users/xiaomage/Desktop/abc.txt";
BOOL flag = [str writeToFile:path2 atomically:YES encoding:NSUTF8StringEncoding error:nil];
NSLog(@"flag = %i", flag);
*/
return 0;
}
- URL路径的path
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
//1.文件读取
/*
//1.创建URL
//协议头 + 主机地址 + 文件路径
// NSString *path = @"file://192.168.199.199/Users/NJ-Lee/Desktop/lnj.txt";
// NSString *path = @"http://www.baidu.com";
//注意:如果加载的资源是本机上的资源,那么URL中的主机地址可以省略
//虽然主机地址可以省略,但是需要注意,文件路劲中最前面的/不能省略,文件路径最前面的/代表根路径
// NSString *path = @"file:///Users/NJ-Lee/Desktop/lnj.txt";
// NSURL *url = [NSURL URLWithString:path];
// NSString *path = @"file:///Users/NJ-Lee/Desktop/lnj.txt";
//注意:如果是通过NSURL的fileURLWithPath:方法创建URL,那么系统会自动给我们传入的字符串添加协议头(file://),所以字符串中不需要再写file://
// 注意:开发中一 般情况下,如果是访问本机的资源,创建URL的时候,建议使用fileURLWithPath方法创建
//因为url不支持中文,如果URL中包含中文,那么无法访问;但是如果是通过fileURLWithString方法创建URL,哪怕URL中包含中文也可以进行访问,系统内部会自动对URL中包含的中文进行处理
// NSURL *url = [NSURL fileURLWithPath:path];
NSString *path = @"file:///Users/NJ-Lee/Desktop/未命名文件夹/lnj.txt";
//如果URL中包含中文,又非不通过fileURLWithPath创建,也可以破
//如果想破就必须在创建URL之前先对字符串中的中文进行处理,进行百分号编码
NSLog(@"处理前:%@", path);
path = [path stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
NSLog(@"处理后:%@", path);
NSURL *url = [NSURL URLWithString:path];
NSLog(@"url = %@", url);
//2.根据URL加载文件中的字符串
NSString *str = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:nil];
NSLog(@"str = %@", str);
*/
//2.文件写入
NSString *str = @"lnj";
// NSString *path = @"file:///Users/NJ-Lee/Desktop/未命名文件夹/abc.txt";
// path = [path stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
// NSURL *url = [NSURL URLWithString:path];
NSString *path = @"/Users/NJ-Lee/Desktop/未命名文件夹/abc.txt";
NSURL *url = [NSURL fileURLWithPath:path];
[str writeToURL:url atomically:YES encoding:NSUTF8StringEncoding error:nil];
//注意点:如果多次往同一个文件中写入内容,那么后一次的会覆盖前一次的
NSString *str2 = @"xxoo";
[str2 writeToURL:url atomically:YES encoding:NSUTF8StringEncoding error:nil];
return 0;
}
3,字符串的比较
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
NSString *str1 = @"abc";
NSString *str2 = @"ABC";
/*
// 比较两个字符串的"内容"是否相同
BOOL flag = [str1 isEqualToString:str2];
NSLog(@"flag = %i", flag);
// 下面这个方法, 是比较两个字符串的"地址"是否相同
flag = (str1 == str2);
NSLog(@"flag = %i", flag);
*/
// 比较字符串的大小
/*
// NSOrderedAscending 前面的小于后面的
// NSOrderedSame, 两个字符串相等
// NSOrderedDescending 前面的大于后面的
switch ([str1 compare:str2]) {
case NSOrderedAscending:
NSLog(@"str1小于str2");
break;
case NSOrderedSame:
NSLog(@"str1等于str2");
break;
case NSOrderedDescending:
NSLog(@"str1大于str2");
break;
default:
break;
}
*/
/*
// 忽略大小写进行比较
switch ([str1 caseInsensitiveCompare:str2]) {
case NSOrderedAscending:
NSLog(@"str1小于str2");
break;
case NSOrderedSame:
NSLog(@"str1等于str2");
break;
case NSOrderedDescending:
NSLog(@"str1大于str2");
break;
default:
break;
}
*/
return 0;
}
4,字符串搜索
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
// NSString *str = @"http://www.520it.com/img/lnj.gif";
// 1.判断是否以什么开头
/*
// 本质就是从字符串的第一个字符开始匹配, 只要不匹配就返回NO
if ([str hasPrefix:@"http://"]) {
NSLog(@"是一个URL");
}else
{
NSLog(@"不是一个URL");
}
*/
// 2.判断是否以什么结尾
/*
// 本质就是从字符串的最后一个字符开始匹配, 只要不匹配就返回NO
if ([str hasSuffix:@".gif"]) {
NSLog(@"动态图片");
}else{
NSLog(@"不是动态图片");
}
*/
// 3.判断字符串中是否包含520it.com
/*
NSString *str = @"abcd";
// 只要str中包含该字符串, 那么就会返回该字符串在str中的起始位置以及该字符串的长度
// location从0开始 , length从1开始
// 如果str中没有需要查找的字符串, 那么返回的range的length=0, location = NSNotFound
NSRange range = [str rangeOfString:@"lnj"];
// if (range.location == NSNotFound) {
if (range.length == 0){
NSLog(@"str中没有需要查找的字符串");
}else{
NSLog(@"str中有需要查找的字符串");
NSLog(@"location = %lu, length = %lu", range.location, range.length);
}
*/
return 0;
}
5,字符串截取
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
NSString *str = @"<head>小码哥</head>";
/*
// NSRange : 位置/长度
// NSRange range = {6, 3};
// NSRange range;
// range.location = 6;
// range.length = 3;
// 只要是OC提供的结构体, 一般都可以使用NSMakeXXX来创建
// NSRange range = NSMakeRange(6, 3);
*/
/*
// 1.动态获取截取的起始位置 值:6
NSUInteger location = [str rangeOfString:@">"].location + 1;
// 2.动态获取截取的长度 值:3
// 注意:rangeOfString是从左至右的开始查找, 只要找到就不找了
// NSUInteger length = [str rangeOfString:@"<" options:NSBackwardsSearch].location - location;
NSUInteger length = [str rangeOfString:@"</"].location - location;
NSLog(@"location = %lu, length = %lu", location, length);
NSRange range = NSMakeRange(location, length);
NSString *newStr = [str substringWithRange:range];
NSLog(@"str = %@", str);
NSLog(@"newStr = %@", newStr);
// NSString *temp = @"abcdefa";
// NSRange range =[temp rangeOfString:@"a" options:NSBackwardsSearch];
// NSLog(@"%lu", range.location);
*/
// 从什么地方开始截取, 一直截取到最后
// NSString *newStr = [str substringFromIndex:6];
// NSLog(@"newStr = %@", newStr);
// 从开头开始截取, 一直截取到什么位置
// NSString *newStr = [str substringToIndex:6];
// NSLog(@"newStr = %@", newStr);
/*
<head>小码哥</head> --> 小码哥</head> --> 小码哥
<head>小码哥</head> --> <head>小码哥 --> 小码哥
*/
NSLog(@"str = %@", str);
NSUInteger location = [str rangeOfString:@">"].location + 1;
NSString *newStr = [str substringFromIndex:location];
NSLog(@"newStr = %@", newStr);
location = [newStr rangeOfString:@"</"].location;
// 改变了指针的指向, 并不是修改了原来的字符串
newStr = [newStr substringToIndex:location];
NSLog(@"newStr = %@", newStr);
return 0;
}
6,字符串替换
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
/*
// 需求: 将&符号替换为/
NSString *str = @"http:&&www.520it.com&img&lnj.gif";
// OccurrencesOfString: 要替换谁
// withString: 用谁替换
NSString *newStr = [str stringByReplacingOccurrencesOfString:@"&" withString:@"/"];
NSLog(@"newStr = %@", newStr);
*/
/*
// 1.去除空格 2.将&替换为/
NSString *str = @" http: &&www. 520it.com &img&lnj.gif ";
// 1.去除空格
NSString *newStr = [str stringByReplacingOccurrencesOfString:@" " withString:@""];
NSLog(@"newStr = |%@|", newStr);
NSString *newStr2 = [newStr stringByReplacingOccurrencesOfString:@"&" withString:@"/"];
NSLog(@"newStr2 = |%@|", newStr2);
*/
// 3.替换首尾 set的用法 专门去除首尾的空格,大,小写字母,*号等等
// NSString *str = @" http:&&www.520it.com&img&lnj.gif ";
NSString *str = @"HTTP://www.520it.com/img/LNJ.GIF";
// NSCharacterSet *set = [NSCharacterSet whitespaceCharacterSet]; 去首尾的空格
// NSString *newStr = [str stringByTrimmingCharactersInSet:set];
// 去除首尾的大写字母
NSCharacterSet *set = [NSCharacterSet uppercaseLetterCharacterSet];
NSString *newStr = [str stringByTrimmingCharactersInSet:set];
NSLog(@"newStr = |%@|", newStr);
return 0;
}
7,字符串与路径
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
NSString *str = @"User/lnj/Desktop/lnj.txt.jpg";
// 1.判断是否是绝对路径
/*
// 其实本质就是判断字符串是否以/开头
if([str isAbsolutePath])
{
NSLog(@"是绝对路径");
}else{
NSLog(@"不是绝对路径");
}
*/
// 2.获取文件路径中的最后一个目录
// 本质就是获取路径中最后一个/后面的内容
/*
NSString *newStr = [str lastPathComponent];
NSLog(@"%@", newStr);
*/
// 3.删除文件路径中的最后一个目录
/*
// 本质就是删除最后一个/后面的内容, 包括/也会被删除
NSString *newStr = [str stringByDeletingLastPathComponent];
NSLog(@"%@", newStr);
*/
// 4.给文件路径添加一个目录
/*
// 本质就是在字符串的末尾加上一个/ 和指定的内容
// 注意: 如果路径后面已经有了/, 那么就不会添加了
// 如果路径后面有多个/, 那么会自动删除多余的/, 只保留一个
NSString *newStr = [str stringByAppendingPathComponent:@"xmg"];
NSLog(@"%@", newStr);
*/
// 5.获取路径中文件的扩展名
/*
// 本质就是从字符串的末尾开始查找., 截取第一个.后面的内容
NSString *newStr = [str pathExtension];
NSLog(@"%@", newStr);
*/
// 6.删除路径中文件的扩展名
/*
// 本质就是从字符串的末尾开始查找.,删除第一个.和.后面的内容
NSString *newStr = [str stringByDeletingPathExtension];
NSLog(@"%@", newStr);
*/
// 7.给文件路径添加一个扩展名
// 本质就是在字符串的末尾加上一个.和指定的内容
NSString *newStr = [str stringByAppendingPathExtension:@"jpg"];
NSLog(@"%@", newStr);
return 0;
}
8,字符串的转换
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
// NSString *str = @"abc";
// 1.将字符串转换为大写
/*
NSString *newStr = [str uppercaseString];
NSLog(@"%@", newStr);
*/
// 2.将字符串转换为小写
/*
NSString *newStr2 = [newStr lowercaseString];
NSLog(@"%@", newStr2);
// htpp://www.520it.com/img/lnj.GIF;
*/
// 3.将字符串的首字符转换为大写
/*
NSString *newStr = [str capitalizedString];
NSLog(@"%@", newStr);
*/
// 4.字符串与基本数据类型的转换
/*
NSString *str1 = @"110";
NSString *str2 = @"120";
// str1 + str2; // 错误
int value1 = [str1 intValue];
int value2 = [str2 intValue];
NSLog(@"sum = %i", value1 + value2);
// 注意: 如果不是int,double,float,bool,integer,longlong这些类型就不要乱用
NSString *str3 = @"abc";
int value3 = [str3 intValue];
NSLog(@"value3 = %i", value3);
*/
// 5.C语言字符串和OC字符串之间的转换
/*
char *cStr = "lnj";
NSString *str = [NSString stringWithUTF8String:cStr];
NSLog(@"str = %@", str);
NSString *newStr = @"lmj";
const char *cStr2 = [newStr UTF8String];
NSLog(@"cStr2 = %s", cStr2);
*/
return 0;
}
9,NSMutableString基本概念
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
/*
NSString *str = @"lnj"; // 一开始str指向@"lnj"对应的内存
str = @"lmj"; // 修改了str指针的指向, 让它指向@"lmj"对应的内存
NSString *newStr = [str stringByReplacingOccurrencesOfString:@"l" withString:@"X"];
NSLog(@"%@", newStr);
*/
// 创建一个空的字符串
NSMutableString *str = [NSMutableString string];
NSLog(@"修改前: %@", str);
[str appendString:@"lnj"];
NSLog(@"修改后: %@", str);
NSMutableString *strM = [[NSMutableString alloc] init];
// strM = [NSMutableString alloc] initWithFormat:<#(NSString *), ...#>
// strM = [NSMutableString stringWithFormat:<#(NSString *), ...#>]
return 0;
}
10,NSMutableString常用方法
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
NSMutableString *strM = [NSMutableString stringWithFormat:@"www.520it.com.520"];
// 1.在字符串后面添加/image
/*
[strM appendString:@"/image"];
// [strM appendFormat:@"/age is %i", 10];
NSLog(@"strM = %@", strM);
*/
// 2.删除字符串中的520
/*
// 技巧: 在开发中, 我们经常利用rangeOfString和deleteCharactersInRange方法配合起来删除指定的字符串
// 2.1先查找出520在字符串中的位置
NSRange range = [strM rangeOfString:@"520"];
// 2.2删除520
[strM deleteCharactersInRange:range];
NSLog(@"strM = %@", strM);
*/
// 3.在520前面插入love这个单词
/*
// insertString : 需要插入的字符串
// atIndex: 从哪里开始插入
NSRange range = [strM rangeOfString:@"520"];
[strM insertString:@"love" atIndex:range.location];
NSLog(@"strM = %@", strM);
*/
// 4.要求将字符串中的520替换为530
// 注意: 如果是调用NSString的字符串替换方法, 不会修改原有字符串, 而是生成一个新的字符串
// NSString *newStr =[strM stringByReplacingOccurrencesOfString:@"520" withString:@"530"];
// 注意: 一般情况下OC方法要求传入一个参数如果没有*, 大部分都是枚举
// 一般情况下如果不想使用枚举的值, 可以传入0, 代表按照系统默认的方式处理
// OccurrencesOfString: 需要替换的字符串
// withString: 用什么替换
// options: 替换时的搜索方式
// range: 搜索的范围
// 返回值: 代表替换了多少个字符串
NSUInteger count = [strM replaceOccurrencesOfString:@"520" withString:@"530" options:0 range:NSMakeRange(0, strM.length)];
NSLog(@"strM = %@", strM);
NSLog(@"count = %lu", count);
// NSLog(@"newStr = %@", newStr);
return 0;
}
11,NSMutableString的练习
#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
/*
需求: 将3个520it拼接在一起, 中间用空格隔开
520it 520it 520it
*/
NSString *subStr = @"520it";
/*
// 520it-
NSString *newStr = [subStr stringByAppendingString:@" "];
// 520it-520it
newStr = [newStr stringByAppendingString:subStr];
// 520it-520it-
newStr = [newStr stringByAppendingString:@" "];
// 520it-520-520it
newStr = [newStr stringByAppendingString:subStr];
*/
/*
// 注意: 在开发中如果需要对字符串进行频繁的操作, 不要使用不可变的字符串
NSString *newStr = [subStr stringByAppendingString:@" "];;
for (int i = 0; i < 2; ++i) {
newStr = [newStr stringByAppendingString:subStr];
newStr = [newStr stringByAppendingString:@" "];
}
// 多了一个空格,第一种做法是替换最后一个空格
// 第二种做法是去掉首尾的空格的set来做
// newStr = [newStr stringByReplacingCharactersInRange:NSMakeRange(newStr.length -1 , 1) withString:@""];
newStr = [newStr stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
NSLog(@"newStr = |%@|", newStr);
*/
// 创建一个空得字符串
NSMutableString *strM = [NSMutableString string];
for (int i = 0; i < 3; ++i) {
// 1.添加一个520it
[strM appendString:subStr];
// 2.添加一个空格
[strM appendString:@" "];
}
[strM deleteCharactersInRange:NSMakeRange(strM.length - 1, 1)];
NSLog(@"strM = |%@|", strM);
return 0;
}