Swift快速入门(一)之 Swift基础语法

Swift快速入门(一)之 Swift基础语法

近期重新学习了一遍Swift语言,主要以《Swift编程权威指南(第2版)》为教程,学习过程中结合比对Java语言特性将学习心得作为三篇笔记:Swift基础语法,Swift面向对象编程,Swift高级编程。

本文为第一篇《Swift基础语法》主要涵盖以下内容

  • 类型、常量和变量
  • if-else条件语句
  • switch语句
  • 数 整数和浮点数
  • 循环 for和while
  • 字符串
  • 可空类型
  • 数组 Array
  • 集合 Set
  • 字典 Dictionary
    [图片上传失败...(image-9a706b-1559060935851)]

类型、常量和变量

//常量的声明 let
//声明一个Int常量,常量一旦声明后不可变化
let num:Int = 2019
print(num)

//变量的声明 var
//声明一个字符串变量:声明一个名为name的变量,其类型是String,初始值是”Tom“
var name:String = "Tom"

//总结:对必须变化的实例用变量,其他都用常量


//Switf支持类型推断:声明变量或常量时无需声明类型,编译器会根据初始值内容进行推断
let n = 2019  //根据2019内容自行推断 实例n的类型为Int
let firstName = "Nik"
var lastName = "Lily"
print(n)

Swift中数 分为 整数和浮点数

  1. 整数: Int 64位
  2. 浮点数:Float 32位,Double 64位
  3. 整数操作符
  4. 加减乘除:+、—、*、/
  5. 取余: %
//1. 整数类型的变量的声明
var i1:Int = 10
var f1:Float = 10.0   //32位
var d1:Double = 10.0  //64位

//2. 类型转换
// Java中两个不同精度的数值相加会自动转换为较高精度,但Swift中需要手动转换类型使之匹配
//let number = i1 + f1  //不同类型数据相加 在swift中会异常
let number = Double(i1) + d1
print(str)
print(number)


//3. 浮点数
//浮点数:Float 32位,Double 64位
//浮点数默认的类型推断是 Double
let n2 = 1.98
print(type(of: n2))  //输出结果 Double

条件语句

if-else 由特定的条件逻辑执行代码,通常要处理一个非此即彼的状况

let age = 18
if age > 18{
    print("成年人")
}else
{
    print("未成年人")
}

if-else if 的使用

let age = 58
if age <= 18{
    print("未成年人")
}else if age<55
{
    print("青壮年")
}else
{
    print("老年人")
}

if-else 嵌套语句

let age = 88
if age > 18{
    if age>60{  //
        print("老年人")
    }else
    {
        print("成年人")
    }
}else
{
    print("未成年人")
}

三目运算 :理解为简化版 if-else使用

a?b:c "如果a为真,则执行b;否则执行c"

let age = 20
var result = age > 18 ? "成年人" : "未成年人";
print(result)

逻辑运算符

&& 逻辑与:当且仅当两者都为真时结果为真
|| 逻辑或:两者任意之一为真结果为真
! 逻辑非:真变假,假变真

switch语句:用于处理多重条件的情况

区别:if else用于处理”非此即彼“的情况,swith用于处理多重条件的情况

ps:先上结论

switch语句特性

  1. 每个case分支后不需要break
  2. swift语句中 default分支不是必需的
  3. fallthrough 状态转移语句,会先执行自己分支代码后,再把控制权转移到下面紧挨着的分支
  4. case 匹配值 支持区间 如 case 1...100 表示 1到100的区间内
  5. case 匹配值可支持 多个
  6. case 匹配值可使用where条件子句

swith 使用汇总

var code = 401
var erroInfo:String = "erro code is:"
switch code {
case 200,201,202:
    print("Http success ")
case 300...399:  //...区间
    print("300系列错误")
case let code where (code >= 400 && code <= 499) :  //where语句的使用,筛选条件
    erroInfo += "\(code) 400系列错误"
//    fallthrough //不需要匹配进入下一个分支并执行代码
default:
    erroInfo += "\(code)"  //字符串插值
    print("其他错误")
}

关于switch语法使用一一道来

switch语句

var httpcode=400
switch  httpcode{
case 200:
    print(httpcode)
case 400:
    print(httpcode)
default:
    print("default")
}

switch分支可以使用多个值

var httpcode=400
switch  httpcode{
case 200:
    print(httpcode)
case 400,401,402:
    print("服务器异常")
default:
    print("default")
}

switch可以是一个值,多个值,区间

var httpcode=322
switch  httpcode{
case 200:
    print(httpcode)
case 300...399:
    print(httpcode)
case 400,401,402:
    print("服务器异常")
default:
    print("default")
}

值绑定(字符串插值)占位符

var httpcode=322
var resultstr=""
switch  httpcode{
case 200:
    resultstr="OK"
case 300...309:
    resultstr="redirect:\(httpcode)"
case 400,401,402:
    resultstr="client erro:\(httpcode)"
case let unkwnowncode:
    resultstr="\(unkwnowncode) is not known erro code"
}

