39. I2C实验

ops/2025/2/3 19:04:40/

一、IIC协议详解

1、ALPHA开发板上有个AP3216C,这是一个IIC接口的器件,这是一个环境光传感器。AP3216C连接到了I2C1上:
I2C1_SCL: 使用的是UART4_TXD这个IO,复用位ALT2
I2C1_SDA: 使用的是UART4_RXD这个IO。复用为ALT2

2、I2C分为SCL和SDA,这两个必须要接上拉电阻到VCC,比如3.3V,一般是4.7K上拉电阻。
3、I2C总线支持多从机,通过从机地址来区分访问哪个从机。

二、6ULL I2C接口详解

1、6UL的I2C频率标准模式100kbit/S,快速模式400Kbit/S
2、时钟源选择perclk_clk_root=ipg_clk_root=66MHz
3、IFDR寄存器设置I2C频率,bit5:0设置频分值,假如我们现在需要100kbit的速率,那么66000000/100000=660。经过查找IC位设置位0X38或0X15的时候,为640分频,66000000/640=103.125Kbit.
4、I2CR寄存器,bit7为I2C使能位,置1使能I2C。bit5为主从模式选择位,为0表示从机,为1表示主机。Bit4为发送/接收设置位,为0的时候是接收,为1的时候是发送
5、I2SR寄存器,bit7为传输完成位,为0表示正在发送,为1表示发送完成。Bit5是I2C忙闲位,为0的时候I2C总线空闲,为1的时候I2C总线忙。Bit0是读确认位,也就是ACK信号
6、I2DR寄存器,数据寄存器。

三、AP3216C简介

1、AP3216C是一个三合一的环境光传感器,ALS+PS+IRLED,ALS是环境光,PS是接近传感器,IR是红外LED灯。I2C接口,最高400Kbit/S的频率。
2、环境光,ALS是16位输出。
3、接近传感器PS,10bit输出。IR传感器也是10bit
4、AP3216C的从机地址位0X1E。
5、0X0A是IR Ddata low。Bit7为0的时候表示IR和PS数据有效,为1的时候IR和PS数据无效。Bit1:0是IR的低2位。
6、0X0B是IR Data high,big7:0是高字节。与0X0A一起组成10bit的数据。
7、0X0C和0X0D分别位ALS的低8位和高8位。
8、0X0E的bit3:0是低4位数据,0X0F的bit5:0是高6位数据。加起来就是10位
9、0X00是系统配置寄存器,bit2:0设置AP3216C开始那些传感器,我们需要设置位011,也就是0x3,表示开始ALS+PS+IR。

四、实验程序编写

//bsp_i2c.c

