【iOS-系统框架】

news2025/1/11 21:53:36

文章目录

  • 前言
  • 47.熟悉系统框架
        • CoreFoundation框架
        • 其他框架
    • 要点
    • 48. 多用块枚举,少用for循环
        • for循环
        • NSEnumerator遍历
        • 快速遍历
        • 基于块的遍历方式
    • 要点
  • 49.对自定义其内存管理语义的collection使用无缝桥接
    • 要点
  • 50.构建缓存时选用NSCache而非NSDictionary
        • NSCache
        • NSCache实例
    • 要点
  • 51.精简initialize与load的实现代码
        • load
        • initalize
        • initalize方法尽量精简
    • 要点
  • 52.别忘了NSTimer会保留其目标对象
        • 创建计时器
        • 保留环
        • 使用块的特点打破保留环
    • 要点
  • 总结

前言

小蓝书的最后一张学习的主要内容是OC的系统框架,对于OC而言Foundation框架是OC最基本最重要的框架了

47.熟悉系统框架

OC的Foundation框架,像NSObject NSArray, NSDictionary等类都在其中。Foundation框架里的类都是用NS前缀,因为OC之前作为NeXTSTEP操作系统确定的。

将一系列代码封装为动态库,并在其中放入描述其接口的头文件,这样做出来的东西就叫框架。

CoreFoundation框架

Foundation框架提供了collection等基础核心功能,而且还提供了字符串处理等复杂功能。还存在一个CoreFoundation框架,在之前了解过他是不属于OC框架之内的,但是OC应用程序的编写离不开这个框架,Foundation框架的许多功在CoreFoundation框架都可以找到对应的C语言API

他其中的很多类都和Foundation框架相似,并且我们还可以通过“无缝桥接”功能实现CoreFoundation框架中的C语言数据结构平滑转换为Foundation框架中的OC对象,也可以反向转换。无缝桥接技术是用某些相当复杂的代码实现出来的,这些代码可以使运行期系统把CoreFoundation框架中的对象视为普通的OC对象。

NSString所对应的就是CFString对象。

其他框架

  • CFNetWork 此框架提供了 C语言級别的网络通信能力,它将 “BSD 套接字"(BSDsockeD) 抽象成易于使用的网络接口。而Foundation 则将该框架里的部分内容封装为
    Objective-c语言的接口,以便进行网络通信,例如可以用 NSURLConncction 从 URI
    中下载数据。
  • CoreAudio 该框架所提供的C语言 APT可用来操作设备上的音频硬件。这个框架屆手比较难用的那科,因为育频处理本身就很复东。所幸由这套 ^PL 可以抽象出另外一套Objective-C 式 API,用后者米处理音频问题会更简单些。
  • AvFoundation 此框架所提供的 Objective-C对象可用水回放并录制音频及视频,比如能碰在 UI视凶类里播放视物。
  • CoreData 此框架所提供的 Objective-C 按口可将对象放人数据,便于特久保存。
    CoreData 会处理数据的狀取及存储事宜,而且可以路越 Mac OS X 及 iOS 平台。
  • CoreText 此框架提供的 C诺青接口可以高效执行文字排版及谊染操作。

OC编程的重要特点就是:经常需要使用底层的C语言级 API。用c语言实现 APTI的好处是,可以统过 Objeotive-C 的运行期系统,从而提升执行速度

在编写新的工具类之前可以在系统框架搜一下,通常有写好的类可以供直接使用

要点

  • 许多系统框架都可以直接使用。其中最重要的是Foundation与CoreFoundation,这两个框架提供了构建应用程序所需的许多核心功能。
  • 很多常见任务都能用框架来做,例如音频与视频处理、网络通信、数据管理等。
  • 请记住:用纯C写成的框架与用OC写成的一样重要,若想成为优秀的OC开发者,应该掌握C语言的核心概念。

48. 多用块枚举,少用for循环

在OC里,列举collection中的元素可以使用C语言的for循环,还可以使用快速遍历。当学习了block块特性的时候,又提供了多种遍历collection的方式,可以传入块。

