通过前面JS的学习,想必大家对函数已经非常了解了。 TS 定义函数的方法和 JS 基本一样,不同的是 TS 可以要求有无返回值。
接下来就介绍一下ts中有关函数的一些东西。
一、函数的定义
1.es5定义函数的方法
//函数声明法
function run(){
return 'run';
}
//匿名函数
var run2=function(){
return 'run2';
}
2.ts中定义函数
接下来我们同样写一个计算两个数之和的方法:
function total(one: number, two: number) {
return one + two;
}
const total = getTotal(1, 2);
这时我们并没有定义total
的返回值类型,虽然TypeScript
可以自己推断出返回值是number
类型。 但是如果这时候我们的代码不小心写错了,比如:
function total(one: number, two: number) {
return one + two + "i love you";
}
const total = total(1, 2);
这时候total
的值就不是number
类型了,但是不会报错。为了保证我们的结果也是number类型,有的时候你可以这样写:
const count: number = total(1, 2);
这样写虽然可以让编辑器报错,但是代码不够严谨,没有解决根本原因,这是total()
函数的错误。所以合适的做法是给函数的返回值加上类型注解。
在ts中需要指定函数的返回值类型。
function total(one: number, two: number): number {
return one + two;
}
const total = total(1, 2);
这样代码就比较严谨了。
二、定义函数返回值
1.ts中定义函数返回值
//函数声明法
function run():string{
return 'run';
}
//错误写法
function run():string{
return 123;
}
//匿名函数
var fun2=function():number{
return 123;
}
alert(fun2()); /*调用方法*/
2.函数无返回值
类型为:void
function run():void{
console.log('run'); //只打印出‘run’,没有返回值
}
run();
3.never
返回类型
never
返回类型前面已经介绍过了。 如果一个函数是永远也执行不完的,就可以定义返回值为never
,那什么样的函数是永远也执行不完的那?比如执行的时候,抛出了异常,这时候就无法执行完了。
function errF(): never {
throw new Error();
console.log("this is error");
}
还有一种是一直循环,也是我们常说的死循环,这样也运行不完,比如下面的代码:
function forever(): never {
while (true) {}
console.log("Hello,honey");
}
三、函数参数
- 可选参数
- 默认参数
- 剩余参数
1.ts中方法传参
方法传参刚才已经提到了,在传参的时候指定参数的类型:
function getInfo(name:string,age:number):string{
return `${name} --- ${age}`;
}
alert(getInfo('zhangsan',20));
2.参数为对象(解构)时
比如我们写一个对象参数:
function add({ num1, num2 }) {
return num1 + num2;
}
const total = add({ num1: 1, num2: 2 });
这时直接报错了,意思是total
有可能会是任何类型,那我们要如何给这样的参数加类型注解?
function add({ num1: number, num2: number }) {
return num1 + num2;
}
const total = add({ num1: 1, num2: 2 });
你在编辑器中会看到这种写法是完全错误的。那正确的写法应该是这样的。
function add({ num1, num2}:{num1:number,num2:number}):number {
return num1 + num2;
}
const total = add({ num1: 1, num2: 2 });
console.log(total); //3
解构一个函数的参数是 对象 的方式是:在后面跟着一个对象类型注解.
如果参数只有一个属性时,应该这样写:
function getcount({ one }: { one: number }): number {
return one;
}
const num = getcount({ one: 1 });
3.可选参数
es5里面方法的实参和行参可以不一样,但是ts中必须一样,如果不一样就需要配置可选参数
function getWorker(name:string,age?:number):string{
if(age){
return `我的名字是:${name} --- 我的年龄是:${age}`;
}else{
return `我的名字是:${name} --- 我的年龄是保密`;
}
}
console.log(getWorker('孙玉鑫')); //我的名字是:孙玉鑫 --- 我的年龄是保密
console.log(getWorker('潘小小',19)); //我的名字是:潘小小 --- 我的年龄是:19
注意:可选参数必须配置到参数的最后面
//错误的写法
function getWorker(name?:string,age:number):string{
if(age){
return `我的名字是:${name} --- 我的年龄是:${age}`;
}else{
return `我的名字是:${name} --- 我的年龄是保密`;
}
}
console.log(getWorker('谭成铭'));
4.默认参数
es5里面没法设置默认参数,es6和ts中都可以设置默认参数
function getCodeFarmer(name:string,age:number=22):string{
if(age==22){
return `我的名字是:${name} ---我的年龄是默认值:${age},我是一个码农`;
}else{
return `我的名字是:${name} ---我的年龄是:${age},我是一个码农`;
}
}
console.log( getCodeFarmer('代高飞'));//我的名字是:代高飞 ---我的年龄是默认值:22,我是一个码农
console.log( getCodeFarmer('高威',15)); //我的名字是:高威 ---我的年龄是:15,我是一个码农
5.剩余参数
刚才我们讨论的默认参数和可选参数有个共同点:它们表示某一个参数。 有时,你想同时操作多个参数,或者你并不知道会有多少参数传递进来。 在JavaScript里,你可以使用 arguments来访问所有传入的参数。在TypeScript里,你可以把所有参数收集到一个变量里。剩余参数会被当做个数不限的可选参数。 可以一个都没有,同样也可以有任意个。三点运算符 接受新参传过来的值。
比如写一个多个数相加的函数
function sum(a:number,b:number,c:number,d:number):number{
return a+b+c+d;
}
console.log(sum(1,2,3,4))
如果还有很多我们不确定的数相加,可以这样写:
function sum(...result:number[]):number{
var sum=0;
for(var i=0;i<result.length;i++){
sum+=result[i];
}
return sum;
}
console.log(sum(1,2,3,4,5,6)) ;
普通参数可以和剩余参数同时传递:
function sum(a:number,b:number,...result:number[]):number{
var sum=a+b;
for(var i=0;i<result.length;i++){
sum+=result[i];
}
return sum;
}
console.log(sum(1,2,3,4,5,6)) ;
6.ts函数重载
typescript中的重载:通过为同一个函数提供多个函数类型定义来实现多种功能的目的。
在JS中依据不同参数类型或参数个数执行一些不同函数体的实现很常见,在TypeScript中,会有需要用到这种声明的地方。
关于函数重载,必须要把精确的定义放在前面,最后函数实现时,需要使用 |
操作符或者?
操作符,把所有可能的输入类型全部包含进去,以具体实现。
ts为了兼容es5 以及 es6 重载的写法和java中有区别。
//es5中出现同名方法,下面的会替换上面的方法
function getName(name){
return name;
}
function getName(name,work){
return name+work;
}
在ts中我们写一个add函数,它可以接收string
类型的参数进行拼接,也可以接收number类型的参数进行相加。
/函数重载
// 声明
function some (arg1: string, arg2: string): string
function some (arg1: number, arg2: number): number
// 具体实现
function some (arg1: string | number, arg2: string | number) {
// 在实现上我们要注意严格判断两个参数的类型是否相等,而不能简单的写一个 arg1 + arg2
if (typeof arg1 === 'string' && typeof arg2 === 'string') {
return `拼接两个字符串:${arg1} + ${arg2}`
} else if (typeof arg1 === 'number' && typeof arg2 === 'number') {
return arg1 + arg2
}
}
console.log(some("皮皮虾","跟我走")); //拼接两个字符串:皮皮虾 + 跟我走
console.log(some(12,12)); //24
console.log(some("你好",25)) //错误
function getUser(name:string):string;
function getUser(name:string,age:number):string;
function getUser(name:any,age?:any):any{
if(age){
return '我叫:'+name+'我的年龄是'+age;
}else{
return '我叫:'+name;
}
}
console.log(getUser('曹西西')); /*正确*/
console.log(getUser(123)); //错误
console.log(getUser('韩尚尚',20)); //正确
TypeScript 中的函数重载也只是多个函数的声明,具体的逻辑还需要自己去写,它并不会真的将你的多个重名 function 的函数体进行合并。
interface Uodefarmer {
name: string;
age: number;
work:string;
}
interface ProductManager {
name: string;
age: number;
work:string;
rest:string;
}
declare function doSomething(params: Uodefarmer | ProductManager, flag?: boolean): string;
在这个 doSomething
函数里,我们的目的是当传入参数 params是 Uodefarmer(码农)时,不传 flag,当传入 params是 ProductManager(产品经理时) 时,传入 flag。TypeScript 并不知道这些,当你传入 params为 Uodefarmer时,flag 同样允许你传入:
const user={
name:"王冲冲",
age:10,
work:"码农"
}
// 没有报错,但是不符合我们的设想
doSomething(user,true)
使用函数重载能帮助我们实现:
interface Ucodefarmer {
name: string;
age: number;
work:string;
}
interface ProductManager {
name: string;
age: number;
work:string;
rest:string;
}
declare function doSomething(params: Ucodefarmer): string;
declare function doSomething(params: ProductManager, flag: boolean): string;
const user={
name:"王冲冲",
age:10,
work:"码农"
}
const user2={
name:"张三",
age:10,
work:"码农",
rest:"ahhah"
}
doSomething(user,true) //error
doSomething(user2,true) //正确
函数重载的意义在于能够让你知道传入不同的参数得到不同的结果,如果传入的参数不同,但是得到的结果(类型)却相同,那么这里就不要使用函数重载(没有意义)。如果函数的返回值类型相同,那么就不需要使用函数重载
function func (a: number): number
function func (a: number, b: number): number
// 像这样的是参数个数的区别,我们可以使用可选参数来代替函数重载的定义
function func (a: number, b?: number): number
// 注意第二个参数在类型前边多了一个`?`
// 或是一些参数类型的区别导致的
function func (a: number): number
function func (a: string): number
// 这时我们应该使用联合类型来代替函数重载
function func (a: number | string): number