【Effective Objective - C】—— 系统框架

【Effective Objective - C】—— 系统框架

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

47.熟悉系统框架

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

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

CoreFoundation框架

与Foundation相伴的框架,叫做CoreFoundation框架,虽然从技术上讲它不是OC框架,但是它却是编写OC应用程序时所应熟悉的重要框架。他其中的很多类都和Foundation框架相似,并且我们还可以通过 “无缝桥接”,功能实现CoreFoundation框架中的C语言数据结构平滑转换为Foundation框架中的OC对象,也可以反向转换。无缝桥接技术是用某些相当复杂的代码实现出来的,这些代码可以使运行期系统把CoreFoundation框架中的对象视为普通的OC对象。

NSString所对应的就是CFString对象。

其他框架

系统库介绍
CFNetwork提供C语言级别的网络通信能力,将“BDS套接字”抽象成为易于使用的网络接口。Foundation将该框架中的部分内容封装为Objective-C语言的接口,以便进行网络通信
CoreAudio提供的C语言API可以用来操作设备上的音频硬件。这个框架比较难用。这套API可以抽象出另一套Objective-C式API
AVFoundation提供的Objective-C对象可以用来回放并录制音频及视频
CoreData提供的Objective-C接口可以将对象放入数据库,便于持久保存。Coredata会处理数据的获取及存储等事宜,可以跨Mac OS X及iOS平台
CoreText提供的C语言接口可以高效的执行文字排版及渲染操作
CoreAnimation是用OC语言写成的,它提供了一些工具,而UI框架则用这些工具来渲染图形并播放动画。但是它本身并不是框架,它只是QuartzCore框架的一部分。
CoreGraphics框架以C语言写成,其中提供了2D渲染所必备的数据结构与函数,例如:CGPoint、CGSize、CGRect等等。
MapKit可以为iOS程序提供地图功能。
Social为Mac OS X及iOS程序提供了社交网络功能。

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

要点

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

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

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

for循环

便利数组的第一种方法就是采用老式的for循环。这是一个很基本的方法,功能非常有限。

NSArray* array = [NSArray arrayWithObjects:@"111", @"222", @"333" ,nil];
    for (int i = 0; i < array.count; i++){
        NSLog(@"%@", array[i]);
    }

但是对于集合和字典,集合和字典是无序的,要将他们转化为数组才可以进行for循环遍历。

NSDictionary* dictionary = [NSDictionary dictionaryWithObjects:@[@"111", @"222", @"333", ] forKeys:@[@"1", @"2", @"3"]];
    NSArray* keys = [dictionary allKeys];
    for (int i = 0; i < keys.count; i++){
        NSLog(@"%@", dictionary[keys[i]]);
        
    }

NSSet* set = [[NSSet alloc] initWithObjects:@"111", @"222", @"333", nil];
    NSArray* array = [set allObjects];
    for (int i = 0; i  < set.count; i++){
        NSLog(@"%@", array[i]);
    }

使用Objective-C 1.0的NSEnumerator遍历

NSEnumerator是一个抽象基类。方法nextObject返回枚举里的下一个对象。每次调用此方法时, 内部数据都会更新,使得下次调用方法时能返回下一个对象。

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        NSLog(@"Hello, World!");
        NSArray* array = [NSArray arrayWithObjects:@"111", @"222", @"333" ,nil];
        NSEnumerator* enumerator = [array objectEnumerator];
        id object;
        while ((object = [enumerator nextObject] )!= nil) {
            NSLog(@"%@", object);
        }
    }
    return 0;
}

遍历字典的方式和数组和set不太一样,先根据给定的键把对应的值提取出来。

NSDictionary* dictionary = [NSDictionary dictionaryWithObjects:@[@"111", @"222", @"333", ] forKeys:@[@"1", @"2", @"3"]];
        NSEnumerator* enumerator = [dictionary keyEnumerator];
        id key;
        while (( key = [enumerator nextObject]) != nil) {
            id value = dictionary[key];
            NSLog(@"%@", value);
        }
