说三道四技术文摘-感悟人生的经典句子
说三道四 > 文档快照

Java程序优化的一些最佳实践

HTML文档下载 WORD文档下载 PDF文档下载
本文介绍了Java代码优化的过程,总结了优化Java程序的一些最佳实践,分析了进行优化的方法并解释了性能提升的原因。多角度分析导致性能低的原因并逐个进行优化使得程序性能得到极大提升,代码可读性、可扩展性更强。

作者通过经历的一个项目实例,介绍Java代码优化的过程,总结了优化Java程序的一些最佳实践,分析了进行优化的方法,并解释了性能提升的原因。作者从多个角度分析导致性能低的原因,并逐个进行优化,最终使得程序的性能得到极大提升,增强了代码的可读性、可扩展性。

一、衡量程序的标准
衡量一个程序是否优质,可以从多个角度进行分析。其中,最常见的衡量标准是程序的时间复杂度、空间复杂度,以及代码的可读性、可扩展性。针对程序的时间复杂度和空间复杂度,想要优化程序代码,需要对数据结构与算法有深入的理解,并且熟悉计算机系统的基本概念和原理;而针对代码的可读性和可扩展性,想要优化程序代码,需要深入理解软件架构设计,熟知并会应用合适的设计模式。

  • 首先,如今计算机系统的存储空间已经足够大了,达到了 TB 级别,因此相比于空间复杂度,时间复杂度是程序员首要考虑的因素。为了追求高性能,在某些频繁操作执行时,甚至可以考虑用空间换取时间。
  • 其次,由于受到处理器制造工艺的物理限制、成本限制,CPU主频的增长遇到了瓶颈,摩尔定律已渐渐失效,每隔18个月CPU主频即翻倍的时代已经过去了,程序员的编程方式发生了彻底的改变。在目前这个多核多处理器的时代,涌现了原生支持多线程的语言(如Java)以及分布式并行计算框架(如Hadoop)。为了使程序充分地利用多核CPU,简单地实现一个单线程的程序是远远不够的,程序员需要能够编写出并发或者并行的多线程程序。
  • 最后,大型软件系统的代码行数达到了百万级,如果没有一个设计良好的软件架构,想在已有代码的基础上进行开发,开发代价和维护成本是无法想象的。一个设计良好的软件应该具有可读性和可扩展性,遵循“开闭原则”、“依赖倒置原则”、“面向接口编程”等。
二、项目介绍

本文将介绍笔者经历的一个项目中的一部分,通过这个实例剖析代码优化的过程。下面简要地介绍该系统的相关部分。

该系统的开发语言为Java,部署在共拥有4核CPU的Linux服务器上,相关部分主要有以下操作:通过某外部系统D提供的RESTAPI获取信息,从中提取出有效的信息,并通过JDBC 存储到某数据库系统S中,供系统其他部分使用,上述操作的执行频率为每天一次,一般在午夜当系统空闲时定时执行。为了实现高可用性(HighAvailability),外部系统D部署在两台服务器上,因此需要分别从这两台服务器上获取信息并将信息插入数据库中,有效信息的条数达到了上千条,数据库插入操作次数则为有效信息条数的两倍。

图 1.系统体系结构图


为了快速地实现预期效果,在最初的实现中优先考虑了功能的实现,而未考虑系统性能和代码可读性等。系统大致有以下的实现: 
  1. REST API获取信息、数据库操作可能抛出的异常信息都被记录到日志文件中,作为调试用;
  2. 共有5次数据库连接操作,包括第一次清空数据库表,针对两个外部系统D各有两次数据库插入操作,这5个连接都是独立的,用完之后即释放;
  3. 所有的数据库插入语句都是使用java.sql.Statement类生成的;
  4. 所有的数据库插入语句,都是单条执行的,即生成一条执行一条;
  5. 整个过程都是在单个线程中执行的,包括数据库表清空操作,数据库插入操作,释放数据库连接;
  6. 数据库插入操作的JDBC代码散布在代码中。虽然这个版本的系统可以正常运行,达到了预期的效果,但是效率很低,从通过 REST API获取信息,到解析并提取有效信息,再到数据库插入操作,总共耗时100秒左右。而预期的时间应该在一分钟以内,这显然是不符合要求的。
三、代码优化过程

