最近我试着搭建了方便大家一键试玩的 Nebula Graph 中的Spark 相关的项目,今天就把它们整理成文分享给大家。而且,我趟出来了 PySpark 下的 Nebula Spark Connector 的使用方式,后边也会一并贡献到文档里。

NebulaGraph 的三个 Spark 子项目

我曾经围绕 NebulaGraph 的所有数据导入方法画过一个草图,其中已经包含了 Spark Connector,Nebula Exchange 的简单介绍。在这篇文章中我将它们和另外的 Nebula Algorithm 进行稍微深入的探讨。

注:这篇文档 也很清楚为我们列举了不同导入工具的选择。

TL;DR

  • Nebula Spark Connector 是一个 Spark Lib,它能让 Spark 应用程序能够以 dataframe 的形式从 NebulaGraph 中读取和写入图数据。
  • Nebula Exchange 建立在 Nebula Spark Connector 之上,作为一个 Spark Lib 同时可以直接被 Spark 提交 JAR 包执行的应用程序,它的设计目标是和 NebulaGraph 交换不同的数据源(对于开源版本,它是单向的:写入,而对于企业版本,它是双向的)。Nebula Exchange 支持的很多不同类型的数据源如:MySQL、 Neo4j 、 PostgreSQL 、 ClickHouse 、Hive 等。除了直接写入 NebulaGraph,它还可以选择生成 SST 文件,并将其注入 NebulaGraph,以便使用 NebulaGraph 集群之外算力帮助排序底层。
  • Nebula Algorithm,建立在 Nebula Spark Connector 和 GraphX 之上,也是一个Spark Lib 和 Spark 上的应用程序,它用来在 NebulaGraph 的图上运行常用的图算法(pagerank,LPA等)。

Nebula Spark Connector

  • 代码: https://github.com/vesoft-inc...
  • 文档: https://docs.nebula-graph.io/...
  • JAR 包: https://repo1.maven.org/maven...
  • 代码例子:example

NebulaGraph Spark Reader

为了从 NebulaGraph 中读取数据,比如读 vertex,Nebula Spark Connector 将扫描所有带有给定 TAG 的 Nebula StorageD,比如这样表示扫描 player 这个 TAG : withLabel("player") ,我们还可以指定 vertex 的属性: withReturnCols(List("name", "age")) 。

指定好所有的读 TAG 相关的配置之后,调用 spark.read.nebula.loadVerticesToDF 返回得到的就是扫描 NebulaGraph 之后转换为 Dataframe 的图数据,像这样:

def readVertex(spark: SparkSession): Unit = {LOG.info("start to read nebula vertices")val config =NebulaConnectionConfig.builder().withMetaAddress("metad0:9559,metad1:9559,metad2:9559").withConenctionRetry(2).build()val nebulaReadVertexConfig: ReadNebulaConfig = ReadNebulaConfig.builder().withSpace("basketballplayer").withLabel("player").withNoColumn(false).withReturnCols(List("name", "age")).withLimit(10).withPartitionNum(10).build()val vertex = spark.read.nebula(config, nebulaReadVertexConfig).loadVerticesToDF()vertex.printSchema()vertex.show(20)println("vertex count: " + vertex.count())}

写入的例子我这里不列出,不过,前边给出的代码示例的链接里是有更详细的例子,这里值得一提的是, Spark Connector 读数据为了满足图分析、图计算的大量数据场景 ,和大部分其他客户端非常不同,它直接绕过了 GraphD,通过扫描 MetaD 和 StorageD 获得数据,但是写入的情况则是通过 GraphD 发起 nGQL DML 语句写入的。

接下来我们来做一个上手练习吧。

上手 Nebula Spark Connector

先决条件:假设下面的程序是在一台有互联网连接的 Linux 机器上运行的,最好是预装了 Docker 和 Docker-Compose。

拉起环境

首先,让我们用Nebula-Up 部署基于容器的 NebulaGraph Core v3、Nebula Studio、Nebula Console 和 Spark、Hadoop 环境,如果还没安装好它也会尝试为我们安装 Docker 和 Docker-Compose。

# Install Core with Spark Connector, Nebula Algorithm, Nebula Exchange
curl -fsSL nebula-up.siwei.io/all-in-one.sh | bash -s -- v3 spark

