iOS练手项目知识点汇总

news2025/1/19 23:03:06

基础理解篇

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.coloradmin.cn/o/966989.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

给视频添加背景图片,让它们更具魅力!

想要让你的视频更加出彩吗&#xff1f;给它们添加背景图片是不错的选择&#xff01;但是&#xff0c;如何做到呢&#xff1f;不用担心&#xff0c;我们的视频剪辑高手可以帮助你轻松实现&#xff01;我们提供多种背景图片选择&#xff0c;你可以根据自己的喜好和需求进行选择。…

程序员自由创业周记#7:仲裁

没想到 没想到写的周记会有这么多人看&#xff0c;还能收到这么多陌生(或熟悉)朋友的真诚建议、鼓励、甚至是打赏&#xff0c;几乎所有的评论和私信我都认真的回复了&#xff0c;本想的是通过网友和朋友的监督坚定我创业的信念&#xff0c;有点外界压力也能迫使自己持续输出一…

P1886 滑动窗口 /【模板】(双端队列)+双端队列用法

例题 有一个长为 n 的序列 a&#xff0c;以及一个大小为 k 的窗口。现在这个从左边开始向右滑动&#xff0c;每次滑动一个单位&#xff0c;求出每次滑动后窗口中的最大值和最小值。 例如&#xff1a; The array is [1,3,−1,−3,5,3,6,7],and k3。 输入格式 输入一共有两行…

4. 虚拟机栈

4.1. 虚拟机栈概述 4.1.1. 虚拟机栈出现的背景 由于跨平台性的设计&#xff0c;Java的指令都是根据栈来设计的。不同平台CPU架构不同&#xff0c;所以不能设计为基于寄存器的。 优点是跨平台&#xff0c;指令集小&#xff0c;编译器容易实现&#xff0c;缺点是性能下降&…

WideDeep模型介绍

文章目录 1. Wide&Deep模型的记忆能力和泛化能力2. Wide&Deep模型的结构3. Wide&Deep模型的进化——Deep&Cross模型4. Wide&Deep模型的影响力 Wide&Deep模型是 记忆能力和 泛化能力的综合&#xff0c;是谷歌在2016年提出的。正如其名&#xff0c;Wid…

java运行程序流程

java运行程序流程 检查JDK环境 java -version 新建Java文件&#xff08;源文件&#xff09;Hello.java 打开记事本&#xff0c;输入 public class Hello {public static void main(String[] args) {System.out.println("Hello");} } 保存文件&#xff0c;把文件后缀…

C++极简内存泄露检测工具(34行代码实现)

工具特点 1 极度简单&#xff1a;34行代码实现&#xff0c;线程安全 2 入寝式检测&#xff1a;需要给现有重点怀疑的类添加一行入寝代码 3 只针对类类型&#xff1a;动态数组需要使用更为复杂的技术&#xff0c;不在检测能力范围之内。考虑到大部分C代码都以类实现代码&…

Python零基础超详细教程:字典(Dictionary)相关介绍使用

前言 嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! Python字典是另一种可变容器模型&#xff0c; 且可存储任意类型对象&#xff0c;如字符串、数字、元组等其他容器模型。 python更多源码/资料/解答/教程等 点击此处跳转文末名片免费获取 一、创建字典 字典由键和对应值…

高数刷题笔记

任取一米奚落>0,存在德尔塔...&#xff0c;0<|x-x0|<德尔塔 不同点是这个不是|x-x0|趋近与哪一个小范围&#xff0c;而是x趋近与大范围 极限的本质就是逼近&#xff0c;二者都是在确定要逼近到什么程度&#xff01;&#xff01;&#xff01; 放缩 我们要凑出|x-x0|的…

python技术面试题合集(二)

python技术面试题 1、简述django FBV和CBV FBV是基于函数编程&#xff0c;CBV是基于类编程&#xff0c;本质上也是FBV编程&#xff0c;在Djanog中使用CBV&#xff0c;则需要继承View类&#xff0c;在路由中指定as_view函数&#xff0c;返回的还是一个函数 在DRF中的使用的就是…