for循环

遍历数组的时候for循环最基本

- (void)forMethod {
    NSArray *testArray = @[@"kd", @"lbj", @"Curry", @"KW", @"@PG"];
    for (int i = 0; i < testArray.count; i++) {
        // 操作数组
    }
}

不过对于字典或者集合,字典和set都是无序的,所以要先把它转换为数组才可以正常使用for循环来使用

 NSDictionary *textDic = @{@"LA": @"LBJ", @"PHX" :@"KD"};
    NSArray *allKeys = [textDic allKeys];
    for (int i = 0; i < allKeys.count; i++) {
        // 操作字典或集合
    }

for循环还有个比较好的地方就是反向遍历,在需要执行反向遍历的时候for循环往往更方便。

NSEnumerator遍历

NSEnumerator是个抽象基类,其中只定义了两个方法,供其具体子类来实现:请添加图片描述
关键的是其中的nextObject对象,它返回枚举里的下个对象,当返回不为nil的时候就会一直调用下一个对象,常用while语句

  • 数组
    • NSEnumerator类还提供了反向枚举器
- (void)NSEnumerator {
 // Array
    NSArray *testArray = @[@"kd", @"lbj", @"Curry", @"KW", @"@PG"];
    NSEnumerator *enumerator = [testArray objectEnumerator];
    // 正向
    id object;
    while ((object = [enumerator nextObject]) != nil) {
        // 操作数组
    }
 
    // 反向
    NSEnumerator *reverseEnum = [testArray reverseObjectEnumerator];
    id object2;
    while ((object2 = [reverseEnum nextObject]) != nil) {
        // 操作数组
    }
    

  • 字典和集合
// Dict And Set
    NSDictionary *textDic = @{@"LA": @"LBJ", @"PHX" :@"KD"};
    NSEnumerator *enumertorDic = [textDic keyEnumerator];
    id key;
    while ((key = [enumertorDic nextObject]) != nil) {
        // 操作字典
        id value = textDic[key];
    }
}

快速遍历

快速遍历是OC2.0引入的语法功能,引入了in关键字,语法更加简洁了collection的遍历过程。尤其是字典类

- (void)fast {
    NSArray *testArray = @[@"kd", @"lbj", @"Curry", @"KW", @"@PG"];
    for (id object in testArray) {
        //
    }
    NSDictionary *textDic = @{@"LA": @"LBJ", @"PHX" :@"KD"};
    for (id key in textDic) {
        NSLog(@"%@", textDic[key]);
    }
}

基于块的遍历方式

对于块的引入,数组字典和集合都有自己的块遍历方法

  • 数组
    请添加图片描述
    • 参数1是每次枚举的对象, idx是下标, stop则是代表是否停止遍历
    • obj = array[idx];
- (void)block {
    NSArray *testArray = @[@"kd", @"lbj", @"Curry", @"KW", @"@PG"];
    __block NSInteger x = 2;
    [testArray enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        if (x < 4) {
            x++;
            NSLog(@"%@", obj);
            // 等效
            NSLog(@"%@", testArray[idx]);
            *stop = NO;
        } else {
            *stop = YES;
            NSLog(@"x == 4 STOP");
        }
    }];
    //
}

请添加图片描述
字典和集合是一样的思路。
遍历时可以直接从块里获取更多信息,并且它能够修改块的方法签名,以免进行类型转换操作。
块的遍历也有反向遍历数组字典集合,我们需要传入一个新的选项掩码请添加图片描述
我知道反向遍历是通过NSEnumerationReserve来实现的,当然反向遍历只针对有序的数组和集合

要点

  • 遍历collection有四种方式。最基本的办法是for循环,其次是NSEnumerator遍历法及快速遍历法,最新、最先进的方式则是“块枚举法”。
  • “块枚举法”本身就能通过GCD来并发执行遍历操作,无须另行编写代码。而采用其他遍历方式则无法轻易实现这一点。
  • 若提前知道待遍历的collection含有何种对象,则应修改块签名,指出对象的具体类型