笔者开始分析整个过程有哪些耗时操作,以及如何提升效率,缩短程序执行的时间。通过REST API获取信息,因为是使用外部系统提供的API,所以无法在此处提升效率;取得信息之后解析出有效部分,因为是对特定格式的信息进行解析,所以也无效率提升的空间。所以,效率可以大幅度提升的空间在数据库操作部分以及程序控制部分。下面,分条叙述对耗时操作的改进方法。

1.  针对日志记录的优化

关闭日志记录,或者更改日志输出级别。因为从两台服务器的外部系统D上获取到的信息是相同的,所以数据库插入操作会抛出异常,异常信息类似于“Attemptto insert duplicate record”,这样的异常信息跟有效信息的条数相等,有上千条。这种情况是能预料到的,所以可以考虑关闭日志记录,或者不关闭日志记录而是更改日志输出级别,只记录严重级别(severe level)的错误信息,并将此类操作的日志级别调整为警告级别(warning level),这样就不会记录以上异常信息了。本项目使用的是Java 自带的日志记录类,以下配置文件将日志输出级别设置为严重级别。

清单 1. log.properties 设置日志输出级别的片段 

default file output is in user ’ s home directory. levels can be: SEVERE, WARNING, INFO, FINE, FINER, FINEST  java.util.logging.ConsoleHandler.level=SEVERE  java.util.logging.FileHandler.formatter=java.util.logging.SimpleFormatter  java.util.logging.FileHandler.append=true 

通过上述的优化之后,性能有了大幅度的提升,从原来的 100 秒左右降到了 50 秒左右。为什么仅仅不记录日志就能有如此大幅度的性能提升呢?查阅资料,发现已经有人做了相关的研究与实验。经常听到 Java 程序比 C/C++ 程序慢的言论,但是运行速度慢的真正原因是什么,估计很多人并不清楚。对于 CPU 密集型的程序(即程序中包含大量计算),Java 程序可以达到 C/C++ 程序同等级别的速度,但是对于 I/O 密集型的程序(即程序中包含大量 I/O 操作),Java 程序的速度就远远慢于 C/C++ 程序了,很大程度上是因为 C/C++ 程序能直接访问底层的存储设备。因此,不记录日志而得到大幅度性能提升的原因是,Java 程序的 I/O 操作较慢,是一个很耗时的操作。

2.  针对数据库连接的优化

共享数据库连接。共有 5 次数据库连接操作,每次都需重新建立数据库连接,数据库插入操作完成之后又立即释放了,数据库连接没有被复用。为了做到共享数据库连接,可以通过单例模式 (Singleton Pattern)获得一个相同的数据库连接,每次数据库连接操作都共享这个数据库连接。这里没有使用数据库连接池(Database Connection Pool)是因为在程序只有少量的数据库连接操作,只有在大量并发数据库连接的时候才需要连接池。

清单 2. 共享数据库连接的代码片段 