docker笔记3 Docker常规安装

1.安装tomcat docker hub上面查找tomcat镜像 docker search tomcat 从docker hub上拉取tomcat镜像到本地 docker pull tomcat docker images查看是否有拉取到的tomcat 使用tomcat镜像创建容器实例(也叫运行镜像) docker run -it -p 8080:8080 tomcat -p 小写&#xff0c;主…

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书杭州师范大学图书馆

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书杭州师范大学图书馆

MySQL访问和配置

目录 1.使用MySQL自带的客户端工具访问 2.使用DOS访问(命令行窗口WinR → cmd) 3.连接工具&#xff08;SQLyog或其它&#xff09; MySQL从小白到总裁完整教程目录:https://blog.csdn.net/weixin_67859959/article/details/129334507?spm1001.2014.3001.5502 1.使用MySQL自…

AMEYA360代理 | 佰维eMMC、LPDDR存储芯片赋能电视终端流畅体验

5G、AI、VR、AR等技术的发展&#xff0c;助推智能电视、机顶盒等电视终端成为智能家居领域不可忽视的重要设备。随着4K超高清(UHD)技术、虚拟现实技术(VR)和增强现实技术(AR)的普及&#xff0c;并向8K超高清技术不断渗透&#xff0c;电视终端将可以为消费者提供更清晰的视觉体验…

解决DCNv2不能使用高版本pytorch编译的问题

可变形卷积网络GitHub - CharlesShang/DCNv2: Deformable Convolutional Networks v2 with Pytorch代码已经出来好几年了&#xff0c;虽然声称"Now the master branch is for pytorch 1.x"&#xff0c;实际上由于pytorch自1.11版开始发生了很大变化&#xff0c;原来基…

时序预测 | MATLAB实现TCN-GRU时间卷积门控循环单元时间序列预测

时序预测 | MATLAB实现TCN-GRU时间卷积门控循环单元时间序列预测 目录 时序预测 | MATLAB实现TCN-GRU时间卷积门控循环单元时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 1.MATLAB实现TCN-GRU时间卷积门控循环单元时间序列预测&#xff1b; 2.运行环…

精益创业的规划框架:愿景,战略,产品

精益创业的规划框架&#xff1a;愿景&#xff0c;战略&#xff0c;产品【安志强趣讲276期】 趣讲大白话&#xff1a;愿景管战略&#xff0c;战略管产品 **************************** 愿景&#xff1a;创业企业有个清晰的方向 愿景怎么来的&#xff1f;发现社会问题&#xff0c…

electron笔记无边框窗口、DLL调用、DLL函数返回指针

无边框 const win new BrowserWindow({width: 1290,height: 736,minHeight: 736,minWidth: 1040,maxHeight: 736,maxWidth: 1290,frame: false, // 无边框webPreferences: {// preload: process.env.WEBPACK_DEV_SERVER_URL ? __dirname /preload.js : app://./preload.js,…

leetcode172. 阶乘后的零(java)

阶乘后的零 题目描述巧妙的解法代码演示 上期经典 题目描述 难度 - 中等 172. 阶乘后的零 给定一个整数 n &#xff0c;返回 n! 结果中尾随零的数量。 提示 n! n * (n - 1) * (n - 2) * … * 3 * 2 * 1 示例 1&#xff1a; 输入&#xff1a;n 3 输出&#xff1a;0 解释&#…

【算法刷题-栈与队列篇】

目录 1.leetcode-232. 用栈实现队列2.leetcode-225. 用队列实现栈3.leetcode-20. 有效的括号&#xff08;1&#xff09;代码1&#xff08;2&#xff09;代码2 4.leetcode-1047. 删除字符串中的所有相邻重复项5.leetcode-150. 逆波兰表达式求值6.leetcode-239. 滑动窗口最大值7.…