你知道吗Nebula-UP 可以一键装更多东西,如果你的环境配置大一点(比如 8 GB RAM) curl -fsSL nebula-up.siwei.io/all-in-one.sh | bash 可以装更多东西,但是请注意Nebula-UP 不是为生产环境准备的。

上述边脚本执行后,让我们用Nebula-Console(Nebula Graph 的命令行客户端)来连接它。

# Connect to nebula with console
~/.nebula-up/console.sh
# Execute any queryies like
~/.nebula-up/console.sh -e "SHOW HOSTS"

加载一份数据进去,并执行一个图查询:

# Load the sample dataset
~/.nebula-up/load-basketballplayer-dataset.sh
# 等一分钟左右# Make a Graph Query the sample dataset
~/.nebula-up/console.sh -e 'USE basketballplayer; FIND ALL PATH FROM "player100" TO "team204" OVER * WHERE follow.degree is EMPTY or follow.degree >=0 YIELD path AS p;'

进入 Spark 环境

执行下面这一行,我们就可以进入到 Spark 环境:

docker exec -it spark_master_1 bash

如果我们想执行编译,可以在里边安装 mvn :

docker exec -it spark_master_1 bash
# in the container shellexport MAVEN_VERSION=3.5.4
export MAVEN_HOME=/usr/lib/mvn
export PATH=$MAVEN_HOME/bin:$PATHwget http://archive.apache.org/dist/maven/maven-3/$MAVEN_VERSION/binaries/apache-maven-$MAVEN_VERSION-bin.tar.gz && \tar -zxvf apache-maven-$MAVEN_VERSION-bin.tar.gz && \rm apache-maven-$MAVEN_VERSION-bin.tar.gz && \mv apache-maven-$MAVEN_VERSION /usr/lib/mvn

跑 Spark Connector 的例子

选项 1(推荐):通过 PySpark

  • 进入 PySpark Shell
~/.nebula-up/nebula-pyspark.sh
  • 调用 Nebula Spark Reader
# call Nebula Spark Connector Reader
df = spark.read.format("com.vesoft.nebula.connector.NebulaDataSource").option("type", "vertex").option("spaceName", "basketballplayer").option("label", "player").option("returnCols", "name,age").option("metaAddress", "metad0:9559").option("partitionNumber", 1).load()# show the dataframe with limit of 2
df.show(n=2)
  • 返回结果例子
