Julia语言程序基础

Julia-lang

新兴的Julia语言,Julia 一开始就是为高性能而设计的。 Julia 程序通过 LLVM 编译成高效的多平台机器码。
Julia中文社区: https://cn.julialang.org/
Julia的诞生: 一群拥有各种语言丰富编程经验的Matlab高级用户,对现有的科学计算编程工具感到不满——这些软件对自己专长的领域表现得非常棒,但在其它领域却非常糟糕。他们想要的是一个开源的软件,它要像C语言一般快速而又拥有如同Ruby的动态性;要具有Lisp般真正的同像性而又有Matlab般熟悉的数学记号;要像Python般通用、像R般在统计分析上得心应手、像Perl般自然地处理字符串、像Matlab般具有强大的线性代数运算能力、像shell般胶水语言的能力,易于学习而又不让真正的黑客感到无聊;它应该是交互式的,同时又是编译型的JIT(Just-In-Time)。

  • julia与传统动态语言最重要的区别是:

    • 核心语言很小:标准库是使用Julia下的,包括整数运算这样的基础运算
    • 丰富的基础类型:既可用于定义和描述对象,也可用于做可选的数据标注
    • 通过多重派发,可根据类型的不同,来调用同名函数的不同实现
    • 为了不同的参数类型,自动生成高效、专用的代码
    • 接近C语言的性能

Julia 变量

Julia语言中,变量是与某个值关联的名字。你可以用它来保存一个值

x = 10 
x + 1
x = "hello world"

Julia提供了非常灵活的变量命名的策略。变量名是大小写敏感的,且不包含语义,意思就是说,Julia会根据变量的名字区别对待

julia> x = 1.0 1.0
julia> y = -3 -3
julia> Z = "My string" "My string"
julia> customary_phrase = "Hello world!" "Hello world!"
julia> UniversalDeclarationOfHumanRightsStart = " 人人生而自由,在尊严和权利上一律平等。"
  • 变量的命名

变量名字必须以英文字母开头。

命名规范:变量名字采用小写

用下划线分割命名中的单词,不鼓励使用

Type Module类型的名字使用大写字符开头,并且大写字母而不是用下划线分割单词

函数function 和 宏macro的名字使用小写,不使用下划线

会对输入参数进行更改的函数要使用叹号!结尾

  • 数值转换

Julia支持三种数值转换

​ - T(x) 和convert(T, x) 会把x转换成T类型

​ - 如果T是浮点类型,转换的结果就是最近的可表示值,可能会是正无穷大。

​ - 如果T为整数类型,当x不能由 T 类型表示时,会抛出异常InexactError

字符

char类型的值代表单个字符:他只是带有特殊文本表示法和适当算术行为的32位原始类型,不能转换为代表Unicode代码的数值,

  • 字符拼接
text = string("Hello", "World")
println(text)

>>> HelloWorld
# 在Julia中字符串拼接的+号变成了*号
println("test" * "123")
  • 计算长度
# 计算字符串的长度
println(length.(text))
>>> 10

  • 引用变量进行字符拼接
# 引用变量
println("$text" * "123")
>>> HelloWorld123
  • 三引号字符串字面量
str = """
    又来到某个港口
    我不是一定要你回来

"""
println("xix :$str")

  • 常见的操作
# findfirst 获取字符串中是否有相应的字符 如果有返回第一个匹配到的字符下标
flag = findfirst(isequal('0'), "OOOOOOOOOO0OOOOOOO0oo")
println(flag)

# findfirst 获取字符串中是否有相应的字符 如果有返回最后一个匹配到的字符下标
flag = findlast(isequal('0'), "OOOOOOOOOO0OOOOOOO0oo")
println(flag)


# findnext 字符串偏移 之后进行匹配 偏移到从第15个开始
flag = findnext(isequal('0'), "OOOOOOOOOO0OOOOOOO0oo", 15)
println(flag)


# 判断字符是否存在在字符串中 返回bool值
flag = occursin("123", "1233123133")
println(flag)

# 切片拆分 以, 间隔 
join(["apples", "bananas", "pineapples"], ", ", " and ") "apples, bananas and pineapples"

