iOS 解决GIF显示卡顿的情况 等10个奇袭

2024-03-04 02:08

本文主要是介绍iOS 解决GIF显示卡顿的情况 等10个奇袭,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.关于关键字volatile

一个定义为volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。就像大家更熟悉的const一样,volatile是一个类型修饰符。它是被设计用来修饰被不同线程访问和修改的变量。如果不加入volatile,基本上会导致这样的结果:要么无法编写多线程程序,要么编译器失去大量优化的机会。

Volatile变量具有 synchronized 的可见性特性,但是不具备原子特性。这就是说线程能够自动发现 volatile变量的最新值。Volatile变量可用于提供线程安全,但是只能应用于非常有限的一组用例:多个变量之间或者某个变量的当前值与修改后值之间没有约束。因此,单独使用 volatile 还不足以实现计数器、互斥锁或任何具有与多个变量相关的不变式(Invariants)的类(例如 “start <=end”)。

出于简易性或可伸缩性的考虑,您可能倾向于使用 volatile变量而不是锁。当使用 volatile变量而非锁时,某些习惯用法更加易于编码和阅读。此外,volatile变量不会像锁那样造成线程阻塞,因此也很少造成可伸缩性问题。在某些情况下,如果读操作远远大于写操作,volatile变量还可以提供优于锁的性能优势。

代码示例

1
2
3
4
volatile int i=10;
int j = i;
...
int k = i;

volatile 告诉编译器i是随时可能发生变化的,每次使用它的时候必须从i的地址中读取,因而编译器生成的可执行码会重新从i的地址读取数据放在k中。编译器在产生release版可执行码时会进行编译优化,加volatile关键字的变量有关的运算,将不进行编译优化。而优化做法是,由于编译器发现两次从i读数据的代码之间的代码没有对i进行过操作,它会自动把上次读的数据放在k中。而不是重新从i里面读。这样以来,如果i是一个寄存器变量或者表示一个端口数据就容易出错,所以说volatile可以保证对特殊地址的稳定访问,不会出错。

1
int square(volatile int *ptr) {  return  *ptr * *ptr; }

这段代码的目的是用来返指针ptr指向值的平方,但是,由于ptr指向一个volatile型参数,编译器将产生类似下面的代码:

1
2
3
4
5
6
int square(volatile int *ptr) {
  int a,b;
  a = *ptr;
  b = *ptr;
  return  a * b;
}

由于*ptr的值可能被意想不到地该变,因此a和b可能是不同的。结果,这段代码可能返不是你所期望的平方值!正确的代码如下:

1
long square(volatile int *ptr) { int a; a = *ptr;  return  a * a; }

下面是volatile变量的几个使用:

  • 并行设备的硬件寄存器(如:状态寄存器)

  • 一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)

  • 多线程应用中被几个任务共享的变量

那么问题来了:

  • 一个参数既可以是const还可以是volatile吗?答案是是的。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是const因为程序不应该试图去修改它。

  • 一个指针可以是volatile 吗?答案是是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。

在编写多线程程序中使用volatile的关键点:

1)将所有的共享对象声明为volatile;

2)不要将volatile直接作用于基本类型;

3)当定义了共享类的时候,用volatile成员函数来保证线程安全;

在多线程中,我们可以利用锁的机制来保护好资源临界区。在临界区的外面操作共享变量则需要volatile,在临界区的里面则non-volatile了。

2.关键字const的位置

const意味着”只读”,分析下面的含义:

1
2
3
4
5
6
7
8
const int a;
int const a; // 前两个的作用是一样,a是一个常整型数。
const int *a; // 第三个意味着a是一个指向常整型数的指针(整型数是不可修改的,但指针可以)
int * const a; // 第四个意思a是一个指向整型数的常指针(指针指向的整型数是可以修改的,但指针是不可修改的)
int const * const a;
// a是一个指向常整型数的常指针(指针指向的整型数是不可修改的,同时指针也是不可修改的)。
//表示a是一个指针常量,初始化的时候必须固定指向一个int常量或者int变量,之后就不能再指向别的地方了,它总是把它所指向的目标当作一个int常量。
//也可以写成const int* const a;含义相同。

合理地使用关键字const可以使编译器很自然地保护那些不希望被改变的参数,防止其被无意的代码修改。这样可以减少bug的出现。欲阻止一个变量被改变,可以使用 const 关键字。