NSSet* set = [[NSSet alloc] initWithObjects:@"111", @"222", @"333", nil];
    NSEnumerator* enmuator = [set objectEnumerator];
    id object;
    while ((object = [enmuator nextObject]) != nil) {
        NSLog(@"%@", object);
    }

使用NSEnumerator还有一个好处,就是有多种“枚举器”可供使用。反向遍历数组所用的枚举器,如果用它来遍历,就可以按反方向来迭代collection中的元素了。

NSArray* array = [NSArray arrayWithObjects:@"111", @"222", @"333" ,nil];
    NSEnumerator* enumerator = [array reverseObjectEnumerator];
    id object;
    while ((object = [enumerator nextObject]) != nil) {
        NSLog(@"%@", object);

    }

快速遍历

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

NSArray* array = [NSArray arrayWithObjects:@"111", @"222", @"333" ,nil];
    for (id object in array) {
        NSLog(@"%@", object);
    }

如果某个类的对象支持快速遍历,那么就可以说自己遵从名为NSFastEnumerator的协议,令开发者可以采用此语法来迭代该对象。

NSDictionary* dictionary = [NSDictionary dictionaryWithObjects:@[@"111", @"222", @"333", ] forKeys:@[@"1", @"2", @"3"]];
    for (id key in dictionary) {
        id value = dictionary[key];
        NSLog(@"%@", value);
    }

//这里采用的是倒序枚举的快速遍历
NSArray* array = [NSArray arrayWithObjects:@"111", @"222", @"333" ,nil];
    for (id object in [array reverseObjectEnumerator]) {
        NSLog(@"%@", object);
    }

基于块的遍历方式

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

[array enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {

参数1是每次枚举的对象, idx是下标, stop则是代表是否停止遍历.
obj = array[idx];

//块枚举
NSArray* array = @[@"111", @"222", @"333", @"444", @"555", @"666"];
 __block NSInteger x = 2;
[array enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
     if (x < 4) {
         x++;
         NSLog(@"%@", obj);
         // 等效
         NSLog(@"%@", array[idx]);
         *stop = NO;
     } else {
         *stop = YES;
         NSLog(@"x == 4 STOP");
     }
 }];

在这里插入图片描述
字典和集合是一样的思路。
遍历时可以直接从块里获取更多信息,并且它能够修改块的方法签名,以免进行类型转换操作。

要点

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

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

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

NSArray* array = @[@"111", @"222", @"333", @"444", @"555", @"666"];
        CFArrayRef aCFArray = (__bridge  CFArrayRef)array;
        NSLog(@"cfArratSize =  %li", CFArrayGetCount(aCFArray));
        NSLog(@"%@", aCFArray);
        CFRelease(aCFArray);

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

需要加上CFRelease()释放内存。

要点

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

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

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

NSCache

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

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

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

NSCache实例

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

存在一个类叫做NSPurgeableData和NSCache搭配起来用,它是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这个根类,而该类有两个办法,可用来实现这种初始化操作。

load方法:

首先是load方法,其原型如下:

+ (void)load;

加入运行期系统中的每个类和分类来说,会调用此方且仅调用一次,当类和分类的程序载入系统的时候就会执行这个方法,调用顺序是类大于分类。
load方法执行的时候运行期系统处于脆弱状态,在执行子类的load方法之前必须执行所有超类的load方法,其中还会执行代码涉及到的库的load,导致在load方法里面使用其他类不安全。
在这里插入图片描述
在B类里调用A类,无法保证A类已经加载完成,也就是只有A的load方法执行完成才能完整的使用A类。

并且在load方法里调用的类若之前没有加载好,那么再调用程序就会崩溃了,注意:程序在执行load方法时会阻塞系统,并且每个类都只能执行自己的load,不会执行超类的load的。

总的来说,load方法里代码要写的精简,能少写就少写,能不用就不用。

initialize方法:

想执行与类相关的初始化操作,还有个办法,就是覆写下列方法:

+ (void)initialize;

对于每个类来说,该方法会在程序首次用该类之前调用,且只调用一次。它是由运行期系统来调用的,绝不应该通过代码直接调用。

与load的区别:

  • 它是“惰性调用的”,也就是说,只有当程序用到了相关的类时,才会调用。
  • 其次运行期在执行该方法的时候是系统正常状态,也就是安全状态,不影响调用类的其他方法,此为线程安全。
  • 当某个类没实现initialize方法,超类实现后会调用超类的方法,和大多数消息是一样的。

initalize方法尽量精简

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

所以说,initialize方法只应该用来设置内部数据。不应该在其中调用其他方法,即便是本类自己的方法,也最好别调用。若某个全局状态无法在编译期初始化,则可以放在initialize里来做。
在这里插入图片描述
注意,某些OC对象也可以在编译期创建,例如NSString实例。

编写load或initialize方法时,一定要留心这些注意事项。把代码实现的简单一些,能节省很多调试时间。除了初始化全局状态之外,如果还有其他事情要做,那么可以专门创建一个方法来执行这些操作,并要求该类的使用者必须在使用本类之前调用此方法。

要点

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

第52条:别忘了NSTimer会保留其目标对象

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

创建定时器

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimerInterval)seconds target:(id)target selector:(SEL)selector userInfo:(id)userInfo repeats:(BOOL)repeats;