49.对自定义其内存管理语义的collection使用无缝桥接

“无缝桥接”技术其实就是不同库之间相同类型的相互转换。
使用“无缝桥接”技术,可以在定义于Foundation框架中的OC类和定义于CoreFoundation框架中的C数据结构之间互相转换

- (void)seamlessBridging {
    NSArray *testArray = @[@"kd", @"lbj", @"Curry", @"KW", @"@PG"];
    CFArrayRef aCFArray = (__bridge  CFArrayRef)testArray;
    NSLog(@"cfArratSize =  %li", CFArrayGetCount(aCFArray));
}

转换操作中的__bridge告诉ARC如何处理转换所涉及的OC对象。__bridge本身的意思是:ARC仍然具备这个OC对象的所有权。而__bridge_retained则与之相反,意味着ARC将交出对象的所有权。与之相似,反向转换可通过__bridge_transfer来实现,也就是将对象的所有权交给ARC。这三种转换方式称为“桥式转换”

要点

  • 通过无缝桥接技术,可以在Foundation框架中的OC对象与CoreFoundation框架中的C语言数据结构之间来回转换。
  • 在CoreFoundation层面创建collection时,可以指定许多回调函数,这些函数表示此collection应如何处理其元素。然后,可运用无缝桥接技术,将其转换成具备特殊内存管理语义的OC collection。

50.构建缓存时选用NSCache而非NSDictionary

就目前的知识而言,对于从网上下载的图片如何缓存,我会把图片全部放到字典里,使用的时候就无需再次下载了,OC提供了一个NSCache类更好的方便缓存。

NSCache

NSCache类的优势在于当系统资源即将耗尽的时候自动删减缓存,这是字典类不能做到的。并且遵循先删减最久未使用的对象

NSCache并非拷贝 键 而是保留 键。这和字典完全不同,并且NSCache是线程安全的,它在开发者不自己编写安全锁的情况下多个线程可以同时访问NSCache,这对于缓存来说是很重要的,多线程完成这个任务更加方便

NSCache可以令开发者操控缓存删减其内容的时机,可以调整缓存里的对象总数和对象的总开销,就是在讲对象加入缓存的时候开发者可以知道开销值,对象总数或总开销超过上限,缓存就会自动删减不需要的对象,当然这个删减不能确定会不会删减掉某个必要的对象,所以把对象转换成NSData对象之后把数据大小当作缓存值更合适这样避免了复杂的计算开销值,变成了读取数据大小的步骤

NSCache实例

书上提供了实用NSCache的例子,并引入其他的知识。
请添加图片描述
请添加图片描述
下载数据所用的URL,就是缓存的键。若缓存未命中,即缓存中没有访问者所需的数据,则下载数据并将其放入缓存。而数据的“开销值”则设为其长度

存在一个类叫做NSPurgeableDataNSCache搭配起来用,它是NSMutableData的子类,而且实现了NSDiscardableContent协议。如果某个对象所占的内存能够根据需要随时丢弃,那么就可以实现该协议所定义的接口。这就是说,当系统资源紧张时,可以把保存NSPurgeableData对象的那块内存释放掉。NSDiscardableContent协议里定义了名为isContentDiscarded的方法,用来查询相关内存是否释放

如果需要访问某个NSPurgeableData对象,可以调用其beginContentAccess方法,告诉它现在还不应丢弃自己所占据的内存。用完之后,调用endContentAccess方法,告诉它在必要时可以丢弃自己所占据的内存了。这些调用可以嵌套,类似于对象的引用计数机制,为0就告诉系统可以销毁对象了

将NSPurgeableData对象加入NSCache,那么当该对象为系统所丢弃时,也会自动从缓存中清除。通过NSCache的evictsObjectsWithDiscardedContent属性,选择开启或者关闭此功能。