在定义 const 变量时,通常需要对它进行初始化,因为以后就没有机会再去改变它了;对指针来说,可以指定指针本身为 const,也可以指定指针所指的数据为 const,或二者同时指定为 const;在一个函数声明中,const 可以修饰形参,表明它是一个输入参数,在函数内部不能改变其值;对于类的成员函数,若指定其为 const 类型,则表明其是一个常函数,不能修改类的成员变量;对于类的成员函数,有时候必须指定其返回值为 const 类型,以使得其返回值不为“左值”。

3.滑动的时候隐藏navigation bar

1
navigationController.hidesBarsOnSwipe = Yes;

4. 消除导航条返回键带的title

1
2
[[UIBarButtonItem appearance] setBackButtonTitlePositionAdjustment:UIOffsetMake(0, -60)
                                                  forBarMetrics:UIBarMetricsDefault];

5. 将Navigationbar变成透明而不模糊

1
2
3
4
[self.navigationController.navigationBar setBackgroundImage:[UIImage  new ]
                          forBarMetrics:UIBarMetricsDefault];
self.navigationController.navigationBar .shadowImage = [UIImage  new ];
self.navigationController.navigationBar .translucent = YES;

6. static

函数体内 static 变量的作用范围为该函数体,不同于 auto 变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值;在模块内的 static 全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问;在模块内的 static 函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内;在类中的 static 成员变量属于整个类所拥有,对类的所有对象只有一份拷贝;在类中的 static 成员函数属于整个类所拥有,这个函数不接收 this 指针,因而只能访问类的static 成员变量。

7.用一个pan手势来代替UISwipegesture的各个方向

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
- (void)pan:(UIPanGestureRecognizer *)sender
{
typedef NS_ENUM(NSUInteger, UIPanGestureRecognizerDirection) {
UIPanGestureRecognizerDirectionUndefined,
UIPanGestureRecognizerDirectionUp,
UIPanGestureRecognizerDirectionDown,
UIPanGestureRecognizerDirectionLeft,
UIPanGestureRecognizerDirectionRight
};
static  UIPanGestureRecognizerDirection direction = UIPanGestureRecognizerDirectionUndefined;
switch  (sender.state) {
case   UIGestureRecognizerStateBegan: {
if   (direction == UIPanGestureRecognizerDirectionUndefined) {
CGPoint velocity = [sender velocityInView:recognizer.view];
BOOL isVerticalGesture = fabs(velocity.y) > fabs(velocity.x);
if  (isVerticalGesture) {
     if  (velocity.y > 0) {
     direction = UIPanGestureRecognizerDirectionDown;
     }
     else   {
     direction = UIPanGestureRecognizerDirectionUp;
     }
}
else
{
if  (velocity.x > 0) {
direction = UIPanGestureRecognizerDirectionRight;
}
else
{
direction = UIPanGestureRecognizerDirectionLeft;
}
}
}
break  ;
}
case  UIGestureRecognizerStateChanged: {
switch  (direction) {
case  UIPanGestureRecognizerDirectionUp: {
[self handleUpwardsGesture:sender];
break  ;
}
case  UIPanGestureRecognizerDirectionDown: {
[self handleDownwardsGesture:sender];
break ;
}
case   UIPanGestureRecognizerDirectionLeft: {
[self handleLeftGesture:sender];
break ;
}
case  UIPanGestureRecognizerDirectionRight: {
[self handleRightGesture:sender];
break  ;
}
default  : {
break ;
}
}
break ;
}
case   UIGestureRecognizerStateEnded: {
direction = UIPanGestureRecognizerDirectionUndefined;
break ;
}
default :
break ;
}
}

8. 拉伸图片不变形

1468898521688776.gif

等同于:

1
2
[[UIImage imageNamed:@ "" ] stretchableImageWithLeftCapWidth:10 topCapHeight:10];
[[UIImage imageNamed:@ "" ] resizableImageWithCapInsets:UIEdgeInsetsMake(10, 10, 10, 10)];

9. Gif图片显示优化

1170656-956bd6681b580206.gif

FLAnimatedImage可以帮你完成GIF的显示处理。解决GIF显示卡顿的情况。

1
2
3
4
5
FLAnimatedImage *image = [FLAnimatedImage animatedImageWithGIFData:[NSData dataWithContentsOfURL:[NSURL URLWithString:@ "https://upload.wikimedia.org/wikipedia/commons/2/2c/Rotating_earth_%28large%29.gif" ]]];
FLAnimatedImageView *imageView = [[FLAnimatedImageView alloc] init];
imageView.animatedImage = image;
imageView.frame = CGRectMake(0.0, 0.0, 100.0, 100.0);
[self.view addSubview:imageView];