____              __/ __/__  ___ _____/ /___\ \/ _ \/ _ `/ __/  '_//__ / .__/\_,_/_/ /_/\_\   version 2.4.5/_/Using Python version 2.7.16 (default, Jan 14 2020 07:22:06)
SparkSession available as 'spark'.
>>> df = spark.read.format(
...   "com.vesoft.nebula.connector.NebulaDataSource").option(
...     "type", "vertex").option(
...     "spaceName", "basketballplayer").option(
...     "label", "player").option(
...     "returnCols", "name,age").option(
...     "metaAddress", "metad0:9559").option(
...     "partitionNumber", 1).load()
>>> df.show(n=2)
+---------+--------------+---+
|_vertexId|          name|age|
+---------+--------------+---+
|player105|   Danny Green| 31|
|player109|Tiago Splitter| 34|
+---------+--------------+---+
only showing top 2 rows

选项 2:编译、提交示例 JAR 包

  • 先克隆 Spark Connector 和它示例代码的代码仓库,然后编译:

注意,我们使用了 master 分支,因为当下 master 分支是兼容 3.x 的,一定要保证 spark connector 和数据库内核版本是匹配的,版本对应关系参考代码仓库的 README.md 。

cd ~/.nebula-up/nebula-up/spark
git clone https://github.com/vesoft-inc/nebula-spark-connector.gitdocker exec -it spark_master_1 bash
cd /root/nebula-spark-connector
  • 替换示例项目的代码
echo > example/src/main/scala/com/vesoft/nebula/examples/connector/NebulaSparkReaderExample.scalavi example/src/main/scala/com/vesoft/nebula/examples/connector/NebulaSparkReaderExample.scala
  • 把如下的代码粘贴进去,这里边我们对前边加载的图: basketballplayer 上做了顶点和边的读操作:分别调用 readVertex 和 readEdges 。
package com.vesoft.nebula.examples.connectorimport com.facebook.thrift.protocol.TCompactProtocol
import com.vesoft.nebula.connector.connector.NebulaDataFrameReader
import com.vesoft.nebula.connector.{NebulaConnectionConfig, ReadNebulaConfig}
import org.apache.spark.SparkConf
import org.apache.spark.sql.SparkSession
import org.slf4j.LoggerFactoryobject NebulaSparkReaderExample {private val LOG = LoggerFactory.getLogger(this.getClass)def main(args: Array[String]): Unit = {val sparkConf = new SparkConfsparkConf.set("spark.serializer", "org.apache.spark.serializer.KryoSerializer").registerKryoClasses(Array[Class[_]](classOf[TCompactProtocol]))val spark = SparkSession.builder().master("local").config(sparkConf).getOrCreate()readVertex(spark)readEdges(spark)spark.close()sys.exit()}def readVertex(spark: SparkSession): Unit = {LOG.info("start to read nebula vertices")val config =NebulaConnectionConfig.builder().withMetaAddress("metad0:9559,metad1:9559,metad2:9559").withConenctionRetry(2).build()val nebulaReadVertexConfig: ReadNebulaConfig = ReadNebulaConfig.builder().withSpace("basketballplayer").withLabel("player").withNoColumn(false).withReturnCols(List("name", "age")).withLimit(10).withPartitionNum(10).build()val vertex = spark.read.nebula(config, nebulaReadVertexConfig).loadVerticesToDF()vertex.printSchema()vertex.show(20)println("vertex count: " + vertex.count())}def readEdges(spark: SparkSession): Unit = {LOG.info("start to read nebula edges")val config =NebulaConnectionConfig.builder().withMetaAddress("metad0:9559,metad1:9559,metad2:9559").withTimeout(6000).withConenctionRetry(2).build()val nebulaReadEdgeConfig: ReadNebulaConfig = ReadNebulaConfig.builder().withSpace("basketballplayer").withLabel("follow").withNoColumn(false).withReturnCols(List("degree")).withLimit(10).withPartitionNum(10).build()val edge = spark.read.nebula(config, nebulaReadEdgeConfig).loadEdgesToDF()edge.printSchema()edge.show(20)println("edge count: " + edge.count())}}
  • 然后打包成 JAR 包
/usr/lib/mvn/bin/mvn install -Dgpg.skip -Dmaven.javadoc.skip=true -Dmaven.test.skip=true
  • 最后,把它提交到 Spark 里执行:
cd example/spark/bin/spark-submit --master "local" \--class com.vesoft.nebula.examples.connector.NebulaSparkReaderExample \--driver-memory 4g target/example-3.0-SNAPSHOT.jar# 退出 spark 容器
exit
  • 成功之后,我们会得到返回结果:
22/04/19 07:29:34 INFO DAGScheduler: Job 1 finished: show at NebulaSparkReaderExample.scala:57, took 0.199310 s
+---------+------------------+---+
|_vertexId|              name|age|
+---------+------------------+---+
|player105|       Danny Green| 31|
|player109|    Tiago Splitter| 34|
|player111|        David West| 38|
|player118| Russell Westbrook| 30|
|player143|Kristaps Porzingis| 23|
|player114|     Tracy McGrady| 39|
|player150|       Luka Doncic| 20|
|player103|          Rudy Gay| 32|
|player113|   Dejounte Murray| 29|
|player121|        Chris Paul| 33|
|player128|   Carmelo Anthony| 34|
|player130|       Joel Embiid| 25|
|player136|        Steve Nash| 45|
|player108|        Boris Diaw| 36|
|player122|    DeAndre Jordan| 30|
|player123|       Ricky Rubio| 28|
|player139|        Marc Gasol| 34|
|player142|     Klay Thompson| 29|
|player145|      JaVale McGee| 31|
|player102| LaMarcus Aldridge| 33|
+---------+------------------+---+
only showing top 20 rows22/04/19 07:29:36 INFO DAGScheduler: Job 4 finished: show at NebulaSparkReaderExample.scala:82, took 0.135543 s
+---------+---------+-----+------+
|   _srcId|   _dstId|_rank|degree|
+---------+---------+-----+------+
|player105|player100|    0|    70|
|player105|player104|    0|    83|
|player105|player116|    0|    80|
|player109|player100|    0|    80|
|player109|player125|    0|    90|
|player118|player120|    0|    90|
|player118|player131|    0|    90|
|player143|player150|    0|    90|
|player114|player103|    0|    90|
|player114|player115|    0|    90|
|player114|player140|    0|    90|
|player150|player120|    0|    80|
|player150|player137|    0|    90|
|player150|player143|    0|    90|
|player103|player102|    0|    70|
|player113|player100|    0|    99|
|player113|player101|    0|    99|
|player113|player104|    0|    99|
|player113|player105|    0|    99|
|player113|player106|    0|    99|
+---------+---------+-----+------+
only showing top 20 rows

事实上,在这个代码仓库下还有更多的例子,特别是GraphX 的例子,你可以尝试自己去探索这部分。

请注意,在 GraphX 假定顶点 ID 是数字类型的,因此对于字符串类型的顶点 ID 情况,需要进行实时转换,请参考 Nebula Algorithom 中的例子 ,了解如何绕过这一问题。

Nebula Exchange

  • 代码: https://github.com/vesoft-inc...
  • 文档: https://docs.nebula-graph.com...
  • JAR 包: https://github.com/vesoft-inc...
  • 配置例子: exchange-common/src/test/resources/application.conf

Nebula Exchange 是一个 Spark Lib,也是一个可以直接提交执行的 Spark 应用,它被用来从多个数据源读取数据写入 NebulaGraph 或者输出 Nebula Graph SST 文件 。

通过 spark-submit 的方式使用 Nebula Exchange 的方法很直接:

  • 首先创建配置文件,让 Exchange 知道应该如何获取和写入数据
  • 然后用指定的配置文件调用 Exchange 包

现在,让我们用上一章中创建的相同环境做一个实际测试。

一键试玩 Exchange

先跑起来看看吧

请参考前边这一章节,先一键装好环境。

一键执行:

~/.nebula-up/nebula-exchange-example.sh

恭喜你,已经第一次执行成功一个 Exchange 的数据导入任务啦!

再看看一些细节

这个例子里,我们实际上是用 Exchange 从 CSV 文件这一其中支持的数据源中读取数据写入 NebulaGraph 集群的。这个 CSV 文件中第一列是顶点 ID,第二和第三列是 "姓名 "和 "年龄 "的属性:

player800,"Foo Bar",23
player801,"Another Name",21
  • 咱们可以进到 Spark 环境里看看
docker exec -it spark_master_1 bash
cd /root
  • 可以看到我们提交 Exchange 任务时候指定的配置文件 exchange.conf 它是一个 HOCON 格式的文件:

    .nebula
    .tags
    
{# Spark relation configspark: {app: {name: Nebula Exchange}master:localdriver: {cores: 1maxResultSize: 1G}executor: {memory: 1G}cores:{max: 16}}# Nebula Graph relation confignebula: {address:{graph:["graphd:9669"]meta:["metad0:9559", "metad1:9559", "metad2:9559"]}user: rootpswd: nebulaspace: basketballplayer# parameters for SST import, not requiredpath:{local:"/tmp"remote:"/sst"hdfs.namenode: "hdfs://localhost:9000"}# nebula client connection parametersconnection {# socket connect & execute timeout, unit: millisecondtimeout: 30000}error: {# max number of failures, if the number of failures is bigger than max, then exit the application.max: 32# failed import job will be recorded in output pathoutput: /tmp/errors}# use google's RateLimiter to limit the requests send to NebulaGraphrate: {# the stable throughput of RateLimiterlimit: 1024# Acquires a permit from RateLimiter, unit: MILLISECONDS# if it can't be obtained within the specified timeout, then give up the request.timeout: 1000}}# Processing tags# There are tag config examples for different dataSources.tags: [# HDFS csv# Import mode is client, just change type.sink to sst if you want to use client import mode.{name: playertype: {source: csvsink: client}path: "file:///root/player.csv"# if your csv file has no header, then use _c0,_c1,_c2,.. to indicate fieldsfields: [_c1, _c2]nebula.fields: [name, age]vertex: {field:_c0}separator: ","header: falsebatch: 256partition: 32}]
}
  • 我们应该能看到那个 CSV 数据源和这个配置文件都在同一个目录下了:
bash-5.0# ls -l
total 24
drwxrwxr-x    2 1000     1000          4096 Jun  1 04:26 download
-rw-rw-r--    1 1000     1000          1908 Jun  1 04:23 exchange.conf
-rw-rw-r--    1 1000     1000          2593 Jun  1 04:23 hadoop.env
drwxrwxr-x    7 1000     1000          4096 Jun  6 03:27 nebula-spark-connector
-rw-rw-r--    1 1000     1000            51 Jun  1 04:23 player.csv
  • 然后,实际上我们可以手动再次提交一下这个 Exchange 任务
/spark/bin/spark-submit --master local \--class com.vesoft.nebula.exchange.Exchange download/nebula-exchange.jar \-c exchange.conf
  • 部分返回结果
22/06/06 03:56:26 INFO Exchange$: Processing Tag player
22/06/06 03:56:26 INFO Exchange$: field keys: _c1, _c2
22/06/06 03:56:26 INFO Exchange$: nebula keys: name, age
22/06/06 03:56:26 INFO Exchange$: Loading CSV files from file:///root/player.csv
...
22/06/06 03:56:41 INFO Exchange$: import for tag player cost time: 3.35 s
22/06/06 03:56:41 INFO Exchange$: Client-Import: batchSuccess.player: 2
22/06/06 03:56:41 INFO Exchange$: Client-Import: batchFailure.player: 0
...

更多的数据源,请参考文档和配置的例子。

关于 Exchange 输出 SST 文件的实践,你可以参考文档和我的旧文 Nebula Exchange SST 2.x实践指南 。

Nebula Algorithm

  • 代码仓库: https://github.com/vesoft-inc...
  • 文档: https://docs.nebula-graph.com...
  • JAR 包: https://repo1.maven.org/maven...
  • 示例代码: example/src/main/scala/com/vesoft/nebula/algorithm

通过 spark-submit 提交任务

我在这个代码仓库里给出了例子,今天我们借助 Nebula-UP 可以更方便体验它。

参考前边这一章节,先一键装好环境。

在如上通过 Nebula-UP 的 Spark 模式部署了需要的依赖之后

  • 加载LiveJournal 数据集
~/.nebula-up/load-LiveJournal-dataset.sh
  • 在 LiveJournal 数据集上执行一个 PageRank 算法,结果输出到 CSV 文件中
~/.nebula-up/nebula-algo-pagerank-example.sh
  • 检查输出结果:
docker exec -it spark_master_1 bashhead /output/part*000.csv
_id,pagerank
637100,0.9268620883822242
108150,1.1855749056722755
957460,0.923720299211093
257320,0.9967932799358413

配置文件解读

完整文件在这里,这里,我们介绍一下主要的字段:

  • .data 指定了源是 Nebula,表示从集群获取图数据,输出 sink 是 csv ,表示写到本地文件里。
data: {# data source. optional of nebula,csv,jsonsource: nebula# data sink, means the algorithm result will be write into this sink. optional of nebula,csv,textsink: csv# if your algorithm needs weighthasWeight: false}
  • .nebula.read 规定了读 NebulaGraph 集群的对应关系,这里是读取所有 edge type: follow 的边数据为一整张图
nebula: {# algo's data source from Nebula. If data.source is nebula, then this nebula.read config can be valid.read: {# Nebula metad server address, multiple addresses are split by English commametaAddress: "metad0:9559"# Nebula spacespace: livejournal# Nebula edge types, multiple labels means that data from multiple edges will union togetherlabels: ["follow"]# Nebula edge property name for each edge type, this property will be as weight col for algorithm.# Make sure the weightCols are corresponding to labels.weightCols: []}
  • .algorithm 里配置了我们要调用的算法,和算法的配置
  algorithm: {executeAlgo: pagerank# PageRank parameterpagerank: {maxIter: 10resetProb: 0.15  # default 0.15}

作为一个库在 Spark 中调用 Nebula Algoritm

请注意另一方面,我们可以将 Nebula Algoritm 作为一个库调用,它的好处在于:

  • 对算法的输出格式有更多的控制/定制功能
  • 可以对非数字 ID 的情况进行转换,见这里

这里我先不给出例子了,如果大家感兴趣可以给 Nebula-UP 提需求,我也会增加相应的例子。

一文了解 NebulaGraph 上的 Spark 项目相关推荐

  1. linux部署项目文档,Linux上部署综合项目专业资料.doc

    LINUX上布署项目过程 1 准备工作 1.下载安装Xshell.Xftp(用于远程连接Linux主机,具体自行baidu) 2.官网下载Linux下tomcat安装包,我用是: apache-tom ...

  2. 在Eclipse上调试Spark项目

    一.standlone模式 1.1. 设置Master sparkConf.setMaster("spark://chb0-179004:7077"); 1.2.设置Spark依赖 ...

  3. linux npm安装_怎样在Linux上开发vue项目

    怎样在Linux上开发vue项目 一.开发环境搭建:安装node.js环境以及vue cli工具 (1)安装node.js 从官网下载对应的二进制压缩包,如下图: 解压到程序安装目录 xz -d no ...

  4. SharePoint 服务器端对象模型操作文档库(上传/授权/查看权限)

    来源于:http://www.cnblogs.com/jianyus/p/3258863.html 简介:上传文档到文档库,并对项目级授权,查看项目级权限方法         //在列表根目录下创建文 ...

  5. 在阿里云Serverless K8S集群上部署Spark任务并连接OSS(详细步骤)

    在阿里云ASK集群上部署Spark任务并连接OSS 简介 ASK是阿里云的一个产品,属于Serverless Kubernetes 集群,这次实验是要在ASK集群上运行Spark计算任务(以WordC ...

  6. spark项目实战:电商分析平台之各个范围Session步长、访问时长占比统计(需求一)

    spark项目实战:电商分析平台之各个范围Session步长.访问时长占比统计(需求一) 项目基本信息,架构,需要一览 各个范围Session步长.访问时长占比统计概述 各个范围Session步长.访 ...

  7. spark项目实战:电商分析平台之项目概述

    spark项目实战:电商分析平台之项目概述 目录 项目概述 程序架构分析 需求解析 初始代码和完成代码存放在github上面 1. 项目概述 在访问电商网站时,我们的一些访问行为会产生相应的埋点日志( ...

  8. 设置maven 参数调休_IDEA 使用 Maven构建Spark项目

    上一篇讲了普通构建spark项目 这次分享用Maven构建Spark项目,中间遇到了很多坑!其根本原因是Scala 与 Spark的版本不一致! 本次环境: Java1.8 Scala 2.11.8 ...

  9. 科普文:服务器上如何 Node 多版本共存 #31

    科普文:服务器上如何 Node 多版本共存 #31 背景 很多公司的服务器环境没有做隔离,就是全局安装一个 Node.js Runtime,一般很少升级. nvs / nvm 等可以用来切换版本,但无 ...

最新文章

  1. springboot 优雅停机_新姿势,Spring Boot 2.3.0 如何优雅停机?
  2. python中常用的语句元素
  3. javascript DOM基础(一)
  4. 生成静态页面的五种方案(转)
  5. java測試動態方法_java反射学习
  6. YbOJ-网格序列【拉格朗日插值】
  7. java 根据类名示例化类_Java即时类| from()方法与示例
  8. LeetCode 91. 解码方法
  9. 大数据自学1-CentOS 下安装CDH及Cloudera Manager
  10. SpringBoot02——A Simple SpringBoot ProjectHot Deployment
  11. Spring入门(1)
  12. 【Mybatis】maven配置pom.xml时找不到依赖项(已解决)
  13. 学习笔记一 线性代数
  14. android 平方常规字体,android - Android将Roboto字体设置为粗体,斜体,常规,…(类似于自定义字体系列) - 堆栈内存溢出...
  15. 机器学习基石——作业2解答
  16. 哈罗单车获10亿元D2轮融资 共享单车仍有精彩可期待
  17. ts 手动实现 ts 中的map
  18. GD图片处理——缩放、剪切、相框、水印、锐化、旋转、翻转、透明度、反色
  19. scrapy爬取斗鱼图片并且重命名后保存
  20. 【信息保护论】信息保护与密码学

热门文章

  1. 无意间看见老板和秘书那点事,我坐等被辞退的心里准备《程序员创业记》
  2. 详解prototype
  3. Web 3D引擎闲谈
  4. 新视野大学英语4期末考试真题—集大内部期末机考原题—考试所有题目直接背答案—稳过
  5. Correct Bracket Sequence Editor
  6. 初次使用hammer.js做卡片滑动效果
  7. 笛卡尔乘积与数据库连接(join)
  8. c语言如何就是联立方程组,准备出国读研的兄弟们:我们真的不如美国人勤奋---转自网络(转载)...
  9. python解包dump,tcpdump抓包及tshark解包方法介绍
  10. JAVA开发(关于写代码与数学)