coding-style

TypeScript/JavaScript 编码风格

此文件扩展了 ,包含 TypeScript/JavaScript 特定内容。

类型与接口

使用类型使公共 API、共享模型和组件 props 显式、可读且可复用。

公共 API

  • 为导出函数、共享工具和公共类方法添加参数和返回类型
  • 让 TypeScript 推断明显的局部变量类型
  • 将重复的内联对象形状提取为命名类型或接口
typescript
// 错误:导出函数无显式类型
export function formatUser(user) {
  return `${user.firstName} ${user.lastName}`
}

// 正确:公共 API 有显式类型
interface User {
  firstName: string
  lastName: string
}

export function formatUser(user: User): string {
  return `${user.firstName} ${user.lastName}`
}

接口 vs 类型别名

  • 使用 interface 定义可能被扩展或实现的对象形状
  • 使用 type 定义联合、交叉、元组、映射类型和工具类型
  • 优先使用字符串字面量联合而非 enum,除非需要 enum 进行互操作
typescript
interface User {
  id: string
  email: string
}

type UserRole = 'admin' | 'member'
type UserWithRole = User & {
  role: UserRole
}

避免 any

  • 应用代码中避免使用 any
  • 对外部或不可信输入使用 unknown,然后安全收窄
  • 当值类型依赖调用者时使用泛型
typescript
// 错误:any 移除了类型安全
function getErrorMessage(error: any) {
  return error.message
}

// 正确:unknown 强制安全收窄
function getErrorMessage(error: unknown): string {
  if (error instanceof Error) {
    return error.message
  }

  return 'Unexpected error'
}

React Props

  • 使用命名 interfacetype 定义组件 props
  • 显式类型回调 props
  • 无特定理由时不使用 React.FC
typescript
interface User {
  id: string
  email: string
}

interface UserCardProps {
  user: User
  onSelect: (id: string) => void
}

function UserCard({ user, onSelect }: UserCardProps) {
  return <button onClick={() => onSelect(user.id)}>{user.email}</button>
}

JavaScript 文件

  • .js.jsx 文件中,当类型能提升清晰度且迁移 TypeScript 不切实际时使用 JSDoc
  • 保持 JSDoc 与运行时行为一致
javascript
/**
 * @param {{ firstName: string, lastName: string }} user
 * @returns {string}
 */
export function formatUser(user) {
  return `${user.firstName} ${user.lastName}`
}

不可变性

使用展开运算符进行不可变更新:

typescript
interface User {
  id: string
  name: string
}

// 错误:可变操作
function updateUser(user: User, name: string): User {
  user.name = name // 可变性!
  return user
}

// 正确:不可变性
function updateUser(user: Readonly<User>, name: string): User {
  return {
    ...user,
    name
  }
}

错误处理

使用 async/await 配合 try-catch 并安全收窄 unknown 错误:

typescript
interface User {
  id: string
  email: string
}

declare function riskyOperation(userId: string): Promise<User>

function getErrorMessage(error: unknown): string {
  if (error instanceof Error) {
    return error.message
  }

  return 'Unexpected error'
}

const logger = {
  error: (message: string, error: unknown) => {
    // 替换为你的生产日志器(例如 pino 或 winston)。
  }
}

async function loadUser(userId: string): Promise<User> {
  try {
    const result = await riskyOperation(userId)
    return result
  } catch (error: unknown) {
    logger.error('Operation failed', error)
    throw new Error(getErrorMessage(error))
  }
}

输入验证

使用 Zod 进行基于 schema 的验证并从 schema 推断类型:

typescript
import { z } from 'zod'

const userSchema = z.object({
  email: z.string().email(),
  age: z.number().int().min(0).max(150)
})

type UserInput = z.infer<typeof userSchema>

const validated: UserInput = userSchema.parse(input)

Console.log

  • 生产代码中禁止 console.log 语句
  • 使用正规日志库替代
  • 自动检测见 hooks