用此方法创建出来的计时器,会在指定的间隔时间之后执行任务。也可以令其反复执行任务,知道开发者稍后将其手动关闭为止。
target与selector参数表示计时器将在哪个对象上调用哪个方法。计时器会保留其目标对象,等到自身“失效”时再释放此对象。 调用invalidate方法可令计时器失效;执行完相关任务之后,一次性的计时器也会失效。开发者若将计时器设置成重复执行模式,那么必须自己调用invalidate方法,才能令其停止。

由于计时器会保留其目标对象,所以反复执行任务通常会导致应用程序出现保留环。
就像下面这样:
在这里插入图片描述
创建计时器的时候,由于目标对象是self,所以要保留此实例。然而,因为计时器是用实例变量存放的,所以实例也保留了计时器,于是,就产生了保留环。所以说,那么调用stopPolling,要么令系统将此实例回收,只有这样才能打破保留环。

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

使用块的特点打破保留环
使用块和weak关键字合理的打破保留环,块可以传递代码,这一功能可以利用
在这里插入图片描述
在这里插入图片描述
这段代码将计时器所应执行的任务封装成“块”,在调用计时器函数上,把它作为userInfo参数传进去。该参数可用来存放“不透明值”(即万能值),只要计时器还有效,就会一直保留着它。传入参数时要通过copy方法将block拷贝到“堆”上,否则等到稍后要执行它的时候,该块可能已经无效了。计时器现在的target是NSTimer类对象,这是个单例,因此计时器是否会保留它,其实都无所谓。此处依然有保留环,然而因为类对象无须回收,所以不用担心。

新改写的代码这样调用:
在这里插入图片描述
但是其实它还是有保留环的,因为块捕获了self变量,所以块要保留实例。而计时器又通过userInfo参数保留了块。 最后,实例本身还是要保留计时器。不过,只要改用weak引用,即可打破保留环。
在这里插入图片描述
这段代码采用了一种很有效的写法,他先定义了一个弱引用,令其指向self,然后使块捕获这个引用,而不直接去捕获普通的self变量。也就是说,self不会为计时器所保留。当块开始执行时,立刻生成strong引用,以保证实例在执行期间持续存活。

要点

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

虚拟机器centos7无法识别yum 命令异常处理笔记

问题现象 启动虚拟机后执行ipconfig 提示未找到该命令,然后执行yum install -y net-tools提示 curl#6 - "Could not resolve host: mirrorlist.centos.org; 未知的错误"的错误 [roothaqdoop~]# ifconfig -bash: ifconfig: 未找到命令 [roothadoop~]# yum install …