public class JdbcUtil {  private static Connection con;  // 从配置文件读取连接数据库的信息  private static String driverClassName;  private static String url;  private static String username;  private static String password;  private static String currentSchema;  private static Properties properties = new Properties();  static {  // driverClassName, url, username, password, currentSchema 等从配置文件读取,代码略去  try {  Class.forName(driverClassName);  } catch (ClassNotFoundException e) {  e.printStackTrace();  }  properties.setProperty("user", username);  properties.setProperty("password", password);  properties.setProperty("currentSchema", currentSchema);  try {  con = DriverManager.getConnection(url, properties);  } catch (SQLException e) {  e.printStackTrace();  }  }  private JdbcUtil() {}  // 获得一个单例的、共享的数据库连接  public static Connection getConnection() {  return con;  }  public static void close() throws SQLException {  if (con != null)  con.close();  }  } 

通过上述的优化之后,性能有了小幅度的提升,从 50 秒左右降到了 40 秒左右。共享数据库连接而得到的性能提升的原因是,数据库连接是一个耗时耗资源的操作,需要同远程计算机进行网络通信,建立 TCP 连接,还需要维护连接状态表,建立数据缓冲区。如果共享数据库连接,则只需要进行一次数据库连接操作,省去了多次重新建立数据库连接的时间。

3.  针对插入数据库记录的优化 - 1

使用预编译 SQL。具体做法是使用 java.sql.PreparedStatement 代替 java.sql.Statement 生成 SQL 语句。PreparedStatement 使得数据库预先编译好 SQL 语句,可以传入参数。而 Statement 生成的 SQL 语句在每次提交时,数据库都需进行编译。在执行大量类似的 SQL 语句时,可以使用 PreparedStatement 提高执行效率。使用 PreparedStatement 的另一个好处是不需要拼接 SQL 语句,代码的可读性更强。通过上述的优化之后,性能有了小幅度的提升,从 40 秒左右降到了 30~35 秒左右。

清单 3. 使用 Statement 的代码片段 

// 需要拼接 SQL 语句,执行效率不高,代码可读性不强 StringBuilder sql = new StringBuilder(); sql.append("insert into table1(column1,column2) values('"); sql.append(column1Value); sql.append("','"); sql.append(column2Value); sql.append("');"); Statement st; try {  st = con.createStatement();  st.executeUpdate(sql.toString()); } catch (SQLException e) {  e.printStackTrace(); } 

清单 4. 使用 PreparedStatement 的代码片段 

// 预编译 SQL 语句,执行效率高,可读性强 String sql = “insert into table1(column1,column2) values(?,?)”; PreparedStatement pst = con.prepareStatement(sql); pst.setString(1,column1Value); pst.setString(2,column2Value); pst.execute(); 

4.  针对插入数据库记录的优化 - 2

使用 SQL 批处理。通过 java.sql.PreparedStatement 的 addBatch 方法将 SQL 语句加入到批处理,这样在调用 execute 方法时,就会一次性地执行 SQL 批处理,而不是逐条执行。通过上述的优化之后,性能有了小幅度的提升,从 30~35 秒左右降到了 30 秒左右。

5.  针对多线程的优化

使用多线程实现并发 / 并行。清空数据库表的操作、把从 2 个外部系统 D 取得的数据插入数据库记录的操作,是相互独立的任务,可以给每个任务分配一个线程执行。清空数据库表的操作应该先于数据库插入操作完成,可以通过 java.lang.Thread 类的 join 方法控制线程执行的先后次序。在单核 CPU 时代,操作系统中某一时刻只有一个线程在运行,通过进程 / 线程调度,给每个线程分配一小段执行的时间片,可以实现多个进程 / 线程的并发(concurrent)执行。而在目前的多核多处理器背景下,操作系统中同一时刻可以有多个线程并行(parallel)执行,大大地提高了 计算速度。

清单 5. 使用多线程的代码片段 

Thread t0 = new Thread(new ClearTableTask()); Thread t1 = new Thread(new StoreServersTask(ADDRESS1)); Thread t2 = new Thread(new StoreServersTask(ADDRESS2)); try {  t0.start();  // 执行完清空操作后,再进行后续操作  t0.join();  t1.start();  t2.start();  t1.join();  t2.join(); } catch (InterruptedException e) {  e.printStackTrace(); } // 断开数据库连接 try {  JdbcUtil.close(); } catch (SQLException e) {  e.printStackTrace(); } 

通过上述的优化之后,性能有了大幅度的提升,从 30 秒左右降到了 15 秒以下,10~15 秒之间。使用多线程而得到的性能提升的原因是,系统部署所在的服务器是多核多处理器的,使用多线程,给每个任务分配一个线程执行,可以充分地利用 CPU 计算资源。

笔者试着给每个任务分配两个线程执行,希望能使程序运行得更快,但是事与愿违,此时程序运行的时间反而比每个任务分配一个线程执行的慢,大约 20 秒。笔者推测,这是因为线程较多(相对于 CPU 的内核数),使得 CPU 忙于线程的上下文切换,过多的线程上下文切换使得程序的性能反而不如之前。因此,要根据实际的硬件环境,给任务分配适量的线程执行。

6.  针对设计模式的优化

使用 DAO 模式抽象出数据访问层。原来的代码中混杂着 JDBC 操作数据库的代码,代码结构显得十分凌乱。使用 DAO 模式(Data Access Object Pattern)可以抽象出数据访问层,这样使得程序可以独立于不同的数据库,即便访问数据库的代码发生了改变,上层调用数据访问的代码无需改变。并且程 序员可以摆脱单调繁琐的数据库代码的编写,专注于业务逻辑层面的代码的开发。通过上述的优化之后,性能并未有提升,但是代码的可读性、可扩展性大大地提高 了。

清单 6. 使用 DAO 模式的代码片段 

 // DeviceDAO.java,定义了 DAO 抽象,上层的业务逻辑代码引用该接口,面向接口编程 public interface DeviceDAO {     public void add(Device device);  }  // DeviceDAOImpl.java,DAO 实现,具体的 SQL 语句和数据库操作由该类实现 public class DeviceDAOImpl implements DeviceDAO {     private Connection con;     public DeviceDAOImpl() {         // 获得数据库连接,代码略去    }  @Override  public void add(Device device) {         // 使用 PreparedStatement 进行数据库插入记录操作,代码略去    }  } 

回顾以上代码优化过程:关闭日志记录、共享数据库连接、使用预编译 SQL、使用 SQL 批处理、使用多线程实现并发 / 并行、使用 DAO 模式抽象出数据访问层,程序运行时间从最初的 100 秒左右降低到 15 秒以下,在性能上得到了很大的提升,同时也具有了更好的可读性和可扩展性。 

四、结束语

通过该项目实例,笔者深深地感到,想要写出一个性能优化、可读性可扩展性强的程序,需要对计算机系统的基本概念、原理,编程语言的特性,软件系统 架构设计都有较深入的理解。“纸上得来终觉浅,绝知此事要躬行”,想要将这些基本理论、编程技巧融会贯通,还需要不断地实践,并总结心得体会。 

英文出自:IBM DeveloperWorks

《近匠》RealSense:几行代码,为游戏添增实感交互 MDCC游戏应用征集关键词:中小团队,3D,转型,IP 共话HealthKit:能否颠覆医疗保健生态系统? Perception Neuron系统,让动作捕捉技术不再高冷 2014年DevOps实践调查报告 .NET程序性能的基本要领 百度手机卫士宣布开放三大能力 构建互联网安全生态 火速围观,AMD APU新芯片架构A68H呼之欲出! 对于测试人员而言,TDD意味着什么? 年用电910亿度,那些庞大的数据中心该用什么支撑 走近黑客们的世界,感受网络空间的心跳 VMworld 2014 Day One:Docker、数据中心、混合云新举措 Docker和Kubernetes或将加速SDN发展 前端开发框架三剑客—AngularJS VS. Backone.js VS.Ember.js 【CTO俱乐部读书会】《流程的永恒之道》辛鹏:企业的本质就是流程运营 紧扣微信研发痛点 2014微信开发者大会吸引大量外地参会者 Mobile First!jQuery UI组件集Wijmo五年最大更新 首届TCL杯HTML5智能电视—应用开发大赛火热开启! MDCC智能硬件征集关键词:智能家居、机器人、健康生活 采集颜色、自动同步,还有比这更牛叉的儿童触控笔么? AppCan移动平台:为企业移动化进程加速 学生强则国强,访天猫推荐算法大赛Top 9团队 Mozilla与三星之子——Servo特性解读 360天巡启动公测 打造新一代企业级无线安全解决方案 继Storm和Spark之后,Hortonworks添加对Kafka的支持 一周消息树:国产操作系统最快10月发布,并支持应用商店 技术团队看板方法实践的难点分析 SAP云服务加速转型:调动资源瞄准中国市场 开发测试全承包!移动应用开发工具Telerik平台 火火火火火!看HomeKit如何改变物联网和智能家居? MDCC应用与工具关键词:平台、社交、垂直、解决方案 》》怎么没有人解决这个问题呀,谁解决,我给分!!!!! 为什么快捷方式不能用? ★★★有没有像“delphi属性编辑框”一样的控件或是源码(100-1000分)★★★ 谁有RealPlay格式详细信息 500分求救! 请问大家那里有jrun 4下载 如何自己析构单文档模板? MASM32中标号的疑问 这个错误该如何改 关于在OUTLOOK中如何自创模板的问题? com系列丛书那里有卖 一个关于鼠标事件的简单问题 为什么我在JSP中能得到TEXT文本中的数据,却得不到在SELECT中的数据呢? 求救!为什么我的ACESS数据库文件会莫名其妙的达到25M,我还没输任何数据呢?而生成MDE后只有500K? 我要一个功能强大的弹出日历 Viva La Costa Rica! 我再也不看Chinese Team的比赛了! 如何做jar包呢?还有如何做bak呢?大家救救我吧。 《inside the c++ object model》问题 这句是什么意思??? 问一个Win2000Server域用户客户端登陆问题。 小问题:在线等待:) 如何删除一个ini文件中的关键项目?急 求救!为什么我的ACESS数据库文件会莫名其妙的达到25M,我还没输任何数据呢?而生成MDE后只有500K? 求教关于c++builder的图象预处理的源代码。50分!!还有金钱相送! access的问题 郁闷!!关于数据窗口一个奇怪的问题!! 如何实现断点续传 寻求《软驱监控程序》!!!!!!!!!!!!!!价格面议 用imp恢复数据库,要做什么准备呀? 刚才给分失败,重新散分! 关于SQLserver数据库和MDB数据库的句语问题!!! IE数据保密大全,如何禁止以下功能?菜单,工具栏,收藏夹,鼠标右键,PRINTSCREEN键,键盘快捷键 帮忙:DCOM 无法使用任何配置的协议与计算机 慨n 通信? 查询问题 请教bean中填写方法问题,高手帮忙啊!!!!! 昨天回家遇到打劫,被暴打一顿 本人有一个数据窗口对象中的一个字段,其为DDDW型的,可是在它的选项页中却怎么也不能显示出我想设置的名字显示出来,却总是显示保存在表中 在线求助,当场送分。 用MASM编译DOS程序经常有问题啊!? asp.net问题 有关DateDiff的问题。。。SOS 请问用什么软件制作chm帮助文档? 数据库中记录条数在增加,但字段却是空的呀??? 高分求:如何确定TreeView的SelectedItem为空? ******* 怎样调试硬中断程序? 存储过程中的一个语法问题 servlet中forward到jsp页面的显示问题 SOS:请问一个页面在不同分辨率下适应性的问题! 我在做数据库链接时,出现了编译错误,实在看不错在哪里(在线等待) rpm的升级过程中的问题,请大家关注 有关gethostbyname的问题 求有机高手:在正溴丁烷的制备中,依次加入水、浓硫酸、正丁醇、溴化钠之后,为什么整个烧瓶会变成深红色 误食体温计中的水银有什么危害 什么是十一届全国人大一次会议第五次全体会议第五次全体会议是什么意思? 加料时,先使用溴化钠与浓硫酸混合,然后加正丁醇,为什么? 判断一下题目是否正确.为什么?1.将水加热,Kw增大,pH不变.2.偏铝酸钠的水溶液经加热浓缩、蒸干灼烧后能得到其固体. 1关于牛顿第一定律,下列说法不正确的是1A、不能直接用实验来证明B、是根据实验推理出来的结论C、可以用“斜面实验”来证明D、是多人总结概括的结果2由于实际上不存在不受力的作用的 我吃了两支体温计里的水银,多久会发病? 判断下面题目是否正确,为什么? 1.下列关于催化剂的说法不正确的是( ).A.化学反应前后,催化剂的质量和化学性质都不变B.催化剂可以提高某些化学反应的速率C.催化剂可以降低某些化学反应的速率D.任何化学反应都需要催 我刚才摔了个体温计,体温计里含水银多么? 春秋战国时期的成语典故,要简洁一些 ( )1、下列说法正确的是A.欧姆表的测量范围是从零到无穷 B.用不同档次的欧姆表测量同一个电阻的阻值,误差是一样的 C.用欧姆表测电阻时,指针越接近中央时,误差越大 D.用欧姆表测电阻时 那后来你的宝宝误食体温计内的水银后有没有去医院检查过?之后有没有什么症状? 判断是否正确? 葛洲坝和三峡哪个大 电动汽车的发动机功率是多少 春秋战国时期的成语典故卧薪尝胆\一鸣惊人\纸上谈兵\胡服骑射\围魏救赵\负荆请罪\退避三舍的含义以及典故 三峡葛洲坝位置分别在长江那个位置哥们没那么麻烦就是哪个更偏上一点,不用那么具体…… 电动汽车的发动机是利用什么原理制成的? 判断③是否正确 比三峡小,葛洲坝大的水电站说一下具体地点和水电站的名字 我吃了两支体温计里的水银,多久会发病? 帮我判断是否正确 长江三峡起源及结束开垦开垦坎坎坷坷 如果勿食了体温计里的水银,会死吗 CO2和NaOH反应的离子方程式 区分日常生活中的下列各组物质,鉴别方法正确的是选项 待鉴别的物质 鉴别方法A 氯化铵和碳酸钾 加水溶解B 自来水和蒸馏水 加肥皂水,观察产生泡沫多少C 羊毛纤维和化学纤维 灼烧闻气味, 电动车的电机有声音吗 CO2分别与少量NaoH和足量的NaoH溶液反应的离子方程式?Why? 三峡中各个峡的名称的来历是什么?三峡中的西临峡为什么叫“西临峡”西临峡的一段黄猫峡为什么叫“黄猫峡”?灯影峡为什么叫“灯影峡”?崆岭峡为什么叫“崆岭峡”?米仓峡为什么叫“米 周朝的政治制度与强调平等竞争和奋斗的现代社会有什么根本差异? 电动车电机充电时会有声音?我习惯晚上22点开始充:上海地区分时记价,22点后半价用电22点后一般也应该睡了刚刚起夜,听到有轻微的呼呼声,循声而去,竞是充电器仰或电池! 氢氧化钠溶液和二氧化碳的反应离子方程式? 关于周朝的政治制度,第一:周朝分封给诸侯的封地是不是属于诸侯个人管辖?封地内的事务,如任命官员和管理百姓,是不是都由分封诸侯自行解决?周王朝中央有权干涉吗?第二:周朝的分封诸 AL+H2O+NaOH====NaALO2+H2↑(加热) 线上等啊``帮帮忙`` 印度手工棉纺织业原来很发达,棉纺品曾经大量出口英国.19实际以后,英国凭借强大的实力,采取各种手段,打击印度的棉纺织业,是指迅速衰落下去.著名的纺织业城市达卡,187年有15万人口,8年后 周朝河伯国究竟在哪?居民属于什么民族? Al+NaOH+H2O=NaAlO2+H2配平配平,写出思路,方法, 越南语翻译,别人发我,我看不懂. 为什么说修建葛洲坝是修建三峡的基石 那位专家能为我详细剖析下 “Al+NaNO3+NaOH=NaAlO2+N2+H2O”怎么配平 帮我翻译成越南语,谢谢(叫你姐姐打电话给我!)不要软件翻译,他看不懂 周朝后期政治混乱的表现 AL+NaOH+H2O=NaAlO2+3H2 这个化学方程式用用化合价的升降法来配平怎么配》? 英语翻译如图所示..这玩意有何作用 鉴别KOH溶液、稀H2SO4、CaCl2溶液下列试剂中,能把KOH溶液、稀H2SO4、CaCl2溶液鉴别出来的是1.KCl 2.K2CO3 3.稀HCl为什么 NaOH+Al=NaAlO2+H2怎么配平? 下列说法正确的是 ( ) A a的系数是0 B 1下列说法正确的是 ( )A a的系数是0 B 1/y是一次单项式C -5x的系数是5 D 0是单项式 用一种试剂即可一次鉴别出稀h2so4溶液,bacl2溶液和kno3溶液,这种试剂是A, NaOH B,Na2CO3C,AgNO3D,CaCO3 NaOH与NaHSO4的反应方程式 1.关于自由落体运动,下列说法正确的是A.物体竖直向下的运动一定是自由落体运动B.自由落体运动是初速度为零、加速度为g的竖直向下的匀加速直线运动C.物体只在重力作用下从静止开 制备1-溴丁烷时,为什么浓硫酸能除去正丁醚和正丁醇 小孩误食体温计里的水银怎么办? 1.下列有关气体压强的说法中,正确的是()A.气体分子的密度减小,气体的压强一定增大.B.气体分子的密度增大,气体的压强一定增大.C.气体分子的平均动能增大,气体的压强一定增大.D.气体分 实验室里用加热正丁醇、溴化钠和浓H2SO4的混合物的方法来制备1-溴丁烷时,还会有烯、醚和溴等副产物生成.反应结束后将反应混合物蒸馏,分离得到1-溴丁烷,已知有关的有机物的性质如下 幼儿误食体温计水银怎么办? 1.关于无形资产的后续计量,下列说法中正确的有( ). 为什么正丁醇和HBr反应制1-溴丁烷的时候浓硫酸适当稀释可以减少HBr的挥发? 磷酸二氢铵溶液与 足量氢氧化钠溶液的反应方程式及离子方程式, 氢氧化钠与碳酸钠反应的方程式.注:氢氧化钠 NaOH 碳酸钠 Na2CO3 我没打错吧,确实是这样的。
备案号:鲁ICP备13029499号-2 说三道四 www.s3d4.cn 说三道四技术文摘