函数

在Julia中,函数是将一个参数值元祖映射到返回值的对象。Julia的函数不是说纯粹的数学函数,在某种意义上,函数可以改变并受程序的全局状态的影响。在Julia中定义函数的基本语法是:

  • 函数基本语法
function f(x, y)
    return x + y

    end
println(f(1,2))


# 函数还可以进行缩写 简洁语法
foo(x, y) = x * y
println(foo(2,3))

  • 函数指定返回类型
function bar(x, y)::Int64
    return x * y
    end

println(
bar(3, 5))
  • 元祖

Julia有一个和函数参数与返回值密切相关的内置数据结构叫元祖。一个元祖是一个固定的长度的容器,可以容纳任何值。但是不可以被修改

yuanzu = (1,2,2,3,3)
println(typeof(yuanzu))

>>> NTuple{5, Int64}
  • 具名元祖

元祖的元素可以有名字。

x = (a=2, b=2^2)
println(x.a)
# 或者 x[1]
>>> 2
  • 多返回值

Julia中一个函数可以返回多个值 ,使用元祖来表示

function foo(a, b)
    return a + b, a * b
end

println(foo(2,3))
>>>(5, 6)
  • 变参函数
function bar(a, b, x...)
    return a, b, x

end

println(bar(1, 2, 3,3,4,4))

>>>(1, 2, (3, 3, 4, 4))
  • 日期内置函数
# 导入日期库
using Dates
println(Date(2021, 1, 10))
# >>> 2021-01-10

println(Date(2021,2))
# >>> 2021-02-01

println(Date(2021))
# >>> 2021-01-01
  • 关键字参数
# 关键字参数
function plot(a, b, c=2)
    println("$a, $b, $c")

end
 
plot(6, 8, 1)
>>> 6, 8, 1

plot(6, 8)
>>> 6, 8, 2
  • 函数参数中的Do结构
map([1, 2, 3]) do x
    if x < 0&& iseven(x) # iseven功能是判断这个参数是否被2整除
        return 0
    elseif  x == 0
        return 1
    else
        return x
    end
end

# 程序会将map中的值遍历赋值为x
  • Julia读写文件操作
open("test.txt", "w") do io
    write(io, "123")
    end

# convert("需要转换的类型", 数据data)  convert可以将数据转换成相关的数据
open("test.txt", "r") do io
    line = readline(io)
    println(line)
    end

# 循环读取字符串  一定要记得if for等语句需要使用end结束
open("test.txt", "r") do io
    line = readline(io)
    println(line)
    for i in line
        println(i)
        end
    end
  • 向量化函数的点语法
# 函数向量化的.点语法 a => A  list => tsil  of => Of  strings => 7
test_list = ["a", "list", "of", "strings"] .|> [uppercase, reverse, titlecase, length]
println(test_list)


流程控制

Julia提供了大量的流程控制构件

  • 复合表达式:begin 和 ;
z = begin
    x=2
    y=3
    x+y

end
println(z)

z = (x=1; y=2; x+y)
println(z)

  • 条件表达式:if - elseif - else 和?:(三元运算符)
# if else 不做太多解释
if 条件
  println
elseif 条件
  println
else
  println
end
  • 短路求值:&& || 和链式比较
# and 和 or 也不做太多解释
• 在表达式 a && b 中,子表达式 b 仅当 a 为 true 的时候才会被执行。
• 在表达式 a || b 中,子表达式 b 仅在 a 为 false 的时候才会被执行。
  • 重复执行:循环 while for
# while 循环 从0 到100
i = 0
while i <= 100
     println(i)
     global i+=1
    end


# for 循环从1到100
for i = 1:100
    println(i)

end


# 循环列表
for i in [1,2,3]
    println(i)
end



  • 异常处理 try-catch error 和 throw

当一个意外的报错发生时,会抛出Exception。下面列出的内置exception会中断程序的正常运行流程

Exception
ArgumentError
BoundsError
CompositeException
DimensionMismatch
DivideError
DomainError
EOFError
ErrorException
InexactError
InitError
InterruptException
InvalidStateException
KeyError
注: 上述为常见的异常

