2018-05-21 swift日常踩坑

swift4.1

今天开始使用swift写新的项目,有很多小语法不是很熟悉,这个就是日常总结,以后忘记了再用到也好找。

1.textfield设置placeholder字体的颜色大小

 let placeholserAttributes = [NSAttributedStringKey.foregroundColor : ColorFromString(hex: "#999999"),NSAttributedStringKey.font : UIFont.systemFont(ofSize: 14)]
        searchText.attributedPlaceholder = NSAttributedString(string: "QQ号/手机号/邮箱",attributes: placeholserAttributes)

2.tableview的group模式下顶部留空白的问题,可以直接通过代理设置头部view高度。

也可以这样:

table.tableHeaderView = UIView.init(frame: CGRect(x: 0, y: 0, width: ScreenW, height: CGFloat.leastNormalMagnitude))

3.在以前的OC开发中,MVC模式下,为了避免model和view的直接交互或者在model的值变化的时候更新UI,我们一般采用下边这种重写set方法的形式

-(void)setName:(NSString *)name{
      _name = name
      lable.text = _name
}

在swift中的set方法和get方法写法如下

var _name:String?
var name:String?{
get{
//get方法
return _name
}
set{
//set方法
//只要外界通过.name给对象的name赋值,那么值就会保存在newValue
_name = newValue

我尝试使用这种OC思维去重写set方法来更新UI,发现不管怎么样,始终崩溃在get方法里,但是不写get方法编译器又不给编译通过。难受啊马飞。

后来发现了willset和didset方法(汗颜,因为是直接上手的项目没有再去复习一遍,都忘了。),简直不能更开心,写法如下

感觉跟OC里的KVO原理有点像啊,OC的KVO是基于runtime来新建类添加willChangeObjec和didChangeObject,这swift日常就有,那思考一下swift的KVO的原理是否基于didSet和willSet(后续研究一下#mark),据说swift5会加入runtime机制,那岂不是要上天?

var name:String?{
//设置完值之后调用
didSet{
//。一般我们重写set方法就是为了赋值后更新UI。。而Swift通常在这个方法里边写更新UI的代码
//因为这个方法会在赋值后调用
print(name)
}
}

4.这里推荐一个键盘控制的三方库IQKeyboardManagerSwift

swift版本的基本用法如下 在AppDelegate中调用该方法

func keyboardSetUp() {
        IQKeyboardManager.shared.enable = true
        //控制点击背景是否收起键盘
        IQKeyboardManager.shared.shouldResignOnTouchOutside = true
        //控制键盘上的工具条文字颜色是否用户自定义
        //将右边Done改成完成
        IQKeyboardManager.shared.toolbarDoneBarButtonItemText = "完成"
        // 控制是否显示键盘上的工具条
        IQKeyboardManager.shared.enableAutoToolbar = true
        //最新版的设置键盘的returnKey的关键字 ,可以点击键盘上的next键,自动跳转到下一个输入框,最后一个输入框点击完成,自动收起键盘
        IQKeyboardManager.shared.toolbarManageBehaviour = .byPosition
    }

5.swift去除KVO观察者模式,在swift中可以用deinit来代替dealloc

6.如果使用系统自带的UITableViewCell,或者使用XIB创建的自定义cell,如果tableview没有去掉分割线的话,会造成即便是没有数据的cell也会显示分割线,加上下边这句就可以了。

table.tableFooterView = UIView.init()

7.越写越想单开一个文章仔细写,过段时间吧,慢慢铺开,目前忙业务。

在swift中,声明protocol的方式我们都知道,但是我有一个需求是有些方法是可选的,在OC中我们可以用 @required、@optional,但是在swift中懵逼了。。。

不说了,上代码,PS:记得方法名之前也要加上@objc

@objc protocol TAViewControllerCellDelegate {
    @objc optional func didCallbackCurrentProgress(progress:Float)
}

8.在swift中使用switch语句,经常会提示Default will never be executed,

这个原因可能是因为case已经把所有的可能性都列举出来了,代码如下:

enum SelectButtonStates {
    case keystoreState
    case helpWordState
    case privateKeyState
}

switch selectState {
case .keystoreState:
    break
case .helpWordState:
    break
case .privateKeyState: break            
default: break
//这个时候就会提示 Default will never be executed,其实也没啥大影响,主要看你自己是如何选择了。

9.在iOS11中,WKWebView的坐标X Y都为0,但是顶部却顶不到最上边,而是在状态栏以下,通过查看图层关系,并不是WKWebVIew没有顶到顶端,而是webviewcontentview自动向下缩进了20,解决办法

        if #available(iOS 11.0, *) {
            webView.scrollView.contentInsetAdjustmentBehavior = .never
        } else {
            // Fallback on earlier versions
        }

10.今天一直莫名崩溃,以为是方法的问题,但是try!也没throw错误,控制台输出

warning: could not execute support code to read Objective-C class data in the process. This may reduce the quality of type information available.

Google了一下,发现可能是死循环造成的,刚好发现在一个GCD方法中使用了self.xxx来当做方法参数,于是暂时用一个最笨的方法解决了,控制台也不这样输出了

        let address = walletModel.address
        let oldP = oldPassword
        let newP = newPassword
        DispatchQueue.global(qos: .userInteractive).async {
            WalletCryptService.updateEncryptPrivateKey(oldPassword: oldP, newPassword: newP,walletAddress:address)
            DispatchQueue.main.async {
                NeuLoad.hidHUD()
                NeuLoad.showToast(text: "密码修改成功,请牢记!")
                self.navigationController?.popViewController(animated: true)
            }
        }

你可能感兴趣的:(2018-05-21 swift日常踩坑)