要点

  • 实现缓存时应选用NSCache而非NSDictionary对象。因为NSCache可以提供优雅的自动删减功能,而且是“线程安全的”,此外,它与字典不同,并不会拷贝键。
  • 可以给NSCache对象设置上限,用以限制缓存中的对象总个数及“总成本”,而这些尺度则定义了缓存删减其中对象的时机。但是绝对不要把这些尺度当成可靠的“硬限制”,它们仅对NSCache起指导作用。
  • NSPurgeableData与NSCache搭配使用,可实现自动清除数据的功能,也就是说,当NSPurgeableData对象所内存为系统所丢弃时,该对象自身也会从缓存中移除。
  • 如果缓存使用得当,那么应用程序的响应速度就能提高。只有那种“重新计算起来很费事的”数据,才值得放入缓存,比如那些需要从网络获取或从磁盘读取的数据。

51.精简initialize与load的实现代码

在OC里一个类必须初始化才能使用,大多数类继承与NSObject这个根类,提供了loadinitalize方法
请添加图片描述

load

加入运行期系统中的每个类和分类来说,会调用此方且仅调用一次,当类和分类的程序载入系统的时候就会执行这个方法,调用顺序是类大于分类。

load方法执行的时候运行期系统处于脆弱状态,在执行子类的load方法之前必须执行所有超类的load方法,其中还会执行代码涉及到的库的load,导致在load方法里面使用其他类不安全
请添加图片描述
在B类里调用A类,无法保证A类已经加载完成,也就是只有A的load方法执行完成才能完整的使用A类。

某个类没实现load方法,那么他的超类不论实现该方法,都不会调用。
load尽量不用它。

initalize

该方法是在程序首次使用该类之前调用且仅有一次,是由运行期系统调用的,不通过代码调用。和load存在区别

  • 惰性调用:当程序用到了相关类的时候才会调用,类似于懒加载模式。而load是所有类不管用不用先load方法之后再说
  • 其次运行期在执行该方法的时候是系统正常状态,也就是安全状态,不影响调用类的其他方法,此为线程安全
  • 当某个类没实现initialize方法,超类实现后会调用超类的方法,和大多数消息是一样的。

initalize方法尽量精简

  • 首先,大家都不想看到自己的应用程序“挂起”,若写的太过繁琐,导致其运行很慢那就适得其反了。
  • 开发者无法控制类的初始化时机。
  • 最后,如果某个类的实现代码很复杂,那么其中可能会直接或间接用到其他类。若那些类尚未初始化,则系统会迫使其初始化。然而,本类的初始化方法此时尚未运行完毕。其他类在运行其initialize方法时,有可能会依赖本类中的某些数据,而这些数据此时也许还未初始化好,就会造成依赖环

initalize方法只应该用来设置内部数据,不能在内部调用其他的方法。

initalize还可以初始化某个无法在编译器初始化的全局变量

请添加图片描述
无法初始化数组等一些类,整数可以在编译期定义。
请添加图片描述
放到initalize里面即可在调用前完成该全局变量的初始化
请添加图片描述

+ (void)initialize {
    if (self == [ViewController class]) {
        staticArray = [NSArray new];
    }
}
  • 别忘了单例类也可以实现该目的

要点

  • 在加载阶段,如果类实现了load方法,那么系统就会调用它。分类里也可以定义此方法,类的load方法要比分类中的先调用。与其他方法不同,load方法不参与覆写机制。
  • 首次使用某个类之前,系统会向其发送initialize消息。由于此方法遵从普通的覆写规则,所以通常应该在里面判断当前要初始化的是那个类。
  • load与initialize方法都应该实现的精简一些,这有助于保持应用程序的响应能力,也能减少引入“依赖环”的几率。
  • 无法在编译期设定的全局常量,可以放在initialize方法里初始化。

52.别忘了NSTimer会保留其目标对象

Foundation框架中有个类叫NSTimer,开发者可以指定绝对的日期与时间,以便到时执行任务,计时器要和“运行循环(run loop)”相关联,运行循环到时候会触发任务。创建NSTimer时,可以将其“预先安排”在当前的运行循环中,也可以先创建好,然后由开发者自己来调度。无论采用哪种方式,只有把计时器放在运行环里,它才能正常触发任务