default 语句(不是必需的)

var httpcode=800
switch  httpcode{
case 200:
    print(httpcode)
case 300...399:
    print(httpcode)
case 400,401,402:
    print("服务器异常")
default:
    print("default httpcode:\(httpcode)")  
}

用where语句创建筛选条件

var httpcode:Int = 205
switch  httpcode{
case 200:
    print(httpcode)
case 300...399:
    print(httpcode)
case 400,401,402:
    print("服务器异常")
case let unknowcode where (unknowcode>201 && unknowcode<208): //指定httpcode 范围
    print("unknowcode where httpcode:\(httpcode)")   //where
default:
        print("default httpcode:\(httpcode)") 
}

元组和模式匹配

元组是把具有逻辑关联的两个或多个变量的组合,组合的结果是一个有序列表
var httpCode=200
var msg="message:"
switch httpCode{
case 200:
    msg+="成功"
case 300...399:
    msg+="重定向"
case 400,401,402:
    msg+="服务器错误"
default:
    msg+="未知"
}
let erro=(httpCode,msg)
//访问元组
erro.0
erro.1  //通过序号访问元组元素
//1.9给元组的元素别名
let erro2=(code:httpCode,erroInfo:msg)
erro2.code
erro2.erroInfo

用元组做模式匹配

var firstCode:Int=201
var secondCode:Int=404
let erro=(firstCode,secondCode)
switch  erro{
case (201,400):
    print("201,400")
case (200,_):
    print("200,_")
case (_,404):
    print("_,404")
default :
    print("default")
}

switch 和 if-else

var age=23
switch  age{
case 18...28:
    print("年轻人")
default:
    print("age =:\(age)")   //default 语句
}
//使用if-case替代,不需要写不关注的default分支,更关注核心代码逻辑
var age=23
if case 18...28=age{
        print("年轻人")
}

var age=26
if case 18...28=age,age>25{
    print("大于25岁的年轻人")
}

循环

  • for循环
  • while 循环

for-in 循环的用法

var number:Int = 0;
for i in 1...5{  //遍历 1至5
//    print(i)
    number += 1
    print(number)
}

不需要迭代器i的用法

for _ in 1...5
{
    number += 1
    print(number)
}

带where语句的for-in循环

for i in 1...10 where i % 3 == 0
{
    print(" 能被3整除的数:\(i)")
}

while循环的使用

var number:Int = 3
while number > 0 {
    print(" while 循环 \(number)")
    number -= 1;
}

repeat-while 循环,相当于java中的 do-while循环

repeat{
    print(number)
    number += 1
}while number <= 3

break和continue的区别

  1. break终结当前所处层的的循环
  2. continue终结当前这一次循环
for i in 1...10
{
    if(i==3)
    {
        break
    }
    print(i)
}

print("continue的区别")
for i in 1...10
{
    if(i==3)
    {
        continue  //跳过3
    }
    print(i)
}

字符串

var str = "Hello, playground"  //声明字符串变量
let str1 = "Hello"
var str2 : String = "HELLO 2"
str2.append(" swift")  //累加

str += "!"  //给可变字符串添加
print(str)
print(str.count)  //获取字符串的长度
print(str.contains(str1))
print(str.starts(with: str1))
//print(st)
print(str2)

可空类型 Optional

可空类型Optional是Swift的独特特性,用来指定某个实例没有值。一个可空类型的实例,要么已经可用,要么没有值就称为nil

声明可空类型

通过这种方式,编译器知道实例是否可能为nil,这种显示的声明让代码更具表达力,也更安全。

var erroInfo :String? //声明可空类型 ?
print(erroInfo)  //输出nil
//erroInfo += "test"
erroInfo = "test"
print(str)   //输出 ”Swift“
print(erroInfo)  //输出 Optional("test")

增加条件语句

if erroInfo != nil
{
    let theErro = erroInfo!  //感叹号的作用是强制展开可空类型
    print(theErro)  //输出"test"
}

可空类型的示例绑定

if let theErrors = erroInfo
{
    //当erroInfo不为nil 则给theErrors赋值,并执行当前代码块
    print(theErrors)
}

展开多个可空实例

var code : Int?
code = 401
if let theError = erroInfo, let theCode = code
{
    print("展开多个可空实例")
    print(theCode)
    print(theError)
}else
{
    print("不满足")
}

展开多个可空实例,和额外检查

if let theError = erroInfo, let theCode = code, code == 402
{
    print("展开多个可空实例")
    print(theCode)
    print(theError)
}

隐式展开可空实例

  1. 隐式展开可空实例的类型用 !声明,相比普通可空类型它不需要展开
  2. 表示要比普通可空类型更有信心
  3. 但存在着危险性:如果可空实例没有值访问其值会导致运行时错误
var erroCodeInfo:String!
erroCodeInfo = "no data"
print("erroCodeInfo info is \(erroCodeInfo)")