#include "bsp_i2c.h"
#include "bsp_delay.h"
#include "stdio.h"/** @description     : 初始化I2C,波特率100KHZ* @param - base    : 要初始化的IIC设置* @return          : 无*/
void i2c_init(I2C_Type *base)
{/* 1、配置I2C */base->I2CR &= ~(1 << 7); /* 要访问I2C的寄存器,首先需要先关闭I2C *//* 设置波特率为100K* I2C的时钟源来源于IPG_CLK_ROOT=66Mhz* IC2 时钟 = PERCLK_ROOT/dividison(IFDR寄存器)* 设置寄存器IFDR,IFDR寄存器参考IMX6UL参考手册P1260页,表29-3,* 根据表29-3里面的值,挑选出一个还是的分频数,比如本例程我们* 设置I2C的波特率为100K, 因此当分频值=66000000/100000=660.* 在表29-3里面查找,没有660这个值,但是有640,因此就用640,* 即寄存器IFDR的IC位设置为0X15*/base->IFDR = 0X15 << 0;/** 设置寄存器I2CR,开启I2C* bit[7] : 1 使能I2C,I2CR寄存器其他位其作用之前,此位必须最先置1*/base->I2CR |= (1<<7);
}/** @description         : 发送重新开始信号* @param - base        : 要使用的IIC* @param - addrss      : 设备地址* @param - direction   : 方向* @return              : 0 正常 其他值 出错*/
unsigned char i2c_master_repeated_start(I2C_Type *base, unsigned char address,  enum i2c_direction direction)
{/* I2C忙并且工作在从模式,跳出 */if(base->I2SR & (1 << 5) && (((base->I2CR) & (1 << 5)) == 0))       return 1;/** 设置寄存器I2CR* bit[4]: 1 发送* bit[2]: 1 产生重新开始信号*/base->I2CR |=  (1 << 4) | (1 << 2);/** 设置寄存器I2DR* bit[7:0] : 要发送的数据,这里写入从设备地址*            参考资料:IMX6UL参考手册P1249*/ base->I2DR = ((unsigned int)address << 1) | ((direction == kI2C_Read)? 1 : 0);return 0;
}/** @description         : 发送开始信号* @param - base        : 要使用的IIC* @param - addrss      : 设备地址* @param - direction   : 方向* @return              : 0 正常 其他值 出错*/
unsigned char i2c_master_start(I2C_Type *base, unsigned char address,  enum i2c_direction direction)
{if(base->I2SR & (1 << 5))           /* I2C忙 */return 1;/** 设置寄存器I2CR* bit[5]: 1 主模式* bit[4]: 1 发送*/base->I2CR |=  (1 << 5) | (1 << 4);/** 设置寄存器I2DR* bit[7:0] : 要发送的数据,这里写入从设备地址*            参考资料:IMX6UL参考手册P1249*/ base->I2DR = ((unsigned int)address << 1) | ((direction == kI2C_Read)? 1 : 0);return 0;
}/** @description     : 检查并清除错误* @param - base    : 要使用的IIC* @param - status  : 状态* @return          : 状态结果*/
unsigned char i2c_check_and_clear_error(I2C_Type *base, unsigned int status)
{/* 检查是否发生仲裁丢失错误 */if(status & (1<<4)){base->I2SR &= ~(1<<4);      /* 清除仲裁丢失错误位            */base->I2CR &= ~(1 << 7);    /* 先关闭I2C               */base->I2CR |= (1 << 7);     /* 重新打开I2C              */return I2C_STATUS_ARBITRATIONLOST;} else if(status & (1 << 0))      /* 没有接收到从机的应答信号 */{return I2C_STATUS_NAK;      /* 返回NAK(No acknowledge) */}return I2C_STATUS_OK;
}/** @description     : 停止信号* @param - base    : 要使用的IIC* @param           : 无* @return          : 状态结果*/
unsigned char i2c_master_stop(I2C_Type *base)
{unsigned short timeout = 0xffff;/** 清除I2CR的bit[5:3]这三位*/base->I2CR &= ~((1 << 5) | (1 << 4) | (1 << 3));/* 等待忙结束 */while((base->I2SR & (1 << 5))){timeout--;if(timeout == 0)    /* 超时跳出 */return I2C_STATUS_TIMEOUT;}return I2C_STATUS_OK;
}/** @description     : 发送数据* @param - base    : 要使用的IIC* @param - buf     : 要发送的数据* @param - size    : 要发送的数据大小* @param - flags   : 标志* @return          : 无*/
void i2c_master_write(I2C_Type *base, const unsigned char *buf, unsigned int size)
{/* 等待传输完成 */while(!(base->I2SR & (1 << 7))); base->I2SR &= ~(1 << 1);    /* 清除标志位 */base->I2CR |= 1 << 4;       /* 发送数据 */while(size--){base->I2DR = *buf++;    /* 将buf中的数据写入到I2DR寄存器 */while(!(base->I2SR & (1 << 1)));    /* 等待传输完成 */    base->I2SR &= ~(1 << 1);            /* 清除标志位 *//* 检查ACK */if(i2c_check_and_clear_error(base, base->I2SR))break;}base->I2SR &= ~(1 << 1);i2c_master_stop(base);  /* 发送停止信号 */
}/** @description     : 读取数据* @param - base    : 要使用的IIC* @param - buf     : 读取到数据* @param - size    : 要读取的数据大小* @return          : 无*/
void i2c_master_read(I2C_Type *base, unsigned char *buf, unsigned int size)
{volatile uint8_t dummy = 0;dummy++;    /* 防止编译报错 *//* 等待传输完成 */while(!(base->I2SR & (1 << 7))); base->I2SR &= ~(1 << 1);                /* 清除中断挂起位 */base->I2CR &= ~((1 << 4) | (1 << 3));   /* 接收数据 *//* 如果只接收一个字节数据的话发送NACK信号 */if(size == 1)base->I2CR |= (1 << 3);dummy = base->I2DR; /* 假读 */while(size--){while(!(base->I2SR & (1 << 1)));    /* 等待传输完成 */    base->I2SR &= ~(1 << 1);            /* 清除标志位 */if(size == 0){i2c_master_stop(base);          /* 发送停止信号 */}if(size == 1){base->I2CR |= (1 << 3);}*buf++ = base->I2DR;}
}/** @description : I2C数据传输,包括读和写* @param - base: 要使用的IIC* @param - xfer: 传输结构体* @return      : 传输结果,0 成功,其他值 失败;*/
unsigned char i2c_master_transfer(I2C_Type *base, struct i2c_transfer *xfer)
{unsigned char ret = 0;enum i2c_direction direction = xfer->direction;    base->I2SR &= ~((1 << 1) | (1 << 4));           /* 清除标志位 *//* 等待传输完成 */while(!((base->I2SR >> 7) & 0X1)){}; /* 如果是读的话,要先发送寄存器地址,所以要先将方向改为写 */if ((xfer->subaddressSize > 0) && (xfer->direction == kI2C_Read)){direction = kI2C_Write;}ret = i2c_master_start(base, xfer->slaveAddress, direction); /* 发送开始信号 */if(ret){   return ret;}while(!(base->I2SR & (1 << 1))){};          /* 等待传输完成 */ret = i2c_check_and_clear_error(base, base->I2SR);  /* 检查是否出现传输错误 */if(ret){i2c_master_stop(base);                      /* 发送出错,发送停止信号 */return ret;}/* 发送寄存器地址 */if(xfer->subaddressSize){do{base->I2SR &= ~(1 << 1);            /* 清除标志位 */xfer->subaddressSize--;             /* 地址长度减一 */base->I2DR =  ((xfer->subaddress) >> (8 * xfer->subaddressSize)); //向I2DR寄存器写入子地址while(!(base->I2SR & (1 << 1)));    /* 等待传输完成 *//* 检查是否有错误发生 */ret = i2c_check_and_clear_error(base, base->I2SR);if(ret){i2c_master_stop(base);              /* 发送停止信号 */return ret;}  } while ((xfer->subaddressSize > 0) && (ret == I2C_STATUS_OK));if(xfer->direction == kI2C_Read)        /* 读取数据 */{base->I2SR &= ~(1 << 1);            /* 清除中断挂起位 */i2c_master_repeated_start(base, xfer->slaveAddress, kI2C_Read); /* 发送重复开始信号和从机地址 */while(!(base->I2SR & (1 << 1))){};/* 等待传输完成 *//* 检查是否有错误发生 */ret = i2c_check_and_clear_error(base, base->I2SR);if(ret){ret = I2C_STATUS_ADDRNAK;i2c_master_stop(base);      /* 发送停止信号 */return ret;  }}}   /* 发送数据 */if ((xfer->direction == kI2C_Write) && (xfer->dataSize > 0)){i2c_master_write(base, xfer->data, xfer->dataSize);}/* 读取数据 */if ((xfer->direction == kI2C_Read) && (xfer->dataSize > 0)){i2c_master_read(base, xfer->data, xfer->dataSize);}return 0;   
}

//bsp_i2c.h

#ifndef _BSP_I2C_H
#define _BSP_I2C_H
#include "imx6ul.h"/* 相关宏定义 */
#define I2C_STATUS_OK               (0)
#define I2C_STATUS_BUSY             (1)
#define I2C_STATUS_IDLE             (2)
#define I2C_STATUS_NAK              (3)
#define I2C_STATUS_ARBITRATIONLOST  (4)
#define I2C_STATUS_TIMEOUT          (5)
#define I2C_STATUS_ADDRNAK          (6)/** I2C方向枚举类型*/
enum i2c_direction
{kI2C_Write = 0x0,       /* 主机向从机写数据 */kI2C_Read = 0x1,        /* 主机从从机读数据 */
} ;/** 主机传输结构体*/
struct i2c_transfer
{unsigned char slaveAddress;         /* 7位从机地址           */enum i2c_direction direction;       /* 传输方向             */unsigned int subaddress;            /* 寄存器地址            */unsigned char subaddressSize;       /* 寄存器地址长度          */unsigned char *volatile data;       /* 数据缓冲区            */volatile unsigned int dataSize;     /* 数据缓冲区长度          */
};/**函数声明*/
void i2c_init(I2C_Type *base);
unsigned char i2c_master_start(I2C_Type *base, unsigned char address, enum i2c_direction direction);
unsigned char i2c_master_repeated_start(I2C_Type *base, unsigned char address,  enum i2c_direction direction);
unsigned char i2c_check_and_clear_error(I2C_Type *base, unsigned int status);
unsigned char i2c_master_stop(I2C_Type *base);
void i2c_master_write(I2C_Type *base, const unsigned char *buf, unsigned int size);
void i2c_master_read(I2C_Type *base, unsigned char *buf, unsigned int size);
unsigned char i2c_master_transfer(I2C_Type *base, struct i2c_transfer *xfer);#endif

//bsp_ap3216c.c

#include "bsp_ap3216c.h"
#include "bsp_i2c.h"
#include "bsp_delay.h"
#include "cc.h"
#include "stdio.h"/** @description : 初始化AP3216C* @param       : 无* @return      : 0 成功,其他值 错误代码*/
unsigned char ap3216c_init(void)
{unsigned char data = 0;/* 1、IO初始化,配置I2C IO属性   * I2C1_SCL -> UART4_TXD* I2C1_SDA -> UART4_RXD*/IOMUXC_SetPinMux(IOMUXC_UART4_TX_DATA_I2C1_SCL, 1);IOMUXC_SetPinMux(IOMUXC_UART4_RX_DATA_I2C1_SDA, 1);/* *bit 16:0 HYS关闭*bit [15:14]: 1 默认47K上拉*bit [13]: 1 pull功能*bit [12]: 1 pull/keeper使能 *bit [11]: 0 关闭开路输出*bit [7:6]: 10 速度100Mhz*bit [5:3]: 110 驱动能力为R0/6*bit [0]: 1 高转换率*/IOMUXC_SetPinConfig(IOMUXC_UART4_TX_DATA_I2C1_SCL, 0x70B0);IOMUXC_SetPinConfig(IOMUXC_UART4_RX_DATA_I2C1_SDA, 0X70B0);i2c_init(I2C1);     /* 初始化I2C1 *//* 2、初始化AP3216C */ap3216c_writeonebyte(AP3216C_ADDR, AP3216C_SYSTEMCONG, 0X04);   /* 复位AP3216C            */delayms(50);                                                    /* AP33216C复位至少10ms */ap3216c_writeonebyte(AP3216C_ADDR, AP3216C_SYSTEMCONG, 0X03);   /* 开启ALS、PS+IR          */data = ap3216c_readonebyte(AP3216C_ADDR, AP3216C_SYSTEMCONG);   /* 读取刚刚写进去的0X03 */if(data == 0X03)return 0;   /* AP3216C正常    */else return 1;   /* AP3216C失败    */
}/** @description : 向AP3216C写入数据* @param - addr: 设备地址* @param - reg : 要写入的寄存器* @param - data: 要写入的数据* @return      : 操作结果*/
unsigned char ap3216c_writeonebyte(unsigned char addr,unsigned char reg, unsigned char data)
{unsigned char status=0;unsigned char writedata=data;struct i2c_transfer masterXfer;/* 配置I2C xfer结构体 */masterXfer.slaveAddress = addr;             /* 设备地址                 */masterXfer.direction = kI2C_Write;          /* 写入数据                 */masterXfer.subaddress = reg;                /* 要写入的寄存器地址            */masterXfer.subaddressSize = 1;              /* 地址长度一个字节             */masterXfer.data = &writedata;               /* 要写入的数据               */masterXfer.dataSize = 1;                    /* 写入数据长度1个字节           */if(i2c_master_transfer(I2C1, &masterXfer))status=1;return status;
}/** @description : 从AP3216C读取一个字节的数据* @param - addr: 设备地址* @param - reg : 要读取的寄存器* @return      : 读取到的数据。*/
unsigned char ap3216c_readonebyte(unsigned char addr,unsigned char reg)
{unsigned char val=0;struct i2c_transfer masterXfer; masterXfer.slaveAddress = addr;             /* 设备地址                 */masterXfer.direction = kI2C_Read;           /* 读取数据                 */masterXfer.subaddress = reg;                /* 要读取的寄存器地址            */masterXfer.subaddressSize = 1;              /* 地址长度一个字节             */masterXfer.data = &val;                     /* 接收数据缓冲区              */masterXfer.dataSize = 1;                    /* 读取数据长度1个字节           */i2c_master_transfer(I2C1, &masterXfer);return val;
}/** @description : 读取AP3216C的数据,读取原始数据,包括ALS,PS和IR, 注意!*              : 如果同时打开ALS,IR+PS的话两次数据读取的时间间隔要大于112.5ms* @param - ir  : ir数据* @param - ps  : ps数据* @param - ps  : als数据 * @return      : 无。*/
void ap3216c_readdata(unsigned short *ir, unsigned short *ps, unsigned short *als)
{unsigned char buf[6];unsigned char i;/* 循环读取所有传感器数据 */for(i = 0; i < 6; i++)  {buf[i] = ap3216c_readonebyte(AP3216C_ADDR, AP3216C_IRDATALOW + i);  }if(buf[0] & 0X80)   /* IR_OF位为1,则数据无效 */*ir = 0;                    else                /* 读取IR传感器的数据           */*ir = ((unsigned short)buf[1] << 2) | (buf[0] & 0X03);          *als = ((unsigned short)buf[3] << 8) | buf[2];  /* 读取ALS传感器的数据           */  if(buf[4] & 0x40)   /* IR_OF位为1,则数据无效           */*ps = 0;                                                        else                /* 读取PS传感器的数据    */*ps = ((unsigned short)(buf[5] & 0X3F) << 4) | (buf[4] & 0X0F);     
}

//bsp_ap3216c.h

#ifndef _BSP_AP3216C_H
#define _BSP_AP3216C_H
#include "imx6ul.h"#define AP3216C_ADDR        0X1E    /* AP3216C器件地址 *//* AP3316C寄存器 */
#define AP3216C_SYSTEMCONG  0x00    /* 配置寄存器            */
#define AP3216C_INTSTATUS   0X01    /* 中断状态寄存器          */
#define AP3216C_INTCLEAR    0X02    /* 中断清除寄存器          */
#define AP3216C_IRDATALOW   0x0A    /* IR数据低字节          */
#define AP3216C_IRDATAHIGH  0x0B    /* IR数据高字节          */
#define AP3216C_ALSDATALOW  0x0C    /* ALS数据低字节         */
#define AP3216C_ALSDATAHIGH 0X0D    /* ALS数据高字节         */
#define AP3216C_PSDATALOW   0X0E    /* PS数据低字节          */
#define AP3216C_PSDATAHIGH  0X0F    /* PS数据高字节          *//* 函数声明 */
unsigned char ap3216c_init(void);
unsigned char ap3216c_readonebyte(unsigned char addr,unsigned char reg);
unsigned char ap3216c_writeonebyte(unsigned char addr,unsigned char reg, unsigned char data);
void ap3216c_readdata(unsigned short *ir, unsigned short *ps, unsigned short *als);#endif

//main.c

/**************************************************************
描述     : I.MX6U开发板裸机实验18 IIC实验
其他     : IIC是最常用的接口,ZERO开发板上有多个IIC外设,本实验就来学习如何驱动I.MX6U的IIC接口,并且通过IIC接口读取板载AP3216C的数据值。
**************************************************************/
#include "bsp_clk.h"
#include "bsp_delay.h"
#include "bsp_led.h"
#include "bsp_beep.h"
#include "bsp_key.h"
#include "bsp_int.h"
#include "bsp_uart.h"
#include "bsp_lcd.h"
#include "bsp_rtc.h"
#include "bsp_ap3216c.h"
#include "stdio.h"/** @description : main函数* @param       : 无* @return      : 无*/
int main(void)
{unsigned short ir, als, ps;unsigned char state = OFF;int_init();                 /* 初始化中断(一定要最先调用!) */imx6u_clkinit();            /* 初始化系统时钟          */delay_init();               /* 初始化延时            */clk_enable();               /* 使能所有的时钟          */led_init();                 /* 初始化led           */beep_init();                /* 初始化beep          */uart_init();                /* 初始化串口,波特率115200 */lcd_init();                 /* 初始化LCD           */tftlcd_dev.forecolor = LCD_RED; lcd_show_string(30, 50, 200, 16, 16, (char*)"ZERO-IMX6U IIC TEST");  lcd_show_string(30, 70, 200, 16, 16, (char*)"AP3216C TEST");  lcd_show_string(30, 90, 200, 16, 16, (char*)"ATOM@ALIENTEK");  lcd_show_string(30, 110, 200, 16, 16, (char*)"2019/3/26");  while(ap3216c_init())       /* 检测不到AP3216C */{lcd_show_string(30, 130, 200, 16, 16, (char*)"AP3216C Check Failed!");delayms(500);lcd_show_string(30, 130, 200, 16, 16, (char*)"Please Check!        ");delayms(500);}   lcd_show_string(30, 130, 200, 16, 16, (char*)"AP3216C Ready!");  lcd_show_string(30, 160, 200, 16, 16, (char*)" IR:");    lcd_show_string(30, 180, 200, 16, 16, (char*)" PS:");   lcd_show_string(30, 200, 200, 16, 16, (char*)"ALS:");   tftlcd_dev.forecolor = LCD_BLUE;    while(1)                    {ap3216c_readdata(&ir, &ps, &als);       /* 读取数据         */lcd_shownum(30 + 32, 160, ir, 5, 16);   /* 显示IR数据       */lcd_shownum(30 + 32, 180, ps, 5, 16);   /* 显示PS数据       */lcd_shownum(30 + 32, 200, als, 5, 16);  /* 显示ALS数据  */ delayms(120);state = !state;led_switch(LED0,state); }return 0;
}

http://www.ppmy.cn/ops/155376.html

相关文章

doris:主键模型的更新并发控制

概览​ Doris 采用多版本并发控制机制&#xff08;MVCC - Multi-Version Concurrency Control&#xff09;来管理并发更新。每次数据写入操作均会分配一个写入事务&#xff0c;该事务确保数据写入的原子性&#xff08;即写入操作要么完全成功&#xff0c;要么完全失败&#xf…

数组—学习

1.基础知识 1. 高精度计算 高精度算法是处理大数&#xff08;超过64位&#xff09;的计算方法。C标准库没有直接支持大数运算&#xff0c;因此需要使用数组来模拟大数的存储和运算。 2. 全局静态数组 全局变量&#xff08;包括静态数组&#xff09;在C中会在程序启动时自动初…

GWO优化LSBooST回归预测matlab

灰狼优化算法&#xff08;Grey Wolf Optimizer&#xff0c;简称 GWO&#xff09;&#xff0c;是一种群智能优化算法&#xff0c;由澳大利亚格里菲斯大学的 Mirjalii 等人于 2014 年提出。该算法的设计灵感源自灰狼群体的捕食行为&#xff0c;核心思想是模仿灰狼社会的结构与行为…

「全网最细 + 实战源码案例」设计模式——享元模式

核心思想 享元模式&#xff08;Flyweight Pattern&#xff09;是一种结构型设计模式&#xff0c;主要用于减少程序中大量对象的内存消耗。该模式通过共享相同的数据来有效减少内存的使用&#xff0c;适用于对象非常多且可以共享一部分状态的场景。核心&#xff1a;将对象的内部…

小程序-基础加强-自定义组件

前言 这次讲自定义组件 1. 准备今天要用到的项目 2. 初步创建并使用自定义组件 这样就成功在home中引入了test组件 在json中引用了这个组件才能用这个组件 现在我们来实现全局引用组件 在app.json这样使用就可以了 3. 自定义组件的样式 发现页面里面的文本和组件里面的文…

STM32 旋转编码器

旋转编码器简介 旋转编码器&#xff1a;用来测量位置、速度或旋转方向的装置&#xff0c;当其旋转轴旋转时&#xff0c;其输出端可以输出与旋转速度和方向对应的方波信号&#xff0c;读取方波信号的频率和相位信息即可得知旋转轴的速度和方向 类型&#xff1a;机械触点式/霍尔传…

pandas(三)Series使用

一、Series基础使用 import pandasd {x:100,y:200,z:300} s1 pandas.Series(d) #将dict转化为Series print(s1)print("") l1 [1, 2, 3] l2 [a, b, c] s2 pandas.Series(l1, indexl2) #list转为Series print(s2)print("") s3 pandas.Series([11…

非根目录部署 nextjs 项目,资源文件 请求404 的问题

最近在学习next项目编写的代码放到服务器上静态资源404 先分析问题 到服务器上查看是有资源目录的是不是项目配置有问题是不是nginx配置有问题 经过排查1和2是没有问题的目前来看只有3 检查一下nginx配置 尝试着把静态资源的配置禁用 问题解决 我的next项目用的是pm2管理…