创建计时器

 _pollTimer = [NSTimer scheduledTimerWithTimeInterval:2 target:self selector:@selector(pDoPoll) userInfo:nil repeats:YES];

用此方法创建出来的计时器,会在指定的间隔时间之后执行任务。也可以令其反复执行任务,知道开发者稍后将其手动关闭为止。

计时器会保留其目标对象,等到自身“失效”时再释放此对象。调用invalidate方法可令计时器失效;执行完相关任务之后,一次性的计时器也会失效。开发者若将计时器设置成重复执行模式,那么必须自己调用invalidate方法,才能令其停止。

由于计时器会保留其目标对象,所以反复执行任务通常会导致应用程序出现保留环。

// NSTimer

- (void)startPolling {
    _pollTimer = [NSTimer scheduledTimerWithTimeInterval:2 target:self selector:@selector(pDoPoll) userInfo:nil repeats:YES];
    [self stopPolling];
}
- (void)stopPolling {
    [_pollTimer invalidate];
    _pollTimer = nil;
}
- (void)pDoPoll {
    NSLog(@"poll");
}
// 为啥不实现dealloc?
- (void)dealloc {
    [_pollTimer invalidate];
}

这里为什么一直打印poll不是很理解 为什么没有调用dealloc方法
请添加图片描述

保留环

创建计时器的时候,由于目标对象是self,所以要保留此实例。然而,因为计时器是用实例变量存放的,所以实例也保留了计时器,于是,就产生了保留环。所以说,调用stopPolling,或者令系统将此实例回收,只有这样才能打破保留环。

因为是类和这个类中的实例出现了保留环,不管你外界怎么对这个类释放,这个计时器始终都会保留这个类,而这个类也会保留这个计时器,互相引用保留导致他们的计数永远都不会降为0请添加图片描述
如果从外界直接先调用stop方法,代码没办法自己检测。

使用块的特点打破保留环

使用块和weak关键字合理的打破保留环,块可以传递代码,这一功能可以利用
请添加图片描述

这段代码将计时器所应执行的任务封装成“块”,在调用计时器函数上,把它作为userInfo参数传进去。该参数可用来存放“不透明值”(即万能值),只要计时器还有效,就会一直保留着它。传入参数时要通过copy方法将block拷贝到“堆”上(之前在blk提过,copy方法把块变成了有引用计数的对象。)

否则等到稍后要执行它的时候,该块可能已经无效了。计时器现在的target是NSTimer类对象,这是个单例,因此计时器是否会保留它,其实都无所谓。此处依然有保留环,使用
方法和weak关键字打破它请添加图片描述

