Android?實(shí)現(xiàn)自定義折線圖控件
前言
日前,有一個(gè)“折現(xiàn)圖”的需求,如下圖所示:
概述
如何自定義折線圖?首先將折線圖的繪制部分拆分成三部分:
- 原點(diǎn)
- X軸
- Y軸
- 折線
原點(diǎn)
第一步,需要定義出“折線圖”原點(diǎn)的位置,由圖得:
可以發(fā)現(xiàn),原點(diǎn)的位置由X軸、Y軸所占空間決定:
OriginX:Y軸寬度 OriginY:View高度 - X軸高度
計(jì)算Y軸寬度
思路:遍歷Y軸的繪制文字,用畫(huà)筆測(cè)量其最大寬度,在加上其左右Margin間距即Y軸寬度
Y軸寬度 = Y軸MarginLeft + Y軸最大文字寬度 + Y軸MariginRight
計(jì)算X軸高度
思路:獲取X軸畫(huà)筆FontMetrics,根據(jù)其top、bottom計(jì)算出X軸文字高度,在加上其上下Margin間距即X軸高度
val fontMetrics = xAxisTextPaint.fontMetrics val lineHeight = fontMetrics.bottom - fontMetrics.top xAxisHeight = lineHeight + xAxisOptions.textMarginTop + xAxisOptions.textMarginBottom
X軸
第二步,根據(jù)原點(diǎn)位置,繪制X軸軸線、網(wǎng)格線、文本
繪制軸線
繪制軸線比較簡(jiǎn)單,沿原點(diǎn)向控件右側(cè)畫(huà)一條直線即可
if (xAxisOptions.isEnableLine) { xAxisLinePaint.strokeWidth = xAxisOptions.lineWidth xAxisLinePaint.color = xAxisOptions.lineColor xAxisLinePaint.pathEffect = xAxisOptions.linePathEffect canvas.drawLine(originX, originY, width.toFloat(), originY, xAxisLinePaint) }
X軸刻度間隔
在繪制網(wǎng)格線、文本之前需要先計(jì)算X軸的刻度間隔:
這里處理的方式比較隨意,直接將X軸等分7份即可(因?yàn)樾枰@示近7天的數(shù)據(jù))
xGap = (width - originX) / 7
網(wǎng)格線、文本
網(wǎng)格線:只需要根據(jù)X軸的刻度,沿Y軸方向依次向控件頂部,畫(huà)直線即可
文本:文本需要通過(guò)畫(huà)筆,提前測(cè)量出待繪制文本的區(qū)域,然后計(jì)算出居中位置繪制即可
xAxisTexts.forEachIndexed { index, text -> val pointX = originX + index * xGap //刻度線 if (xAxisOptions.isEnableRuler) { xAxisLinePaint.strokeWidth = xAxisOptions.rulerWidth xAxisLinePaint.color = xAxisOptions.rulerColor canvas.drawLine( pointX, originY, pointX, originY - xAxisOptions.rulerHeight, xAxisLinePaint ) } //網(wǎng)格線 if (xAxisOptions.isEnableGrid) { xAxisLinePaint.strokeWidth = xAxisOptions.gridWidth xAxisLinePaint.color = xAxisOptions.gridColor xAxisLinePaint.pathEffect = xAxisOptions.gridPathEffect canvas.drawLine(pointX, originY, pointX, 0f, xAxisLinePaint) } //文本 bounds.setEmpty() xAxisTextPaint.textSize = xAxisOptions.textSize xAxisTextPaint.color = xAxisOptions.textColor xAxisTextPaint.getTextBounds(text, 0, text.length, bounds) val fm = xAxisTextPaint.fontMetrics val fontHeight = fm.bottom - fm.top val fontX = originX + index * xGap + (xGap - bounds.width()) / 2f val fontBaseline = originY + (xAxisHeight - fontHeight) / 2f - fm.top canvas.drawText(text, fontX, fontBaseline, xAxisTextPaint) }
Y軸
第三步:根據(jù)原點(diǎn)位置,繪制Y軸軸線、網(wǎng)格線、文本
計(jì)算Y軸分布
個(gè)人認(rèn)為,這里是自定義折線圖的一個(gè)難點(diǎn),這里經(jīng)過(guò)查閱資料,使用該文章中的算法:
基于JavaScript實(shí)現(xiàn)數(shù)值型坐標(biāo)軸刻度計(jì)算算法(echarts的y軸刻度計(jì)算)
/** * 根據(jù)Y軸最大值、數(shù)量獲取Y軸的標(biāo)準(zhǔn)間隔 */ private fun getYInterval(maxY: Int): Int { val yIntervalCount = yAxisCount - 1 val rawInterval = maxY / yIntervalCount.toFloat() val magicPower = floor(log10(rawInterval.toDouble())) var magic = 10.0.pow(magicPower).toFloat() if (magic == rawInterval) { magic = rawInterval } else { magic *= 10 } val rawStandardInterval = rawInterval / magic val standardInterval = getStandardInterval(rawStandardInterval) * magic return standardInterval.roundToInt() } /** * 根據(jù)初始的歸一化后的間隔,轉(zhuǎn)化為目標(biāo)的間隔 */ private fun getStandardInterval(x: Float): Float { return when { x <= 0.1f -> 0.1f x <= 0.2f -> 0.2f x <= 0.25f -> 0.25f x <= 0.5f -> 0.5f x <= 1f -> 1f else -> getStandardInterval(x / 10) * 10 } }
刻度間隔、網(wǎng)格線、文本
Y軸的軸線、網(wǎng)格線、文本剩下的內(nèi)容與X軸的處理方式幾乎一致
//繪制Y軸 //軸線 if (yAxisOptions.isEnableLine) { yAxisLinePaint.strokeWidth = yAxisOptions.lineWidth yAxisLinePaint.color = yAxisOptions.lineColor yAxisLinePaint.pathEffect = yAxisOptions.linePathEffect canvas.drawLine(originX, 0f, originX, originY, yAxisLinePaint) } yAxisTexts.forEachIndexed { index, text -> //刻度線 val pointY = originY - index * yGap if (yAxisOptions.isEnableRuler) { yAxisLinePaint.strokeWidth = yAxisOptions.rulerWidth yAxisLinePaint.color = yAxisOptions.rulerColor canvas.drawLine( originX, pointY, originX + yAxisOptions.rulerHeight, pointY, yAxisLinePaint ) } //網(wǎng)格線 if (yAxisOptions.isEnableGrid) { yAxisLinePaint.strokeWidth = yAxisOptions.gridWidth yAxisLinePaint.color = yAxisOptions.gridColor yAxisLinePaint.pathEffect = yAxisOptions.gridPathEffect canvas.drawLine(originX, pointY, width.toFloat(), pointY, yAxisLinePaint) } //文本 bounds.setEmpty() yAxisTextPaint.textSize = yAxisOptions.textSize yAxisTextPaint.color = yAxisOptions.textColor yAxisTextPaint.getTextBounds(text, 0, text.length, bounds) val fm = yAxisTextPaint.fontMetrics val x = (yAxisWidth - bounds.width()) / 2f val fontHeight = fm.bottom - fm.top val y = originY - index * yGap - fontHeight / 2f - fm.top canvas.drawText(text, x, y, yAxisTextPaint) }
折線
折線的連接,這里使用的是Path,將一個(gè)一個(gè)坐標(biāo)點(diǎn)連接,最后將Path繪制,就形成了圖中的折線圖
//繪制數(shù)據(jù) path.reset() points.forEachIndexed { index, point -> val x = originX + index * xGap + xGap / 2f val y = originY - (point.yAxis.toFloat() / yAxisMaxValue) * (yGap * (yAxisCount - 1)) if (index == 0) { path.moveTo(x, y) } else { path.lineTo(x, y) } //圓點(diǎn) circlePaint.color = dataOptions.circleColor canvas.drawCircle(x, y, dataOptions.circleRadius, circlePaint) } pathPaint.strokeWidth = dataOptions.pathWidth pathPaint.color = dataOptions.pathColor canvas.drawPath(path, pathPaint)
值得注意的是:坐標(biāo)點(diǎn)X根據(jù)間隔是相對(duì)確定的,而坐標(biāo)點(diǎn)Y則需要進(jìn)行百分比換算
代碼
折線圖LineChart
package com.vander.pool.widget.linechart import android.content.Context import android.graphics.* import android.text.TextPaint import android.util.AttributeSet import android.view.View import java.text.DecimalFormat import kotlin.math.floor import kotlin.math.log10 import kotlin.math.pow import kotlin.math.roundToInt class LineChart : View { private var options = ChartOptions() /** * X軸相關(guān) */ private val xAxisTextPaint = TextPaint(Paint.ANTI_ALIAS_FLAG) private val xAxisLinePaint = Paint(Paint.ANTI_ALIAS_FLAG) private val xAxisTexts = mutableListOf<String>() private var xAxisHeight = 0f /** * Y軸相關(guān) */ private val yAxisTextPaint = TextPaint(Paint.ANTI_ALIAS_FLAG) private val yAxisLinePaint = Paint(Paint.ANTI_ALIAS_FLAG) private val yAxisTexts = mutableListOf<String>() private var yAxisWidth = 0f private val yAxisCount = 5 private var yAxisMaxValue: Int = 0 /** * 原點(diǎn) */ private var originX = 0f private var originY = 0f private var xGap = 0f private var yGap = 0f /** * 數(shù)據(jù)相關(guān) */ private val pathPaint = Paint(Paint.ANTI_ALIAS_FLAG).also { it.style = Paint.Style.STROKE } private val circlePaint = Paint(Paint.ANTI_ALIAS_FLAG).also { it.color = Color.parseColor("#79EBCF") it.style = Paint.Style.FILL } private val points = mutableListOf<ChartBean>() private val bounds = Rect() private val path = Path() constructor(context: Context) : this(context, null) constructor(context: Context, attrs: AttributeSet?) : this(context, attrs, 0) constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : super(context, attrs, defStyleAttr) override fun onDraw(canvas: Canvas) { super.onDraw(canvas) if (points.isEmpty()) return val xAxisOptions = options.xAxisOptions val yAxisOptions = options.yAxisOptions val dataOptions = options.dataOptions //設(shè)置原點(diǎn) originX = yAxisWidth originY = height - xAxisHeight //設(shè)置X軸Y軸間隔 xGap = (width - originX) / points.size //Y軸默認(rèn)頂部會(huì)留出一半空間 yGap = originY / (yAxisCount - 1 + 0.5f) //繪制X軸 //軸線 if (xAxisOptions.isEnableLine) { xAxisLinePaint.strokeWidth = xAxisOptions.lineWidth xAxisLinePaint.color = xAxisOptions.lineColor xAxisLinePaint.pathEffect = xAxisOptions.linePathEffect canvas.drawLine(originX, originY, width.toFloat(), originY, xAxisLinePaint) } xAxisTexts.forEachIndexed { index, text -> val pointX = originX + index * xGap //刻度線 if (xAxisOptions.isEnableRuler) { xAxisLinePaint.strokeWidth = xAxisOptions.rulerWidth xAxisLinePaint.color = xAxisOptions.rulerColor canvas.drawLine( pointX, originY, pointX, originY - xAxisOptions.rulerHeight, xAxisLinePaint ) } //網(wǎng)格線 if (xAxisOptions.isEnableGrid) { xAxisLinePaint.strokeWidth = xAxisOptions.gridWidth xAxisLinePaint.color = xAxisOptions.gridColor xAxisLinePaint.pathEffect = xAxisOptions.gridPathEffect canvas.drawLine(pointX, originY, pointX, 0f, xAxisLinePaint) } //文本 bounds.setEmpty() xAxisTextPaint.textSize = xAxisOptions.textSize xAxisTextPaint.color = xAxisOptions.textColor xAxisTextPaint.getTextBounds(text, 0, text.length, bounds) val fm = xAxisTextPaint.fontMetrics val fontHeight = fm.bottom - fm.top val fontX = originX + index * xGap + (xGap - bounds.width()) / 2f val fontBaseline = originY + (xAxisHeight - fontHeight) / 2f - fm.top canvas.drawText(text, fontX, fontBaseline, xAxisTextPaint) } //繪制Y軸 //軸線 if (yAxisOptions.isEnableLine) { yAxisLinePaint.strokeWidth = yAxisOptions.lineWidth yAxisLinePaint.color = yAxisOptions.lineColor yAxisLinePaint.pathEffect = yAxisOptions.linePathEffect canvas.drawLine(originX, 0f, originX, originY, yAxisLinePaint) } yAxisTexts.forEachIndexed { index, text -> //刻度線 val pointY = originY - index * yGap if (yAxisOptions.isEnableRuler) { yAxisLinePaint.strokeWidth = yAxisOptions.rulerWidth yAxisLinePaint.color = yAxisOptions.rulerColor canvas.drawLine( originX, pointY, originX + yAxisOptions.rulerHeight, pointY, yAxisLinePaint ) } //網(wǎng)格線 if (yAxisOptions.isEnableGrid) { yAxisLinePaint.strokeWidth = yAxisOptions.gridWidth yAxisLinePaint.color = yAxisOptions.gridColor yAxisLinePaint.pathEffect = yAxisOptions.gridPathEffect canvas.drawLine(originX, pointY, width.toFloat(), pointY, yAxisLinePaint) } //文本 bounds.setEmpty() yAxisTextPaint.textSize = yAxisOptions.textSize yAxisTextPaint.color = yAxisOptions.textColor yAxisTextPaint.getTextBounds(text, 0, text.length, bounds) val fm = yAxisTextPaint.fontMetrics val x = (yAxisWidth - bounds.width()) / 2f val fontHeight = fm.bottom - fm.top val y = originY - index * yGap - fontHeight / 2f - fm.top canvas.drawText(text, x, y, yAxisTextPaint) } //繪制數(shù)據(jù) path.reset() points.forEachIndexed { index, point -> val x = originX + index * xGap + xGap / 2f val y = originY - (point.yAxis.toFloat() / yAxisMaxValue) * (yGap * (yAxisCount - 1)) if (index == 0) { path.moveTo(x, y) } else { path.lineTo(x, y) } //圓點(diǎn) circlePaint.color = dataOptions.circleColor canvas.drawCircle(x, y, dataOptions.circleRadius, circlePaint) } pathPaint.strokeWidth = dataOptions.pathWidth pathPaint.color = dataOptions.pathColor canvas.drawPath(path, pathPaint) } /** * 設(shè)置數(shù)據(jù) */ fun setData(list: List<ChartBean>) { points.clear() points.addAll(list) //設(shè)置X軸、Y軸數(shù)據(jù) setXAxisData(list) setYAxisData(list) invalidate() } /** * 設(shè)置X軸數(shù)據(jù) */ private fun setXAxisData(list: List<ChartBean>) { val xAxisOptions = options.xAxisOptions val values = list.map { it.xAxis } //X軸文本 xAxisTexts.clear() xAxisTexts.addAll(values) //X軸高度 val fontMetrics = xAxisTextPaint.fontMetrics val lineHeight = fontMetrics.bottom - fontMetrics.top xAxisHeight = lineHeight + xAxisOptions.textMarginTop + xAxisOptions.textMarginBottom } /** * 設(shè)置Y軸數(shù)據(jù) */ private fun setYAxisData(list: List<ChartBean>) { val yAxisOptions = options.yAxisOptions yAxisTextPaint.textSize = yAxisOptions.textSize yAxisTextPaint.color = yAxisOptions.textColor val texts = list.map { it.yAxis.toString() } yAxisTexts.clear() yAxisTexts.addAll(texts) //Y軸高度 val maxTextWidth = yAxisTexts.maxOf { yAxisTextPaint.measureText(it) } yAxisWidth = maxTextWidth + yAxisOptions.textMarginLeft + yAxisOptions.textMarginRight //Y軸間隔 val maxY = list.maxOf { it.yAxis } val interval = when { maxY <= 10 -> getYInterval(10) else -> getYInterval(maxY) } //Y軸文字 yAxisTexts.clear() for (index in 0..yAxisCount) { val value = index * interval yAxisTexts.add(formatNum(value)) } yAxisMaxValue = (yAxisCount - 1) * interval } /** * 格式化數(shù)值 */ private fun formatNum(num: Int): String { val absNum = Math.abs(num) return if (absNum >= 0 && absNum < 1000) { return num.toString() } else { val format = DecimalFormat("0.0") val value = num / 1000f "${format.format(value)}k" } } /** * 根據(jù)Y軸最大值、數(shù)量獲取Y軸的標(biāo)準(zhǔn)間隔 */ private fun getYInterval(maxY: Int): Int { val yIntervalCount = yAxisCount - 1 val rawInterval = maxY / yIntervalCount.toFloat() val magicPower = floor(log10(rawInterval.toDouble())) var magic = 10.0.pow(magicPower).toFloat() if (magic == rawInterval) { magic = rawInterval } else { magic *= 10 } val rawStandardInterval = rawInterval / magic val standardInterval = getStandardInterval(rawStandardInterval) * magic return standardInterval.roundToInt() } /** * 根據(jù)初始的歸一化后的間隔,轉(zhuǎn)化為目標(biāo)的間隔 */ private fun getStandardInterval(x: Float): Float { return when { x <= 0.1f -> 0.1f x <= 0.2f -> 0.2f x <= 0.25f -> 0.25f x <= 0.5f -> 0.5f x <= 1f -> 1f else -> getStandardInterval(x / 10) * 10 } } /** * 重置參數(shù) */ fun setOptions(newOptions: ChartOptions) { this.options = newOptions setData(points) } fun getOptions(): ChartOptions { return options } data class ChartBean(val xAxis: String, val yAxis: Int) }
ChartOptions配置選項(xiàng):
class ChartOptions { //X軸配置 var xAxisOptions = AxisOptions() //Y軸配置 var yAxisOptions = AxisOptions() //數(shù)據(jù)配置 var dataOptions = DataOptions() } /** * 軸線配置參數(shù) */ class AxisOptions { companion object { private const val DEFAULT_TEXT_SIZE = 20f private const val DEFAULT_TEXT_COLOR = Color.BLACK private const val DEFAULT_TEXT_MARGIN = 20 private const val DEFAULT_LINE_WIDTH = 2f private const val DEFAULT_RULER_WIDTH = 10f } /** * 文字大小 */ @FloatRange(from = 1.0) var textSize: Float = DEFAULT_TEXT_SIZE @ColorInt var textColor: Int = DEFAULT_TEXT_COLOR /** * X軸文字內(nèi)容上下兩側(cè)margin */ var textMarginTop: Int = DEFAULT_TEXT_MARGIN var textMarginBottom: Int = DEFAULT_TEXT_MARGIN /** * Y軸文字內(nèi)容左右兩側(cè)margin */ var textMarginLeft: Int = DEFAULT_TEXT_MARGIN var textMarginRight: Int = DEFAULT_TEXT_MARGIN /** * 軸線 */ var lineWidth: Float = DEFAULT_LINE_WIDTH @ColorInt var lineColor: Int = DEFAULT_TEXT_COLOR var isEnableLine = true var linePathEffect: PathEffect? = null /** * 刻度 */ var rulerWidth = DEFAULT_LINE_WIDTH var rulerHeight = DEFAULT_RULER_WIDTH @ColorInt var rulerColor = DEFAULT_TEXT_COLOR var isEnableRuler = true /** * 網(wǎng)格 */ var gridWidth: Float = DEFAULT_LINE_WIDTH @ColorInt var gridColor: Int = DEFAULT_TEXT_COLOR var gridPathEffect: PathEffect? = null var isEnableGrid = true } /** * 數(shù)據(jù)配置參數(shù) */ class DataOptions { companion object { private const val DEFAULT_PATH_WIDTH = 2f private const val DEFAULT_PATH_COLOR = Color.BLACK private const val DEFAULT_CIRCLE_RADIUS = 10f private const val DEFAULT_CIRCLE_COLOR = Color.BLACK } var pathWidth = DEFAULT_PATH_WIDTH var pathColor = DEFAULT_PATH_COLOR var circleRadius = DEFAULT_CIRCLE_RADIUS var circleColor = DEFAULT_CIRCLE_COLOR }
Demo樣式:
private fun initView() { val options = binding.chart.getOptions() //X軸 val xAxisOptions = options.xAxisOptions xAxisOptions.isEnableLine = false xAxisOptions.textColor = Color.parseColor("#999999") xAxisOptions.textSize = dpToPx(12) xAxisOptions.textMarginTop = dpToPx(12).toInt() xAxisOptions.textMarginBottom = dpToPx(12).toInt() xAxisOptions.isEnableGrid = false xAxisOptions.isEnableRuler = false //Y軸 val yAxisOptions = options.yAxisOptions yAxisOptions.isEnableLine = false yAxisOptions.textColor = Color.parseColor("#999999") yAxisOptions.textSize = dpToPx(12) yAxisOptions.textMarginLeft = dpToPx(12).toInt() yAxisOptions.textMarginRight = dpToPx(12).toInt() yAxisOptions.gridColor = Color.parseColor("#999999") yAxisOptions.gridWidth = dpToPx(0.5f) val dashLength = dpToPx(8f) yAxisOptions.gridPathEffect = DashPathEffect(floatArrayOf(dashLength, dashLength / 2), 0f) yAxisOptions.isEnableRuler = false //數(shù)據(jù) val dataOptions = options.dataOptions dataOptions.pathColor = Color.parseColor("#79EBCF") dataOptions.pathWidth = dpToPx(1f) dataOptions.circleColor = Color.parseColor("#79EBCF") dataOptions.circleRadius = dpToPx(3f) binding.chart.setOnClickListener { initChartData() } binding.toolbar.setLeftClick { finish() } } private fun initChartData() { val random = 1000 val list = mutableListOf<LineChart.ChartBean>() list.add(LineChart.ChartBean("05-01", Random.nextInt(random))) list.add(LineChart.ChartBean("05-02", Random.nextInt(random))) list.add(LineChart.ChartBean("05-03", Random.nextInt(random))) list.add(LineChart.ChartBean("05-04", Random.nextInt(random))) list.add(LineChart.ChartBean("05-05", Random.nextInt(random))) list.add(LineChart.ChartBean("05-06", Random.nextInt(random))) list.add(LineChart.ChartBean("05-07", Random.nextInt(random))) binding.chart.setData(list) //文本 val text = list.joinToString("\n") { "x : ${it.xAxis} y:${it.yAxis}" } binding.value.text = text }
到此這篇關(guān)于Android 實(shí)現(xiàn)自定義折線圖控件的文章就介紹到這了,更多相關(guān)Android折線圖控件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android如何調(diào)整線程調(diào)用棧大小
這篇文章主要介紹了Android如何調(diào)整線程調(diào)用棧大小,幫助大家更好的進(jìn)行Android開(kāi)發(fā),完善自身程序,感興趣的朋友可以了解下2020-10-10Android開(kāi)啟閃光燈的方法 Android打開(kāi)手電筒功能
這篇文章主要為大家詳細(xì)介紹了Android開(kāi)啟閃光燈的方法,Android打開(kāi)手電筒功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07Android ListView ImageView實(shí)現(xiàn)單選按鈕實(shí)例
這篇文章主要介紹了Android ListView ImageView實(shí)現(xiàn)單選按鈕的相關(guān)資料,需要的朋友可以參考下2016-10-10Android AutoCompleteTextView控件基本用法示例
這篇文章主要介紹了Android AutoCompleteTextView控件基本用法,結(jié)合實(shí)例形式分析了AutoCompleteTextView控件的功能、使用方法及相關(guān)操作步驟,需要的朋友可以參考下2016-06-06Android開(kāi)發(fā)實(shí)現(xiàn)長(zhǎng)按返回鍵彈出關(guān)機(jī)框功能
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)長(zhǎng)按返回鍵彈出關(guān)機(jī)框功能,涉及Android針對(duì)長(zhǎng)按事件的響應(yīng)與處理相關(guān)操作技巧,需要的朋友可以參考下2017-09-09Android實(shí)現(xiàn)底部導(dǎo)航欄功能(選項(xiàng)卡)
這篇文章主要介紹了Android實(shí)現(xiàn)底部導(dǎo)航欄功能,可以隨意切換不同的頁(yè)面,實(shí)現(xiàn)選項(xiàng)卡功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2015-12-12Android ViewPager與radiogroup實(shí)現(xiàn)關(guān)聯(lián)示例
本篇文章主要介紹了Android ViewPager與radiogroup實(shí)現(xiàn)關(guān)聯(lián)示例,具有一定的參考價(jià)值,有興趣的可以了解一下。2017-03-03Android studio 下JNI編程實(shí)例并生成so庫(kù)的實(shí)現(xiàn)代碼
這篇文章主要介紹了Android studio 下JNI編程實(shí)例并生成so庫(kù),需要的朋友可以參考下2017-09-09