​ - throw函数

​ throw函数就是Python中的raise 、Go中的panic 主动抛出异常

​ - try/catch语句

​ 通过try/catch 语句可以测试Exception并优雅的处理可能会破坏程序运行的代码。

try
  	# 可能会出现异常的代码包含在这里
    throw(InitError)

catch
    println("success")
end

# finally版本
try
    throw(InitError)

catch
    println("success")

finally
    println("finally success")
    throw(InitError)
end

  • task协程 : yieldto
详情查看https://www.geek-book.com/src/docs/julia/julia/docs.julialang.org/en/v1/base/parallel/index-2.html

变量作用域

变量的作用于是代码的一个区域,在这个区域中的这个变量是可见的。给变量划分作用于有助于解决变量命名重复冲突。这个概念是符合直觉的

Julia的变量作用域

作用于结构

结构 作用于类型 Allowed within
module , baremodule 全局 全局
struct 结构体 local(soft) 全局
for, while ,try local(soft) 全局或局部
macro local(hard) 全局
let ,functions, comprehensions, generators local(hard) 全局或局部
注:begin块和if块不会引入新的作用域块

Julia使用词法作用域,也就是说一个函数的作用域不会从其调用者的作用域继承,而从函数定义处的作用于继承。example

module Bar
    x = 1
    foo() = x
end


import .Bar
x = -1
println(Bar.foo())
>>> 1 

  • 全局作用域

    每个模块会引进新的全局作用域,与其他所有模块的全局作用域分开;无所不包的全局作用域不存在。

    模块可以把其他模块的变量引入到自己的作用域中,通过using或者import语句或者通过.点符号有这种通路,也就是每个模块都是所谓的命名空间。

  • let块

    不像局部变量的赋值行为,let语句每次运行都会新建一个变量进行绑定。赋值改变的是已存在值的位置,let会新建新的位置。这个区别通常都不重要,只会在通过闭包跳出作用于的变量的情况下能探测到。let语法接受由逗号隔开的一系列的赋值和变量名:

    let x = 1
        let
            local x=2
            println(x)
        end
        println(x)
    end
    

    通过for循环或者推倒式的迭代变量始终是一个新的变量:

    function  f()
        i = 0
        for outer i = 1:3
    
        end
        println(i)
    end
    
    f()
    
  • 变量

    变量的经常的一个使用方式是给一个特定的不变的值一个名字,这样的变量只会被赋值一次且不能修改,关键字为const。const说明只应该在全局作用域中对全局 变量使用。

    const c, d = 1, 2
    println("a:$c, b:$d")
    

数据类型

​ Julia语言系统是动态的,但通过允许指出某些变量具体的数据类型,获得了静态类型系统的一些优点。这对于生成高效的代码非常有帮助,但更重要的是,它允许针对函数参数类型的方法派发与语言深度集成。方法派发将在方法中详细探讨,但他根植于此系统的类型系统。

​ 在类型被省略时,Julia默认行为是允许变量可以为任何类型。因此可以像Python一样无需明确的声明变量的数据类型.

类型声明

:: 运算符可以用来在程序中声明类型也就是 类型注释 可以提升性能 , 在某些场景下

println(1+2 :: Int)

function foo()::Int8

#     a :: String = "211"
    return 122
end

println(typeof(foo()))
>>> Int8

抽象类型

抽象类型不能实例化,只能作为类型图中的节点使用,从而描述由相关具体类型组成的集合:那些作为其后代的具体类型。

Julia的变量作用域

作用于结构

结构 作用于类型 Allowed within
module , baremodule 全局 全局
struct 结构体 local(soft) 全局
for, while ,try local(soft) 全局或局部
macro local(hard) 全局
let ,functions, comprehensions, generators local(hard) 全局或局部

Julia与Python的区别

  • 切片 Julia 索引从1开始 Python从0开始
  • Python切片左开右闭 julia 左右全闭
  • Julia不支持负数索引
  • Julia的条件语句 使用end结束 Python是以缩进结束

持续更新ing...

posted @ 2021-04-16 15:13  听风走了八千里  阅读(1315)  评论(0编辑  收藏  举报