【QT 5 +Linux下软件桌面快捷方式+qt生成软件创建桌面图标+学习他人文章+第二篇:编写桌面文件.desktop】

【QT 5 Linux下软件桌面快捷方式qt生成软件创建桌面图标学习他人文章第二篇&#xff1a;编写桌面文件.desktop】 1、前言2、实验环境3、自我学习总结-本篇总结1、新手的疑问&#xff0c;做这件事目的2、了解.desktop3、三个关键目录以及文件编写1、目录&#xff1a;/opt/2、目录…

ChatGPT回答模式

你发现了吗&#xff0c;ChatGPT的回答总是遵循这些类型方式。 目录 1.解释模式 2.类比模式 3.列举模式 4.限制模式 5.转换模式 6.增改模式 7.对比模式 8.翻译模式 9.模拟模式 10.推理模式 1.解释模式 ChatGPT 在回答问题或提供信息时&#xff0c;不仅仅给出…

智能科技助力服装业:商品计划管理系统的革命性变革

随着智能科技的飞速发展&#xff0c;服装行业正在经历前所未有的变革。在这股浪潮中&#xff0c;商品计划管理系统的智能化转型成为了行业的核心驱动力。这种变革不仅极大地提高了服装企业的运营效率和市场竞争力&#xff0c;更为整个行业的可持续发展注入了新的活力。 智能商…

相机图像质量研究(35)常见问题总结:图像处理对成像的影响--运动噪声

系列文章目录 相机图像质量研究(1)Camera成像流程介绍 相机图像质量研究(2)ISP专用平台调优介绍 相机图像质量研究(3)图像质量测试介绍 相机图像质量研究(4)常见问题总结&#xff1a;光学结构对成像的影响--焦距 相机图像质量研究(5)常见问题总结&#xff1a;光学结构对成…

T-Dongle-S3开发笔记——移植LVGL

添加lvgl组件 idf.py add-dependency lvgl/lvgl>8.* 新建终端执行命令后出现了新的文件&#xff1a; 清除再编译后才会出现lvgl库 优化为本地组件 以上方式修改了组件文件内容重新编译后文件又会变回去。 所以我们要把lvgl变成本地组件 1、要把 idf_component.yml 文…

【Python】遇到的一些小问题及解决办法汇总

【工具】&#xff1a;pycharm 【环境】&#xff1a;Windows 一、数据集路径导入报错 【错误提示】&#xff1a; SyntaxError: (unicode error) ‘unicodeescape’ codec can’t decode bytes in position 2-3: truncated \UXXXXXXXX escape 如图&#xff1a; 【原因分析】&a…

visual studio2022使用tensorRT配置

只记录tensorRT在vs中使用时的配置&#xff0c;下载和安装的 文章主页自己寻找。 下载好TensorRT和对应的cuda之后&#xff0c;把tensorRT的锻炼了和lib文件复制粘贴到cuda对应的文件夹中&#xff0c;以方便调用。 完成之后打开vs新建一个tensorRT的项目&#xff0c;然后开始配…

文件上传漏洞--Upload-labs--Pass20--数组绕过

一、漏洞原理 漏洞来源&#xff1a;count()函数漏洞。 现自定义一个数组 arr[]&#xff0c;定义arr[0]1,arr[3]2, 此时count(arr)的值为2&#xff0c;则arr[count[arr]]即为arr[2]&#xff0c;但是arr[2]未定义&#xff0c;即为一个空值&#xff0c;若使用count()函数的本意是…

Windows10和Ubuntu22.04双系统安装

概要&#xff1a; 本篇演示Windows10和Ubuntu22.04双系统的安装&#xff0c;先安装Windows10&#xff0c;再安装Ubuntu22.04。 先安装Ubuntu22.04&#xff0c;后安装Windows10见&#xff1a; Ubuntu22.04和Windows10双系统安装-CSDN博客 一、说明 1、电脑 笔者的电脑品牌…

