找回密码
 立即注册

QQ登录

只需一步,快速开始

扫一扫,访问微社区

查看: 1162|回复: 17

[教学]:利用C#进行AutoCAD的二次开发(一,二,三)

[复制链接]
发表于 2003-12-21 21:11:33 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

×
          利用C#进行AutoCAD的二次开发(一)
    众所周知,对AutoCAD进行二次开发用到的主要工具有:ObjectArx,VBA,VLisp。但它们的优缺点是显而易见的:ObjectArx功能强大,编程效率高,但它的缺点是编程者必须掌握VC++,而这门语言非常的难学;VBA和VLisp虽然简单易上手,但它们对于开发大型的程序好象无能为力。那究竟有没有一种语言能结合它们的优点而尽量避免它们的缺点呢? 回答是肯定的,那就是微软新推出的21世纪编程语言C#。关于C#的详细介绍,大家可以参考有关的文章。
      C#是通过AutoCAD ActiveX 这座桥梁来和AutoCAD之间进行通讯的。AutoCAD ActiveX 使用户能够从 AutoCAD 的内部或外部以编程方式来操作 AutoCAD。它是通过将 AutoCAD 对象显示到“外部世界”来做到这一点的。一旦这些对象被显示,许多不同的编程语言和环境就可以访问它们。关于AutoCAD ActiveX 的情况,大家可以参考AutoCAD自带的帮助。
      呵呵,说了这么多无聊的,还是让我们通过一个具体的例子来说明怎样利用C#进行AutoCAD的二次开发吧。在介绍例子之前先讲一下有关的配置:
   (1)Visual Studio .net (2003和2002都可以,我用的是2002)
   (2)AutoCAD2000以上版本(我用的是2004)
     这个例子非常简单,就是通过C#建立的窗体来启动AutoCAD并画一条直线。下面是编程的具体步骤:
  (1)通过Visual Studio .net 建立一C#的windows应用程序。
   (2)在“解决方案资源管理器”中右击“引用”标签,在弹出的菜单中选择“添加引用”,在“添加引用”对话框中选择“com"选项卡下的下拉列表框中的“AutoCAD 2004 Type Library"项(注意:不同版本的CAD的数字不同),单击右边的“选择”按钮,最后单击下面的“确定”按钮。
   (3)在C#窗体中加入两个文本框和一个按钮,分别用于输入直线起点、终点的坐标和在CAD中画直线。程序的原代码放在附件中,下面主要解释一下添加的代码。
    (a)在程序的开头加入:using AutoCAD;//导入AutoCAD引用空间
     (b)在窗体的变量声明部分加入: private AcadApplication a;//声明AutoCAD对象
     (c)在窗体的构造函数部分加入:a=new AcadApplicationClass();//创建AutoCAD对象
                                  a.Visible=true;//使AutoCAD可见
     (d)在按钮的消息处理函数中加入:
        double[] startPoint=new double[3]; //声明直线起点坐标
        double[] endPoint=new double[3];//声明直线终点坐标
        string[] str=textBox1.Text.Split(',');//取出直线起点坐标输入文本框的值,文本框的输入模式为"x,y,z"
        for(int i=0;i<3;i++)
            startPoint=Convert.ToDouble(str);//将str数组转为double型            
     str=textBox2.Text.Split(',');//取出直线终点坐标输入文本框的值
        for(int i=0;i<3;i++)
            endPoint=Convert.ToDouble(str);
        a.ActiveDocument.ModelSpace.AddLine(startPoint,endPoint);//在AutoCAD中画直线
      a.Application.Update();//更新显示
    好了,简单吧,你可以试着编译一下。关于上面一些语句的用法,我会在下一讲中作详细介绍。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
 楼主| 发表于 2003-12-22 22:37:28 | 显示全部楼层

[原创]:利用C#进行AutoCAD的二次开发(二)

利用C#进行AutoCAD的二次开发(二)
                                  C#才鸟
        大家好,今天我继续给各位介绍利用C#进行AutoCAD的二次开发。在这一讲中,主要介绍上一讲例子中存在的问题。
        在上一次的例子中我是通过引用AutoCAD 2004 Type Library来进行C#与AutoCAD之间的通信,但这种方法存在两个致命的缺点。第一个缺点是每次调试程序的时候C#都要重新启动AutoCAD,如果调试的次数非常多(比如跟踪错误然后调试),那么编程的效率就很低,因为启动一次CAD还是需要较长的时间。相对于第一个缺点,第二个缺点则更要命。由于.NET本身的问题,Interop.AutoCAD.dll文件(就是通过它才实现了C#与AutoCAD之间的通信)存在着一些bug,因此虽然有时你的代码是完全正确的,但C#编译器还是抛出莫名其妙的错误。那不是完蛋了吗?我曾经有一阶段就因为这两个要命的东东差一点放弃了C#而想改学ObjectArx了,呵呵,不过还是运气好,我偶尔一次在网上看了一篇外国人写的文章,他专门介绍了这两个问题的解决办法。下面就来解决这两个问题。
        首先来看第二个难题,按以下步骤来进行:
