iOS练手项目知识点汇总

news/2024/7/20 22:38:38 标签: ios, cocoa, macos

基础理解篇

Objective-C是一种面向对象的编程语言,它支持元编程。元编程是指编写程序来生成或操纵其他程序的技术。
Objective-C中,元编程可以使用Objective-C的动态特性来实现。例如可以使用Objective-C的运行时函数来动态地创建类、添加属性和方法等等。

#import <Foundation/Foundation.h>

@interface MyClass : NSObject
@property (nonatomic, strong) NSString *name;
- (void)printName;
@end

@implementation MyClass
- (void)printName {
    NSLog(@"%@", self.name);
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MyClass *myObj = [[MyClass alloc] init];
        [myObj printName];
        
        // 动态添加属性和方法
        [[myObj class] addMethod:@selector(sayHello:) withSignature:@"v"];
        [[myObj class] addProperty:@"age" withType:@integerValue withAttribute:NONE];
        
        // 动态调用方法
        ((void (*)(id, SEL))[[myObj class] performSelector:@selector(sayHello:) withObject:nil]))(myObj, @selector(sayHello:));
        
        // 动态添加属性
        ((void (*)(id, SEL, id))[[myObj class] performSelector:@selector(setProperty:forKey:) withObject:[NSNumber numberWithInt:100] forKey:@"age"])(myObj, @selector(setProperty:forKey:), @100);
        
        // 动态删除属性和方法
        [[myObj class] removeMethod:@selector(sayHello:)];
        [[myObj class] removeProperty:@"age"];
    }
    return 0;
}

Objective-C是一种动态编程语言,它将静态语言在编译和链接时需要做的一些事情给延后到运行时执行。例如方法的调用,只有在程序执行的时候,才能具体定位到哪个类的哪个方法。这就需要一个运行时库,就是Runtime。
Runtime机制是指Objective-C运行时系统,它提供了一些类和方法来帮助我们动态地创建、修改和访问对象。其中最重要的是objc_msgSend函数,它可以用来发送消息并调用对象的方法。除此之外,还有很多其他的类和方法可以用来实现动态行为,比如class_addMethod、method_exchangeImplementations等等。

推荐阅读:什么是元编程(meta-promgramming)?

工程知识篇

1.Xcode创建工程

Xcode创建工程:iOS–>App–>填写工程名选择语言(OC\Swift)–>选择存放路径(一般放在User目录下,访达–>前往–>个人可以打开用户个人文件夹 。理由忘了,欢迎评论区大佬解答)

2.创建工程后AppDelegate 和 ViewController的说明

AppDelegate是应用程序委托对象,其父类是UIResponder类(继承关系),并实现UIApplicationDelegate委托协议.
[UIResponder:实现应用程序的处理响应事件的能力]
[UIApplicationDelegate委托协议使AppDelegate成为应用程序的委托对象,这种对象能够响应应用程序的生命周期]
(生命周期在程序运行的不同阶段进行回调)

//AppDelegate.h

#import <UIKit/UIKit.h>

@interface AppDelegate: UIResponder <UIApplicationDelegate>

@property (strong,nonatomic) UIWindow* window; //ios程序视图中只有一个UIWindow

@end

//AppDelegate.m

#import "AppDelegate.h"

@interface AppDelegate()

@end

@implementation AppDelegate

- (BOOL)application:(UIApplication *)application didFinishLauchingWithOptions:(NSDictionary *)launchOptions{return YES;} 
//应用启动并进行初始化时会调用该方法并发出通知UIApplicationDidFinishLauchingNotification.这个阶段会实例化根视图控制器

- (void)applicationWillResignActive:(UIApplication *)application{} 
//应用从活动状态进入非活动状态时调用该方法并发出通知UIApplicationWillResignActiveNotification

- (void)applicationDidEnterBackground:(UIApplication *)application{} 
//应用进入后台时候调用该方法并发出通知UIApplicationDidEnterBackgroundNotification

- (void)applicationWillEnterForeground:(UIApplication *)application{} 
//应用进入前台时候但未处于活跃状态时候调用该方法并发出通知UIApplicationWillEnterForegroundNotification

- (void)applicationDidBecomeActive:(UIApplication *)application{}

- (void)applicationWillTerminate:(UIApplication *)application{} 
//应用被终止时候调用该方法并发出通知UIApplicationWillTerminateNotification,但内存清除时除外

@end

作者:星辰_入海
链接:https://www.jianshu.com/p/859d11e1e2fd

关于ViewController更详细的说明可参考:viewController详解
文章备用链接:https://www.pianshen.com/article/13491799244/#google_vignette
重点摘录如下:(流程图有些没仔细描述、推荐看原文)

1. view的生命周期