PhpStorm 2023:卓越的代码质量,无与伦比的调试体验 mac/win版

JetBrains PhpStorm 2023是一款针对PHP开发者的强大集成开发环境(IDE)&#xff0c;它提供了丰富的功能和工具&#xff0c;帮助开发者更高效地编写、调试和测试PHP应用程序。 PhpStorm 2023 软件获取 首先&#xff0c;JetBrains PhpStorm 2023具备强大的代码编辑功能&#xff0…

SAP STO VLPOD 报错 QA495 XXX 的问题

原因&#xff1a; 质检物料在中生成采购订单的时候的质检状态不对 PS物料是否需要质检在质量视图里面的

高级语言期末2015级唐班B卷

1.编写函数&#xff0c;按照如下公式计算圆周率π的值&#xff08;精确到1e-5&#xff09; #include <stdio.h>double pai() {double last0;double flag1;int n1;while(flag-last>1e-5) {lastflag;flag*1.0*(2*n)*(2*n)/((2*n-1)*(2*n1));n;}return 2*last; }int main…

3Dmax效果图是如何做出来的?

如果不知道3Dmax效果图是如何做出来的朋友&#xff0c;可以通过这篇文章有一个概念。 我们需要先准备两款通用软件&#xff0c;3Dmax建模软件和Photoshop图形制作软件。 一、构建模型 我们以常见的室内为例&#xff0c;通过“样条线”或者基础“几何体”来创造我们需要的物体…

HTML好玩代码合集(1)

VIP代码合集🧧,这一期是场景式HTML代码,里面的文字也是可以修改的,不知道怎么修改可以私信我。 效果(玩个梗,别在意): 好玩代码: <!DOCTYPE html> <html> {#jishugang#}<head><meta charset="utf-8" /><title>怎么堵船了�…

Docker基础篇(二)

docker run -d docker run -d 容器名或容器ID docker run -d 后台生成容器&#xff0c;并退出容器&#xff08;除容器中在运行脚本&#xff09; docker run -it 交互生成容器 docker run -d centos /bin/sh -c “while true; do echo zen; sleep 2;done” 查看容器中的进程…

提高效率、降低成本:外贸企业必备好用ERP软件盘点

好用的外贸ERP软件有哪些&#xff1f;本期为您盘点的外贸ERP软件有&#xff1a;Zoho Books&#xff0c;孚盟M8&#xff0c;富通天下&#xff0c;睿贝软件。 外贸ERP系统是什么 企业资源规划&#xff08;ERP&#xff09;系统是一种先进的管理软件&#xff0c;尤其是在外贸和国际…

SwiftUI 集合视图(Grid)拖放交换 Cell 的极简实现

概览 自从 SwiftUI 横空出世那天起&#xff0c;小伙伴们都感受到了它惊人的简单与便捷。而在本课中&#xff0c;我们将会用一个小“栗子”更直观的让大家体验到它无与伦比简洁的描述性特质&#xff1a; 如上图所示&#xff0c;我们在 SwiftUI 中实现了 Grid 中拖放交换 Cell 的…

全球游戏市场回暖,Flat Ads推动海外获客增长

摘要:热门游戏品类分析,解读新兴市场与赛道 近日,中国音数协游戏工委发布了《2023年中国游戏出海研究报告》,据报告数据显示,2023年,全球游戏市场规模11773.79亿元,同比增长6.00%,呈现增长回暖趋势。 图源:伽马数据 1.SLG和RPG游戏热度居高不下,休闲游戏增长势头强劲 目前,S…

计算机网络:物理层详解

物理层 单工通道&#xff0c;半双工通道&#xff0c;全双工通道 单工通道&#xff08;Simplex Channel&#xff09;&#xff1a; 单工通道是一种只允许数据在一个方向上传输的通道&#xff0c;通信双方中的一个方向只能发送&#xff0c;另一个方向只能接收&#xff0c;不能同…