1.        随便用Visual Studio .NET建立一个C#应用程序,然后按照上一篇文章中的设置加入AutoCAD 2004 Type Library,然后不加入任何代码,编译你的程序。
2.        在Visual Studio .NET命令行工具下用ildasm.exe(这个工具可以在Visual Studio .NET安装光盘中找到)把Interop.AutoCAD.dll文件(这个文件在步骤1中生成的项目的Bin\Release文件夹中)编译成中间语言Interop. AutoCAD.il。注意:在步骤1中建立的项目的编译设置为Release模式。
    ildasm.exe /source Interop.AutoCAD.dll /output=Interop. AutoCAD.il
    又要注意了:把ildasm.exe,Interop.AutoCAD.dll放在同一目录下。
3.在记事本中打开Interop. AutoCAD.il文件,然后查找结尾是“SinkHelper”而开头为 ".class private auto ansi sealed _DAcad“的语句,把语句中的private 改为public,然后保存Interop. AutoCAD.il文件。
4.使用ilasm.exe把Interop. AutoCAD.il文件编译为Interop.AutoCAD.dll文件,同样是在Visual Studio .NET命令行工具下进行。
    ilasm.exe  /resource=Interop.AutoCAD.res /dll Interop.AutoCAD.il /output=Interop. AutoCAD.dll
Interop.AutoCAD.res文件是在步骤1中生成的。
5.显然你不愿意每次编写应用程序时都通过上一篇文章中介绍的方法来加入Interop. AutoCAD.dll,那太麻烦了。你可以用下面的方法来让程序自动加入该文件:找到C:\Program Files\Microsoft.NET\ Primary Interop Assemblies 文件夹,然后把上面生成的
Interop.AutoCAD.dll文件拷贝进去。       
        好了,第二个问题解决了,接下来看第一个。
        在VBA中,编程者可以使用GetObject函数来获得当前活动的AutoCAD对象,但在C#中却没有,为了这个函数我几乎把MSDN给翻遍了,然后去各种C#论坛问各位高手,结果都没得到解决,呵呵,可能国内使用C#的人比较少吧。还是在老外的论坛上看到了一篇就是讲这个问题的文章才把这个难题给解决了。使用下面的语句就可以获得当前活动的AutoCAD对象了:
   (AcadApplication)Marshal.GetActiveObject("AutoCAD.Application.16")
     (对于CAD2000和CAD2002,则把16改为15)
当然以上语句必须在AutoCAD打开的情况下才能使用,否则会发生错误,对于AutoCAD没打开的情况,可以使用上一篇文章的方法来处理。完整的连接AutoCAD与C#的源程序如下所示:
using System;
using AutoCAD;
using System.Runtime.InteropServices;
namespace AcadExample
{
   public class AutoCADConnector : IDisposable
   {
      private AcadApplication _application;
      private bool _initialized;
      private bool _disposed;
      public AutoCADConnector()
      {
         try
         {
            // Upon creation, attempt to retrieve running instance
            _application = (AcadApplication)Marshal.GetActiveObject("AutoCAD.Application.16");
         }
         catch
         {
            try
            {
               // Create an instance and set flag to indicate this
               _application =  new AcadApplicationClass();
               _initialized = true;
            }
            catch
            {
               throw;
            }
         }
      }
      // If the user doesn't call Dispose, the
      // garbage collector will upon destruction
      ~AutoCADConnector()
      {
         Dispose(false);
      }

      public AcadApplication Application
      {
         get
         {
            // Return our internal instance of AutoCAD
            return _application;
         }
      }
               
      // This is the user-callable version of Dispose.
      // It calls our internal version and removes the
      // object from the garbage collector's queue.
      public void Dispose()
      {
         Dispose(true);
         GC.SuppressFinalize(this);
      }