在这里插入图片描述
当一个视图控制器被创建,并在屏幕上显示的时候。 代码的执行顺序
1、 alloc 创建对象,分配空间
2、init (initWithNibName) 初始化对象,初始化数据
3、loadView 从nib载入视图 ,通常这一步不需要去干涉。除非你没有使用xib文件创建视图
4、viewDidLoad 载入完成,可以进行自定义数据以及动态创建其他控件
5、viewWillAppear 视图将出现在屏幕之前,马上这个视图就会被展现在屏幕上了
6、viewDidAppear 视图已在屏幕上渲染完成
当一个视图被移除屏幕并且销毁的时候的执行顺序,这个顺序差不多和上面的相反
1、viewWillDisappear 视图将被从屏幕上移除之前执行
2、viewDidDisappear 视图已经被从屏幕上移除,用户看不到这个视图了
3、dealloc 视图被销毁,此处需要对你在init和viewDidLoad中创建的对象进行释放
关于viewDidUnload :在发生内存警告的时候如果本视图不是当前屏幕上正在显示的视图的话, viewDidUnload将会被执行,本视图的所有子视图将被销毁,以释放内存,此时开发者需要手动对viewLoad、viewDidLoad中创建 的对象释放内存。 因为当这个视图再次显示在屏幕上的时候,viewLoad、viewDidLoad 再次被调用,以便再次构造视图。

2. view的加载过程

在这里插入图片描述

3. view卸载过程图

在这里插入图片描述

推荐看:ViewController生命周期实例

3. 应用程序状态以及不同场景下的变化

应用程序可能所处的状态:
在这里插入图片描述
不同场景下状态的变化:

应用启动场景

描述:当用户第一次启动程序时候,或者终止后再次启动
(1)Not running --> Inactive
(2)Inactive --> Active
在这里插入图片描述

应用退出场景

描述:分为两种可能:1.可以在后台运行或者挂起;2.不可以在后台运行和挂起

Step1:

(1)Active --> Inactive

(2)Inactive --> Background

(3)Background --> Suspended
在这里插入图片描述

Step2:

(1)Active --> Inactive

(2)Inactive --> Background

(3)Background --> Suspended

(4)Suspended --> Not running

在这里插入图片描述

应用挂起重新运行场景

(1)Suspended --> Background

(2)Background --> Inactive

(3)Inactive --> Active

应用终止状态

内存清除后应用程序终止,可能是强制清除内存,还可以是使用者手动清除

Background --> Suspended --> Not running

作者:星辰_入海
链接:https://www.jianshu.com/p/859d11e1e2fd

4.IOS开发多线程

参考资料:IOS开发多线程讲解(一)–结合斯坦福ios7开发(十)
爬过的坑:

5.UITableviewCell复用机制

参考资料:UITableviewCell复用机制
爬过的坑

6.类方法和实例方法调用

7. 属性strong和copy的区别

8.小概念:nib segue kvc kvo

nib:NIB是Non-interactive Interface Builder的缩写,是一种可视化的界面设计工具。它可以让你通过拖拽控件来设计界面,而无需编写代码。NIB文件通常包含多个窗口、视图和控制器等元素,这些元素可以相互连接并设置属性。
segue:

KVC是Key-Value Coding的缩写,即键值编码。它是一种动态获取对象属性值的方法。KVC可以通过字符串来访问对象的属性,例如:object.valueForKey(“name”)。
KVO是Key-Value Observing的缩写,即键值观察。它是一种动态监听对象属性变化的方法。KVO可以通过添加观察者来实现,当被观察对象的属性发生变化时,观察者会收到通知并作出相应处理。
下面是一个使用KVC和KVO的代码实例:

// 创建一个Person类
@interface Person : NSObject
@property (nonatomic, strong) NSString *name;
@end

@implementation Person
@end

// 创建一个Person对象并设置name属性
Person *person = [[Person alloc] init];
person.name = @"Tom";

// 使用KVC获取name属性值
NSString *name = [person valueForKey:@"name"];
NSLog(@"name is %@", name);

// 使用KVO监听name属性变化
[[person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil]];

// 修改name属性值
person.name = @"Jerry";