使用就是这么简单。

10. CollectionView实现tableview的悬停header

1170656-e32d0d953b7bbf48.gif

CSStickyHeaderFlowLayout可以解决您的疑问。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
#import "CSStickyHeaderFlowLayout.h"
- (void)viewDidLoad {
  [ super  viewDidLoad];  // Locate your layout
CSStickyHeaderFlowLayout *layout = (id)self.collectionViewLayout;
if  ([layout isKindOfClass:[CSStickyHeaderFlowLayout class]]) {
layout.parallaxHeaderReferenceSize = CGSizeMake(320, 200);
  }
}
- (UICollectionReusableView *)collectionView:(UICollectionView *)collectionView viewForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath {  // Check the kind if it's CSStickyHeaderParallaxHeader
if  ([kind isEqualToString:CSStickyHeaderParallaxHeader]) {
UICollectionReusableView *cell = [collectionView dequeueReusableSupplementaryViewOfKind:kind withReuseIdentifier:@ "header"  forIndexPath:indexPath];
  return  cell;
}
}

就是这么简单。

这篇关于iOS 解决GIF显示卡顿的情况 等10个奇袭的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

SpringBoot排查和解决JSON解析错误(400 Bad Request)的方法

《SpringBoot排查和解决JSON解析错误(400BadRequest)的方法》在开发SpringBootRESTfulAPI时,客户端与服务端的数据交互通常使用JSON格式,然而,JSON... 目录问题背景1. 问题描述2. 错误分析解决方案1. 手动重新输入jsON2. 使用工具清理JSON3.

MySQL 设置AUTO_INCREMENT 无效的问题解决

《MySQL设置AUTO_INCREMENT无效的问题解决》本文主要介绍了MySQL设置AUTO_INCREMENT无效的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录快速设置mysql的auto_increment参数一、修改 AUTO_INCREMENT 的值。

关于跨域无效的问题及解决(java后端方案)

《关于跨域无效的问题及解决(java后端方案)》:本文主要介绍关于跨域无效的问题及解决(java后端方案),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录通用后端跨域方法1、@CrossOrigin 注解2、springboot2.0 实现WebMvcConfig

Go语言中泄漏缓冲区的问题解决

《Go语言中泄漏缓冲区的问题解决》缓冲区是一种常见的数据结构,常被用于在不同的并发单元之间传递数据,然而,若缓冲区使用不当,就可能引发泄漏缓冲区问题,本文就来介绍一下问题的解决,感兴趣的可以了解一下... 目录引言泄漏缓冲区的基本概念代码示例:泄漏缓冲区的产生项目场景:Web 服务器中的请求缓冲场景描述代码

解决JSONField、JsonProperty不生效的问题

《解决JSONField、JsonProperty不生效的问题》:本文主要介绍解决JSONField、JsonProperty不生效的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录jsONField、JsonProperty不生效javascript问题排查总结JSONField

github打不开的问题分析及解决

《github打不开的问题分析及解决》:本文主要介绍github打不开的问题分析及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、找到github.com域名解析的ip地址二、找到github.global.ssl.fastly.net网址解析的ip地址三

RedisTemplate默认序列化方式显示中文乱码的解决

《RedisTemplate默认序列化方式显示中文乱码的解决》本文主要介绍了SpringDataRedis默认使用JdkSerializationRedisSerializer导致数据乱码,文中通过示... 目录1. 问题原因2. 解决方案3. 配置类示例4. 配置说明5. 使用示例6. 验证存储结果7.

Swagger在java中的运用及常见问题解决

《Swagger在java中的运用及常见问题解决》Swagger插件是一款深受Java开发者喜爱的工具,它在前后端分离的开发模式下发挥着重要作用,:本文主要介绍Swagger在java中的运用及常... 目录前言1. Swagger 的主要功能1.1 交互式 API 文档1.2 客户端 SDK 生成1.3

java连接opcua的常见问题及解决方法

《java连接opcua的常见问题及解决方法》本文将使用EclipseMilo作为示例库,演示如何在Java中使用匿名、用户名密码以及证书加密三种方式连接到OPCUA服务器,若需要使用其他SDK,原理... 目录一、前言二、准备工作三、匿名方式连接3.1 匿名方式简介3.2 示例代码四、用户名密码方式连接4

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复