[NSTimer scheduledTimerWithTimeInterval:2 repeats:YES block:^

请添加图片描述
这段代码采用了一种很有效的写法,他先定义了一个弱引用,令其指向self,然后使块捕获这个引用,而不直接去捕获普通的self变量。也就是说,self不会为计时器所保留。当块开始执行时,立刻生成strong引用,以保证实例在执行期间持续存活。

要点

  • NSTimer对象会保留其目标,直到计时器本身失效为止,调用invalidate方法可令计时器失效,另外,一次性的计时器在触发完成任务之后也会失效。
  • 反复执行任务的计时器,很容易引人保留环,如果这种计时器的目标对象又保留了计时器本身,那肯定会导致保留环。这种环状保留关系,可能是直接发生的,也可能是通过对象图里的其他对象间接发生的。
  • 可以扩充NSTimer的功能,**用“块” 来打破保留环。**不过,除非NSTimer将来在公共接口里提供此功能,否则必须创建分类,将相关实现代码加入其中。

总结

小蓝书的最后一章刚开始看的不是很理解,有些代码还是需要自己手动打一下才知道原理,其中49.50 就是看着书写的,理解的不是很完全,GCD和block还是需要再次多加学习才能掌握。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/336339.html

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

相关文章

bgp综合实验2

目录实验要求子网划分ip以及各个环回的配置ospf配置及接口网络类型更改bgp的配置路由反射器小知识联邦的小知识bgp宣告实验要求 如图 实验要求&#xff1a; 1&#xff0c;R2-7每台路由器都存在一个环回接口用于建立邻居&#xff0c;同时存在一个环回代表连接用户的接口&…

安全—06day

负载均衡反向代理下的webshell上传负载均衡负载均衡下webshell上传的四大难点难点一&#xff1a;需要在每一台节点的相同位置上传相同内容的webshell难点二&#xff1a;无法预测下一次请求是哪一台机器去执行难点三&#xff1a;当我们需要上传一些工具时&#xff0c;麻烦来了&a…

解决方案 | 亚洲诚信助力互联网行业网络安全建设

行业背景当前&#xff0c;世界正处在从工业经济向数字经济转型过渡的大变革时代&#xff0c;互联网作为工业社会向数字时代迁移的驱动力&#xff0c;是推进新一轮科技革命与产业变革的中坚力量。随着数字化进程的加剧&#xff0c;企业所面临的网络安全形势也日趋多变复杂。尤其…

玩具全球各地检测标准整理

玩具检测认证&#xff1a;REACH法规、ROHS指令、EN 71测试、ASTM F963、GB 6675、CE认证、儿童用品CPC认证等其他认证。测试标准&#xff1a;CPSC 总共公布了 38 个标准&#xff0c;主要涉及的检测内容有&#xff1a;1). CPSIA 总铅和邻苯&#xff1b;2). 美国玩具标准 ASTMF96…

springboot整合单机缓存ehcache

区别于redis的分布式缓存&#xff0c;ehcache是纯java进程内的单机缓存&#xff0c;根据不同的场景可选择使用&#xff0c;以下内容主要为springboot整合ehcache以及注意事项添加pom引用<dependency><groupId>net.sf.ehcache</groupId><artifactId>ehc…

PTA L1-044 稳赢(详解)

前言&#xff1a;内容包括四大模块&#xff1a;题目&#xff0c;代码实习&#xff0c;大致思路&#xff0c;代码解读 题目&#xff1a; 大家应该都会玩“锤子剪刀布”的游戏&#xff1a;两人同时给出手势&#xff0c;胜负规则如图所示&#xff1a; 现要求你编写一个稳赢不输的…

计算机组成结构之数据传输控制方式、总线、CISC和RISC

数据传输控制方式 输入输出控制方式 程序控制&#xff08;查询&#xff09;方式&#xff1a;cpu一直持续不断在查询I/O是否准备好了&#xff0c;准备好就会调用I/O&#xff1b;I/O没有准备好&#xff0c;CPU会持续等待I/O&#xff1b;&#xff08;软件实现&#xff09;程序中…

vcruntime140_1.dll无法继续执行代码,怎么解决这种问题?

经常使用电脑的人&#xff0c;可能对于这个弹出框应该不陌生&#xff0c;“vcruntime140_1.dll无法继续执行代码”&#xff0c;其实会出现这种情况&#xff0c;主要是因为缺少一个动态链接库 (DLL) 文件导致的。这个文件是 Visual C 2015 库的一部分&#xff0c;某些程序需要这…

第五节 字符设备驱动——点亮LED 灯

通过字符设备章节的学习&#xff0c;我们已经了解了字符设备驱动程序的基本框架&#xff0c;主要是掌握如何申请及释放设备号、添加以及注销设备&#xff0c;初始化、添加与删除cdev 结构体&#xff0c;并通过cdev_init 函数建立cdev 和file_operations 之间的关联&#xff0c;…

阿里云ecs服务器搭建CTFd(ubuntu20)

1.更新apt包索引 sudo apt-get update更新源 1、使用快捷键【ctrlaltt】打开终端。 2、输入以下命令备份原有软件源文件。 cp /etc/apt/sources.list /etc/apt/sources.list.bak_yyyymmdd 3、再输入以下命令打开sources.list文件并添加新的软件源地址。 vim /etc/apt/sources.…

本质安全设备标准(IEC60079-11)的理解(四)

本质安全设备标准&#xff08;IEC60079-11&#xff09;的理解&#xff08;四&#xff09; 对于标准中“Separation”的理解 IEC60079-11使用了较长的篇幅来说明设计中需要考虑到的各种间距&#xff0c; 这也从一定程度上说明了间距比较重要&#xff0c;在设计中是需要认真考虑…

VBA提高篇_ 21 随机数 / 模运算

文章目录1. 模Mod运算2. 随机数及其运算2.1 Rnd()函数调用Excel公式: Rand() / RandBetween()使用VBA函数: Rnd()函数2.2 随机小数化整数公式2.3 伪随机数(过程是随机的,但是实际上不是,随机数算法)2.3.1 随机数初始化2.3.2 随机小数被当做下标时会被自动四舍五入VBA的默认属性…

BERT简介

BERT&#xff1a; BERT预训练模型训练步骤&#xff1a; 使用Masked LM方式将语料库中的某一部分的词语掩盖住&#xff0c;模型通过上下文预测被掩盖的信息&#xff0c;从而训练出初步的语言模型在语料库中选出连续的上下语句&#xff0c;并使用Tranformer模块识别语句的连续性通…

PHP基于TCPDF第三方类生成PDF文件

最近在研发招聘的系统 遇到了这个问题 转换pdf 折腾了很久 分享一下PHP基于TCPDF第三方类生成PDF文件最近遇到一个需求&#xff0c;需要根据数据库的字段生成表格式的PDF文件并发送邮箱第一步、我们先去官网上面去下载tcpdf的类&#xff1a;http://www.tcpdf.org/或者是从githu…

LightGBM

目录 1.LightGBM的直方图算法(Histogram) 直方图做差加速 2.LightGBM得两大先进技术(GOSS&EFB) 2.1 单边梯度抽样算法(GOSS) 2.2 互斥特征捆绑算法(EFB) 3.LightGBM得生长策略(leaf-wise) 通过与xgboost对比&#xff0c;在这里列出lgb新提出的几个方面的技术 1.Ligh…

JavaScript 评测代码运行速度的几种方法

一、使用 performance.now() API 在 JavaScript 中&#xff0c;可以使用 performance.now() API 来评测代码的运行速度。该 API 返回当前页面的高精度时间戳&#xff0c;您可以在代码执行前后调用它来计算代码执行所需的时间。 例如&#xff1a; let t0 performance.now();…

计算机组成与设计04——处理器

系列文章目录 本系列博客重点在深圳大学计算机系统&#xff08;3&#xff09;课程的核心内容梳理&#xff0c;参考书目《计算机组成与设计》&#xff08;有问题欢迎在评论区讨论指出&#xff0c;或直接私信联系我&#xff09;。 第一章 计算机组成与设计01——计算机概要与技…

idea整合svn

idea整合svn svn下载 链接&#xff1a;https://pan.baidu.com/s/1yS3R3lEE8lm9c9Ap-ndDKg 提取码&#xff1a;65ur 基础步骤 IDED中配置SVN没有svn.exe解决办法 以下是两种解决方案 需要卸载原 svn&#xff08;不推荐&#xff09; 参考网址&#xff1a; https://blog.csdn.…

ansible以及playbook

文章目录1.ansible1.概念2.工作原理3.安装ansible2.playbook1.组件2.角色详解3.剧本语法3.1基本语法3.2常用字段3.3 选项1.ansible 1.概念 ansible是一个可以服务端控制其他机器完成工作的一个应用工具&#xff0c; 1.服务没有客户端存在&#xff0c;通过模块对节点服务器进行…

设计模式--建造者模式 builder

设计模式--建造者模式 builder&#xff09;建造者模式简介建造者模式--小例子&#xff08;电脑购买&#xff09;1.产品类2.抽象构建者3.实体构建类4.指导者类5.客户端测试类小结建造者模式简介 建造者模式有四个角色,概念划分如下&#xff1a; Product &#xff1a; 产品类&a…