// 手动发送name属性变化的通知
[[person didChangeValueForKeyPath:@"name" notification:nil];

9. weak strong 循环引用

什么是循环引用:
循环引用是指两个对象相互强引用了对方,即retain了对方,从而导致两个对象都无法被释放,引发了内存泄漏现象。
在Objective-C中,循环引用通常发生在两个对象之间,其中一个对象持有另一个对象的强引用,而另一个对象也持有第一个对象的强引用。这种情况下,即使第一个对象被释放了,第二个对象也不会被释放,因为它们之间存在循环引用。这种情况可能会导致内存泄漏和其他问题。

循环引用实例:

@interface Person : NSObject
@property (nonatomic, strong) Person *friend;
@end

@implementation Person
@synthesize friend = _friend;
@end

Person *person1 = [[Person alloc] init];
Person *person2 = [[Person alloc] init];

// person1持有了person2的强引用,而person2也持有了person1的强引用
person1.friend = person2;
person2.friend = person1;

[person1 release]; // 释放person1对象

在这个例子中,Person类有一个属性friend,它是一个指向其他Person对象的指针。当创建两个Person对象时,它们相互持有对方的强引用,形成了循环引用。当其中一个对象被释放时,由于它持有另一个对象的强引用,所以另一个对象也无法被释放,从而引发了内存泄漏问题。

为了避免造成循环引用,代理一般需使用弱引用
例子:

#import <UIKit/UIKit.h>

//新建一个协议,协议的名称一般是由:"类名+Delegate"

@protocol  ViewControllerBDelegate<NSObject>

//代理方必须实现的方法
@required

//代理方可选实现的方法
@optional
- (void)ViewControllerBsendValue:(NSString *)value;
@end

@interface ViewControllerB : UIViewController

//委托代理人,为了避免造成循环引用,代理一般需使用弱引用
@property(weak,nonatomic) id<ViewControllerBDelegate> delegate;
//id是一个通用的数据类型,表示任何对象。在这个例子中,id<ViewControllerBDelegate>表示一个遵循ViewControllerBDelegate协议的对象的通用指针。
//这行代码定义了一个名为delegate的属性,其类型为ViewControllerBDelegate的弱引用。这意味着这个属性可以指向任何遵循ViewControllerBDelegate协议的对象,但是这个对象并不会增加delegate的引用计数。因此,当delegate所指向的对象被释放时,它不会影响ViewControllerB对象的内存管理。
@end 

代码实战篇

2.

//
//  AppDelegate.m
//  ExampleCode
//
//

#import "AppDelegate.h"
#import "ViewController.h"

@interface AppDelegate ()

@end

@implementation AppDelegate


- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {

    // 第一个页面
    ViewController *viewController = [[ViewController alloc] init];
    //UINavigationController *navigationController = [[UINavigationController alloc] initWithRootViewController:viewController];
    
    // 初始化并显示第一个页面
    self.window = [[UIWindow  alloc] init];
    self.window.backgroundColor = [UIColor whiteColor];
    self.window.frame = [[UIScreen mainScreen] bounds];
    
    //获取手机宽高
//    CGSize size_screen = self.window.frame.size;
//    CGFloat scale_screen = [UIScreen mainScreen].scale; //屏幕缩放比
//    NSLog(@"the width of screen is:%f",size_screen.width*scale_screen);  //1170 像素分辨率(px)
//    NSLog(@"the height of screen is:%f",size_screen.height*scale_screen); //2532
//    NSLog(@"the width of screen is:%f",size_screen.width);  //390
//    NSLog(@"the height of screen is:%f",size_screen.height); //844
    
    self.window.rootViewController = viewController;
    
    [self.window makeKeyAndVisible];
    return YES;
}

@end

工具篇

2.

小tips

  1. 如何使用Podfile

http://www.niftyadmin.cn/n/4996810.html

相关文章

AIGC专栏3——Stable Diffusion结构解析-以图像生成图像(图生图,img2img)为例

AIGC专栏3——Stable Diffusion结构解析-以图像生成图像&#xff08;图生图&#xff0c;img2img&#xff09;为例 学习前言源码下载地址网络构建一、什么是Stable Diffusion&#xff08;SD&#xff09;二、Stable Diffusion的组成三、img2img生成流程1、输入图片编码2、文本编码…

hznuoj---python查找最大字母

Description 对于输入的每个字符串&#xff0c;查找其中的最大字母&#xff0c;在该字母后面插入字符串"(max)"。 Input 输入数据包括多个测试实例&#xff0c;每个实例由一行长度不超过100的字符串组成&#xff0c;字符串仅由大小写字母构成。注&#xff1a;A与a是…

UE4 植物生长

这个可以改变SplineMesh朝向

FPGA 学习笔记:Vivado 工程管理技巧

前言 当前使用 Xilinx 的 FPGA,所以需要熟悉 Xilinx FPGA 的 开发利器 Vivado 的工程管理方法 这里初步列举一些实际 Xilinx FPGA 开发基于 Vivado 的项目使用到的工程的管理技巧 代码管理 做过嵌入式软件或者其他软件开发的工程技术人员,都会想到使用代码管理工具,如 SVN 、…

【Day-27满就是快】代码随想录-二叉树-二叉树的最大深度

给定一个二叉树&#xff0c;找出其最大深度。 二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。 说明: 叶子节点是指没有子节点的节点。 ———————————————————————————————————— 1. 递归法 可以使用前序和后序遍历。前序就是…

电子学会 2023年3月 青少年软件编程Python编程等级考试三级真题解析(选择题+判断题+编程题)

青少年编程Python编程等级考试三级真题解析(选择题+判断题+编程题) 2023年3月 一、选择题(共25题,共50分) 十进制数111转换成二进制数是?( ) A. 111 B. 1111011 C. 101111 D. 1101111 答案选:D 考点分析:考察python 进制转换 十进制转二进制,采用除二倒取余数,直到商…

Python:多变量赋值

相关文章 Python专栏https://blog.csdn.net/weixin_45791458/category_12403403.html?spm1001.2014.3001.5482 Python中的赋值语句可以同时对多个变量进行对象绑定&#xff08;赋值&#xff09;&#xff0c;既可以是多变量链式赋值&#xff0c;也可以是多变量平行赋值&#x…

【面试经典150题】跳跃游戏

题目链接 给你一个非负整数数组 nums &#xff0c;你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达最后一个下标&#xff0c;如果可以&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 1 < nums…