      // This version of Dispose gets called by our
      // destructor.
      protected virtual void Dispose(bool disposing)
      {
         // If we created our AutoCAD instance, call its
         // Quit method to avoid leaking memory.
         if(!this._disposed && _initialized)
            _application.Quit();
            
         _disposed = true;         
      }
   }
}
利用Visual Studio.net 把上面的程序编译成一个类库,你就可以在以后的程序中使用它了,下面的这个例子说明了它的用法。(首先把AcadExample类库包含在项目中)
using System;
using AcadExample;
using AutoCAD;
namespace ConsoleApplication6
{
   class Class1
   {
      [STAThread]
      static void Main(string[] args)
      {
         using (AutoCADConnector connector = new AutoCADConnector())
         {
            Console.WriteLine(connector.Application.ActiveDocument.Name);
         }
         Console.ReadLine();
      }
   }
}
这个例子是在C#窗口中显示AutoCAD中当前文档的标题。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-23 09:26:57 | 显示全部楼层
能说说以后怎么调用吗?
我是能在cad中调用这个程序.
而不是用这个程序调用cad.
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

 楼主| 发表于 2003-12-23 21:49:51 | 显示全部楼层

[原创]:利用C#进行AutoCAD的二次开发(三)

利用C#进行AutoCAD的二次开发(三)
                              C#才鸟
        这一讲的主要内容是介绍AutoCAD对象模型,如果你对VBA开发AutoCAD了解的话,这部分内容应该是超简单的。
        对象是 AutoCAD ActiveX 接口的主要构造块,每一个显示的对象均精确代表一个 AutoCAD 组件。AutoCAD ActiveX 接口的主要对象有:
&#8226;        直线、圆弧、文字和标注等图形对象。
&#8226;        线型与标注样式等样式设置对象
&#8226;        图层、编组和块等组织结构对象
&#8226;        视图与视口等图形显示对象。
&#8226;        图形、AutoCAD 应用程序本身也是对象
     所有对象的根对象是AutoCAD 应用程序本身,它用AcadApplication类来表示。获得当前运行的AcadApplication对象可以使用上一讲中介绍的方法来得到。AcadApplication对象下有四个子对象构成,分别是:

&#61557;        AcadPreferences 对象,通过此对象可以访问和设置“选项”对话框中的相关选项
&#61557;        AcadDocuments对象,它表示AutoCAD 图形
&#61557;        AcadMenuBar 对象,它表示AutoCAD主菜单栏                                  (注意不是AcadMenuBars,因为应用程序只有一个主菜单栏)
&#61557;        AcadMenuGroups对象, 它表示AutoCAD 菜单和工具栏
        上面介绍了AutoCAD ActiveX 接口对象模型的大致组成,下面重点介绍AcadDocuments对象,因为大部分的编程都与它有关。首先大家看到它是复数的形式,因此它是当前打开的AutoCAD所有图形的集合,这种对象称为集合对象(呵呵,好像在讲废话)。集合对象有一些比较重要的方法和特性。其中最主要的是:Count特性用于获取集合中的对象个数(从零开始);Item 方法用于获取集合中的任何对象。关于它们的用法我会在下面的例子中介绍。而 AcadDocuments的单数形式AcadDocument表示当前打开的一个AutoCAD图形。AcadDocument对象由以下几个主要对象组成:
&#61557;        AcadModelSpace 集合和 AcadPaperSpace集合,提供对图形对象(直线、圆、圆弧等)的访问
&#61557;        AcadLayers、AcadLinetypes 和 AcadTextStyles,则提供对非图形对象(图层、线型、文本样式等)的访问
&#61557;        AcadPlot 对象提供对“打印”对话框中设置的访问,并为应用过程提供了打印图形的各种方法
&#61557;        AcadUtility 对象提供用户输入和转换函数
        图形对象的创建使用Add<Entityname>方法,比如要创建一个圆,就是用AddCircle方法,而非图形对象的创建使用Add方法。
        下面通过一个简单的例子来说明上面介绍的内容。这个例子是在AutoCAD中建立一个新的层,然后在该层中画一个红色的圆和一条绿色的直线。这是程序的源代码:(请先把上一讲中生成的interop.AutoCAD.dll 和 AutoCADExample.dll文件包含在工程中)
using System;
using AcadExample;
using AutoCAD;