可空链式调用

  1. 可空链式调用运算符 ?.
  2. 与可空绑定类似,可空链式调用提供了一种对可空实例进行查询以判断其是否包含值的机制。
    可空链式调用允许把多个查询串联为一个可空实例的值,如果查询链中任意可空实例是nil,则返回nil
var erroInfo = nil
//当 erroInfow不为空,则转为大写,为空则返回nil
var erroInfoUp = erroInfo?.uppercased()  //
print(erroInfoUp)
//原地修改可空实例
var newErroInfo = erroInfo?.appending("6666")
print(newErroInfo)

nil合并运算符

  1. nil合并运算符 ??
  2. 从可空实例中获取值,或者使用默认值
//要么获取值,要么默认值
let des = erroInfo ?? "no data"  
print(des)

数组 Array

数组是有序的

//数组的两种声明方式
var data:Array<String> //声明一个字符串数组
var datas2:[String] //声明一个字符串数组

var arrs2:[String] = ["xx","yy"]  //初始化数组
var arrs = ["a1"]  //数组声明的类型推断
print(arrs)
arrs.append("appenData")
print(arrs)
arrs[0] += "ap"
arrs[1] = "d2"  //替换指定下标的元素
print(arrs)
print("数组长度count = \(arrs.count)")
arrs += arrs2 //两个数组相加
print(arrs)
print("------指定位置插入一个元素 'EN'-------")
arrs.insert("EN", at: 2)
print(arrs)
print(arrs[0...2])  //打印下标前三的元素

var newArr = ["xx","yy"]
//判断两个数组是否相等,数组是有序的,需元素顺序和内容完全一致才算相等
print(newArr == arrs2) 

print("常量数组 一旦创建不可修改")
let finalArrays = ["a","b"]
print(finalArrays)
//finalArrays[0] = "66"

集合 set

集合是无序、元素唯一

//初始化一个集合
var hashSet = Set<String>() //声明一个String类型的集合
var set1:Set = ["a","b","c"]
print(set1)


//集合增加元素
hashSet.insert("a")
hashSet.insert("b")
print(hashSet)
////循环遍历插入
for i in 4...8
{
    hashSet.insert("number \(i)")
}
//print(hashSet)

//包含指定元素
hashSet.contains("a")
print(hashSet.contains("a"))
//print(hashSet.contains(set1))

//并集 union
let newSet = hashSet.union(set1)
print(newSet)

//交集 intersection,返回两个集合的共同部分的集合
let intersectionSet = hashSet.intersection(set1)
print("intersection \(intersectionSet)")

//不相交,判断两个集合是否不相交, 相交则返回false
let isDisjointSet = hashSet.isDisjoint(with: set1)
print(isDisjointSet)

字典 Dictionary

  1. Dictionary 使用键值对(key-value pair)组织其内容
  2. Dictionary的键值必需是唯一的

Dictionary字典的四种初始化方式

var dict1 : Dictionary<String , Integer> = [:]
var dict2 :[String:Integer] = [:]
var dict3 = Dictionary<String,Integer>()
var dict4 = [String : Integer]()

字典初始化的类型推断

var dict :Dictionary<String,Int> = ["Tom":12 , "Jerry":18]
var dict2 = ["Tom":12 , "Jerry":18]

字典value的修改

var dict :Dictionary<String,Int> = ["Tom":12 , "Jerry":18]
//var dict2 = ["Tom":12 , "Jerry":18] //字典初始化的类型推断
print(dict)

////访问和修改字典内容 ,key类型默认应为可空的数据n类型
print(dict["Tom"])  //输出内容为 Optional(12)
print(dict["jj"])  //输出内容为 nil
dict["Tom"] = 66  //修改字典的值
print(dict["Tom"])
dict.updateValue(99, forKey: "Jerry") //使用updateValued方法修改字典中指定键的值

新建字典元素

//print("新建字典元素")
//如字典中key为“Jake”的元素不存在则新建,否则则修改;
//测试发现字典是无序的,插入的字典值不一定在字典的最末,也可能插入到中间
//dict["Jake"] = 25
//print(dict)
////删除字典元素
//print("删除字典元素:removeValue(forkey: ..)")
//dict.removeValue(forKey: "Jake")
//print(dict)

循环遍历字典

//循环遍历字典
for (key,value) in dict{
    print("The name is \(key), age is \(value)")
}

//遍历字典键值
for key in dict.keys
{
    print("the dict key is \(key)")
}

//只遍历字典的值
for value in dict.values
{
    print("the dict value is \(value)")
}

声明常量字典,不可再进行编辑

let album = ["he":11 , "me":99 , "she":88]
print(album.count)

把字典转为数组 Array

var keys = Array(dict.keys)
print(keys)   //["Tom", "Jerry"]
var values = Array(dict.values)
print(values)  //[12, 18]

value为数组的字典

//创建一个字典用于存储程序开发需要语言,如 key为ios,value为 ["Objective-c","Swift"]
print("创建一个字典用于存储程序开发需要语言")
var codeDict = ["ios":["Objective-c","Swift"] , "android":["java","kotlin"]]
print(codeDict)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容