namespace CircleLine
{
        /// <summary>
        /// Class1 的摘要说明。
        /// </summary>
        class Class1
        {
                /// <summary>
                /// 应用程序的主入口点。
                /// </summary>
                [STAThread]
                static void Main(string[] args)
                {
                        //
                        // TODO: 在此处添加代码以启动应用程序
                        //
                        using(AutoCADConnector connector=new AutoCADConnector())  //连接AutoCAD
                        {
                                AcadDocument aDocument=connector.Application.ActiveDocument;
       //取得当前AutoCAD活动图形对象
                                double[] center=new Double[3]{20,20,0};//设置圆心
                                double radius=20;//设置圆的半径
                                double[] startPoint=new Double[3]{0,0,0};//设置直线的起点
                                double[] endPoint=new Double[3]{40,40,0};//设置直线的终点
                                AcadLayer newLayer=aDocument.Layers.Add("CircleLine");
      //创建一个名为CircleLine的新层
                                aDocument.ActiveLayer=newLayer;//把CircleLine层设置为当前层
                                AcadCircle circle=aDocument.ModelSpace.AddCircle(center,radius);//加入圆
                                AcadLine line=aDocument.ModelSpace.AddLine(startPoint,endPoint);//加入直线
                                circle.color=ACAD_COLOR.acRed;//把圆变为红色
                                line.color=ACAD_COLOR.acGreen;//把直线变为绿色
                                connector.Application.Update();//更新显示
                                for(int i=0;i<aDocument.ModelSpace.Count;i++)
                        Console.WriteLine("这是第{0}个对象:{1}",i+1,aDocument.ModelSpace.Item(i));                //遍历当前图形
                               
                        }
                        Console.ReadLine();
                }
        }
}
好了,今天就到这里。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-25 23:15:54 | 显示全部楼层
可能我没说明白,这样编的程序只能是程序调用cad
我想把他作成菜单在cad里用,怎么做.告诉我吧.
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

 楼主| 发表于 2003-12-26 00:59:02 | 显示全部楼层
你是要在CAD里调用C#窗体吗?这好像要用Shell函数
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-26 08:33:55 | 显示全部楼层
我给做一下说明,这是利用的Automation,其实这个用什么语言都可以,VLISP/VC++/VB/C# /C++Builder/Delphi等等,所以在ARX中也可以(由于ARX用的是VC++,那就更灵活了,直接使用接口都可以),只是写的语句略有不同而已。如果非要用C#,可以做成DLL,然后在ARX中调用也是可以的,这样就可以加命令,然后加到菜单中。再说了楼主写的这个版本是For AutoCad2004的,本身就是在VS.NET环境中开发,使用VC++和VC++.NET、C#、VB.NET可以混编。楼主做的稍嫌复杂了。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

 楼主| 发表于 2003-12-26 14:29:46 | 显示全部楼层
呵呵,版主的做法太复杂,因为这要求懂ARX,其实可以使用以下的方法来实现AutoCAD调用C#程序:
   编辑acad.pgp文件来产生外部命令,要打开 PGP 文件,请在“工具”菜单上,单击“自定义”“编辑自定义文件”“程序参数”(acad.pgp)。
在文件中加入RUNAPP1,start c:\app1,0 ,,
保存文件,然后退出AutoCAD,再启动AutoCAD就可以在命令行中以RUNAPP1启动c:\app1.exe了.
你也可以使用菜单启动外部程序,可以参考AutoCAD的帮助文档。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-26 20:49:17 | 显示全部楼层
呵呵,你仔细看看我所说的,并不复杂,只是里边列了一堆东东,好象复杂。
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-28 10:19:04 | 显示全部楼层
还有一个问题,你的教程三里最后一个
遍历当前图形的那个 你的那个for是怎么用的呀,
for(int i=0;i Console.WriteLine("这是第{0}个对象:{1}",i+1,aDocument.ModelSpace.Item(i));
这个for  怎么循环呢:
我这样做为什么不行呀.
AcadEntity ssobj;
for(int j=0;j<=a.ActiveDocument.ModelSpace.Count;;j++)
ssobj = a.ActiveDocument.ModelSpace.Item(j);
ssobj.Highlight(true);
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

 楼主| 发表于 2003-12-28 13:57:03 | 显示全部楼层
楼上的,你的这一句
for(int j=0;j<=a.ActiveDocument.ModelSpace.Count;;j++),多了个;

并且条件也是错误的应为<而不是<=
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-29 09:31:16 | 显示全部楼层
;是我打错了,调试时说我的ssobj没有赋值,可是我已经
用ssobj = a.ActiveDocument.ModelSpace.Item(j);赋值了呀.
怎么还这样.我的这个赋值有问题吗?
还有用vb与用c#这样编程都是在对ActiveX编程,这样做有什么区别吗?
盼指教.
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-29 10:59:10 | 显示全部楼层
区别的只是语法
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

发表于 2003-12-29 14:21:17 | 显示全部楼层
是我弄错了.
还有问题:
我用getpoint从屏幕取了一个点,但返回值我是必须用object接收的.
但我现在要用这个值和另一个double[3]值做加减,怎么处理呀.
其实就是我要在屏幕上取一个基点,然后以他为基点做
运算,除了这样,还有什么好办法吗?
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

 楼主| 发表于 2003-12-29 20:32:19 | 显示全部楼层
bbs上聊不方便,我的qq :249178521
论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|申请友链|Archiver|手机版|小黑屋|辽公网安备|晓东CAD家园 ( 辽ICP备15016793号 )

GMT+8, 2024-9-21 10:48 , Processed in 0.353973 second(s), 60 queries , Gzip On.

Powered by Discuz! X3.5

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表