Charts繪制K線圖研究過程

示例代碼

正在抽空將KLineView封裝成pod庫移宅,方便使用...

研究過程

Charts圖表全貌
Charts-Source
蠟燭圖詳解
組合圖詳解
K線圖表需求分析
Charts問題解決方案

Charts 源碼需改動 CandleStickChartRenderer

由于_xBounds為internal修飾,所以需將Charts源碼手動拖到項目內镊屎,再新增改動小部分源碼即可。

//1.在該方法中增加繪制最大值最小值
open override func drawValues(context: CGContext)
    {
        guard
            let dataProvider = dataProvider,
            let candleData = dataProvider.candleData
            else { return }
        
        // if values are drawn
        if isDrawingValuesAllowed(dataProvider: dataProvider)
        {
            var dataSets = candleData.dataSets
            
            let phaseY = animator.phaseY
            
            var pt = CGPoint()
            
            for i in 0 ..< dataSets.count
            {
                guard let dataSet = dataSets[i] as? IBarLineScatterCandleBubbleChartDataSet
                    else { continue }
                
                if !shouldDrawValues(forDataSet: dataSet)
                {
                    continue
                }
                
                let valueFont = dataSet.valueFont
                
                guard let formatter = dataSet.valueFormatter else { continue }
                
                let trans = dataProvider.getTransformer(forAxis: dataSet.axisDependency)
                let valueToPixelMatrix = trans.valueToPixelMatrix
                
                let iconsOffset = dataSet.iconsOffset
                
                _xBounds.set(chart: dataProvider, dataSet: dataSet, animator: animator)
                
                let lineHeight = valueFont.lineHeight
                let yOffset: CGFloat = lineHeight + 5.0
                
                for j in stride(from: _xBounds.min, through: _xBounds.range + _xBounds.min, by: 1)
                {
                    guard let e = dataSet.entryForIndex(j) as? CandleChartDataEntry else { break }
                    
                    pt.x = CGFloat(e.x)
                    pt.y = CGFloat(e.high * phaseY)
                    pt = pt.applying(valueToPixelMatrix)
                    
                    if (!viewPortHandler.isInBoundsRight(pt.x))
                    {
                        break
                    }
                    
                    if (!viewPortHandler.isInBoundsLeft(pt.x) || !viewPortHandler.isInBoundsY(pt.y))
                    {
                        continue
                    }
                    
                    if dataSet.isDrawValuesEnabled
                    {
                        ChartUtils.drawText(
                            context: context,
                            text: formatter.stringForValue(
                                e.high,
                                entry: e,
                                dataSetIndex: i,
                                viewPortHandler: viewPortHandler),
                            point: CGPoint(
                                x: pt.x,
                                y: pt.y - yOffset),
                            align: .center,
                            attributes: [NSAttributedStringKey.font: valueFont, NSAttributedStringKey.foregroundColor: dataSet.valueTextColorAt(j)])
                    }
                    
                    if let icon = e.icon, dataSet.isDrawIconsEnabled
                    {
                        ChartUtils.drawImage(context: context,
                                             image: icon,
                                             x: pt.x + iconsOffset.x,
                                             y: pt.y + iconsOffset.y,
                                             size: icon.size)
                    }
                }
            }
        }
        
        //TODO:新增繪制最大值最小值
        var dataSets = candleData.dataSets
        
        let phaseY = animator.phaseY
        
        for i in 0 ..< dataSets.count {
            
            guard let dataSet = dataSets[i] as? ICandleChartDataSet , dataSets.count > 0
                else { continue }
            
            let valueFont = dataSet.valueFont
            
            guard let formatter = dataSet.valueFormatter else
            { continue }
            
            let trans = dataProvider.getTransformer(forAxis: dataSet.axisDependency)
            
            let valueToPixelMatrix = trans.valueToPixelMatrix
            
            _xBounds.set(chart: dataProvider, dataSet: dataSet, animator: animator)
            
            let lineHeight = valueFont.lineHeight
            
            let yOffset:CGFloat = lineHeight
            
            //找出所有畫布內的點
            var values = [CandleChartDataEntry]()
            
            for j in stride(from: _xBounds.min, to: _xBounds.range + _xBounds.min, by: 1){
                guard let e = dataSet.entryForIndex(j) as? CandleChartDataEntry else { break }
                
                var pt = CGPoint()
                pt.x = CGFloat(e.x)
                pt.y = CGFloat(e.high * phaseY)
                pt = pt.applying(valueToPixelMatrix)
                if (!viewPortHandler.isInBoundsRight(pt.x)){
                    break
                }
                if (!viewPortHandler.isInBoundsLeft(pt.x) || !viewPortHandler.isInBoundsY(pt.y)){
                    continue
                }
                values.append(e)
            }
            
            guard values.count > 0 else { continue }
            
            //計算最大值和最小值
            var maxEntry = values[0]
            var minEntry = values[0]
            for entry in values {
                if entry.high > maxEntry.high {
                    maxEntry = entry
                }
                if entry.low < minEntry.low {
                    minEntry = entry
                }
            }
            
            var maxPt = CGPoint(x: maxEntry.x, y: maxEntry.high)
            maxPt = maxPt.applying(valueToPixelMatrix)
            var minPt = CGPoint(x: minEntry.x, y: minEntry.low)
            minPt = minPt.applying(valueToPixelMatrix)
            
            if (maxPt.x > minPt.x){
                //大值畫左向睦优,小值畫右向
                
                var maxTest = formatter.stringForValue(maxEntry.high, entry: maxEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                maxTest = "\(maxTest) →"
                
                let maxPoint = CGPoint(x: maxPt.x, y: maxPt.y - yOffset)
                ChartUtils.drawText(context: context, text: maxTest, point: maxPoint, align: .right, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(maxEntry.x))])
                
                var minTest = formatter.stringForValue(minEntry.low, entry: minEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                minTest = "← \(minTest)"
                let minPoint = CGPoint(x: minPt.x, y: minPt.y)
                ChartUtils.drawText(context: context, text: minTest, point: minPoint, align: .left, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(minEntry.x))])
            }else{
                //大值畫右向辙浑,小值畫左向
                var maxTest = formatter.stringForValue(maxEntry.high, entry: maxEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                maxTest = "← \(maxTest)"
                let maxPoint = CGPoint(x: maxPt.x, y: maxPt.y - yOffset)
                ChartUtils.drawText(context: context, text: maxTest, point: maxPoint, align: .left, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColor])
                
                var minTest = formatter.stringForValue(minEntry.low, entry: minEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                minTest = "\(minTest) →"
                let minPoint = CGPoint(x: minPt.x, y: minPt.y)
                ChartUtils.drawText(context: context, text: minTest, point: minPoint, align: .right, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(minEntry.x))])
            }
        }
        
    }

//2.該方法中修改高亮原點為收盤價
open override func drawHighlighted(context: CGContext, indices: [Highlight])
    {
        guard
            let dataProvider = dataProvider,
            let candleData = dataProvider.candleData
            else { return }
        
        context.saveGState()
        
        for high in indices
        {
            guard
                let set = candleData.getDataSetByIndex(high.dataSetIndex) as? ICandleChartDataSet,
                set.isHighlightEnabled
                else { continue }
            
            guard let e = set.entryForXValue(high.x, closestToY: high.y) as? CandleChartDataEntry else { continue }
            
            if !isInBoundsX(entry: e, dataSet: set)
            {
                continue
            }
            
            let trans = dataProvider.getTransformer(forAxis: set.axisDependency)
            
            context.setStrokeColor(set.highlightColor.cgColor)
            context.setLineWidth(set.highlightLineWidth)
            
            if set.highlightLineDashLengths != nil
            {
                context.setLineDash(phase: set.highlightLineDashPhase, lengths: set.highlightLineDashLengths!)
            }
            else
            {
                context.setLineDash(phase: 0.0, lengths: [])
            }
            
//            let lowValue = e.low * Double(animator.phaseY)
//            let highValue = e.high * Double(animator.phaseY)
//            let y = (lowValue + highValue) / 2.0
            //TODO:修改高亮原點為收盤價
            let y = e.close
            
            let pt = trans.pixelForValues(x: e.x, y: y)
            
            high.setDraw(pt: pt)
            
            // draw the lines
            drawHighlightLines(context: context, point: pt, set: set)
        }
        
        context.restoreGState()
    }

各指標算法

// 格指標算法
class UtilAlgorithm: NSObject {
    
    /// 計算結果并賦值到model的計算屬性
    ///
    /// - Parameter models: 數(shù)據(jù)集
    static func calculationResults(models:[KLineModel]){
        
        guard models.count > 0 else {
            return
        }
        /// 分時均線
        let aveLineArray   = calculateAvePrice(datas: models)
        for i in 0 ..< models.count {
            models[i].avePrice = aveLineArray[i]
        }
        
        /// SMA
        let smaLine1Array  = calculateSMA(dayCount: KLineConst.kSMALine1Days, datas: models)
        let smaLine2Array  = calculateSMA(dayCount: KLineConst.kSMALine2Days, datas: models)
        for i in 0 ..< models.count {
            models[i].smaLine1 = smaLine1Array[i]
            models[i].smaLine2 = smaLine2Array[i]
        }
        
        /// EMA
        let emaLine1Array  = calculateEMA(dayCount: KLineConst.kEMALine1Days, datas: models)
        let emaLine2Array  = calculateEMA(dayCount: KLineConst.kEMALine2Days, datas: models)
        for i in 0 ..< models.count {
            models[i].emaLine1 = emaLine1Array[i]
            models[i].emaLine2 = emaLine2Array[i]
        }
        
        /// BOLL
        let bollSet = calculateBOLL(dayCount: KLineConst.kBOLLDayCount, k: KLineConst.kBOLL_KValue, datas: models)
        let bollArray = bollSet.0
        let ubArray   = bollSet.1
        let lbArray   = bollSet.2
        for i in 0 ..< models.count {
            models[i].boll = bollArray[i]
            models[i].ub   = ubArray[i]
            models[i].lb   = lbArray[i]
        }
        
        /// MACD
        let macdSet = calculateMACD(p1: KLineConst.kMACD_P1, p2: KLineConst.kMACD_P2, p3: KLineConst.kMACD_P3, datas: models)
        let difArray = macdSet.0
        let deaArray = macdSet.1
        let barArray = macdSet.2
        for i in 0 ..< models.count {
            models[i].dif  = difArray[i]
            models[i].dea  = deaArray[i]
            models[i].bar  = barArray[i]
        }
        
        /// KDJ
        let kdjSet = calculateKDJ(p1: KLineConst.kKDJ_P1, p2: KLineConst.kKDJ_P2, p3: KLineConst.kKDJ_p3, datas: models)
        let kArray = kdjSet.0
        let dArray = kdjSet.1
        let jArray = kdjSet.2
        for i in 0 ..< models.count {
            models[i].k = kArray[i]
            models[i].d = dArray[i]
            models[i].j = jArray[i]
        }
        
        /// RSI
        let line1Array = calculateRSI(dayCount: KLineConst.kRSILine1DayCount, datas: models)
        let line2Array = calculateRSI(dayCount: KLineConst.kRSILine2DayCount, datas: models)
        let line3Array = calculateRSI(dayCount: KLineConst.kRSILine3DayCount, datas: models)
        for i in 0 ..< models.count {
            models[i].rsiLine1 = line1Array[i]
            models[i].rsiLine2 = line2Array[i]
            models[i].rsiLine3 = line3Array[i]
        }
    }
}

//MARK: - 算法 (fileprivate)
extension UtilAlgorithm {
    
    /// 分時均線
    ///
    /// - Parameter datas: 數(shù)據(jù)集
    /// - Returns: 均值數(shù)據(jù)
    fileprivate static func calculateAvePrice(datas:[KLineModel]) -> [Double] {
        var result = [Double]()
        var totalAmount = 0.0
        var totalVolume = 0.0
        for i in 0 ..< datas.count {
            let model = datas[i]
            totalVolume += model.volume
            totalAmount += model.close * model.volume
            let avePrice = totalAmount / totalVolume
            result.append(avePrice)
        }
        return result
    }
    
    /// SMA(簡單均線)
    ///
    /// - Parameters:
    ///   - dayCount: 天數(shù)
    ///   - data: 數(shù)據(jù)集
    /// - Returns: 均值數(shù)據(jù)
    fileprivate static func calculateSMA(dayCount: Int, datas: [KLineModel]) -> [Double] {
        let dayCount = dayCount - 1
        
        var result = [Double]()
        for i in 0 ..< datas.count {
            if (i < dayCount) {
                result.append(Double.nan)
                continue
            }
            var sum: Double = 0.0
            for j in 0 ..< dayCount {
                sum = sum + datas[i - j].close
            }
            result.append(abs(sum / Double(dayCount)))
        }
        return result
    }
    
    
    /// EMA(指數(shù)移動平均數(shù))
    /// EMA(N)=2/(N+1)*(close-昨日EMA)+昨日EMA
    ///
    /// - Parameters:
    ///   - dayCount: 天數(shù)
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: 均值數(shù)據(jù)
    fileprivate static func calculateEMA(dayCount: Int, datas: [KLineModel]) -> [Double] {
        var yValues = [Double]()
        var prevEma:Double = 0.0    //前一個ema
        for (index, model) in datas.enumerated() {
            let close = model.close
            var ema: Double = 0.0
            if index > 0 {
                ema = prevEma + (close - prevEma) * 2 / (Double(dayCount) + 1)
            } else {
                ema = close
            }
            yValues.append(ema)
            prevEma = ema
        }
        return yValues
    }
    
    
    /// BOLL(布林軌道算法)
    /// 計算公式
    /// 中軌線=N日的移動平均線
    /// 上軌線=中軌線+兩倍的標準差
    /// 下軌線=中軌線-兩倍的標準差
    /// 計算過程
    /// (1)計算MA
    /// MA=N日內的收盤價之和÷N
    /// (2)計算標準差MD
    /// MD=平方根(N)日的(C-MA)的兩次方之和除以N
    /// (3)計算MB、UP微王、DN線
    /// MB=(N)日的MA
    /// UP=MB+k×MD
    /// DN=MB-k×MD
    /// (K為參數(shù),可根據(jù)股票的特性來做相應的調整品嚣,一般默認為2)
    /// - Parameters:
    ///   - dayCount: 天數(shù)
    ///   - k: 參數(shù)值
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: (BOLL,UB,LB)
    fileprivate static func calculateBOLL(dayCount:Int, k:Int=2 ,datas:[KLineModel]) -> ([Double],[Double],[Double]) {
        
        var bollValues = [Double]()
        var ubValues = [Double]()
        var lbValues = [Double]()
        
        // n天標準差
        var mdArray = calculateBOLLSTD(dayCount: dayCount, datas: datas)
        // n天均值
        var mbArray = calculateSMA(dayCount: dayCount, datas: datas)
        
        for i in 0 ..< datas.count {
            let md = mdArray[i]
            let mb = mbArray[i]
            let ub = mb + Double(k) * md
            let lb = mb - Double(k) * md
            bollValues.append(mbArray[i])
            ubValues.append(ub)
            lbValues.append(lb)
        }
        return (bollValues,ubValues,lbValues)
    }
    
    /// 計算布林線中的MA平方差
    ///
    /// - Parameters:
    ///   - dayCount: 天數(shù)
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: 結果
    fileprivate static func calculateBOLLSTD(dayCount:Int, datas:[KLineModel]) -> [Double] {
        
        var mdArray = [Double]()
        
        // n天均值
        var maArray = calculateSMA(dayCount: dayCount, datas: datas)
        
        for index in 0 ..< datas.count {
            if index + 1 >= dayCount {
                var dx:Double = 0.0
                for i in stride(from: index, through: index + 1 - dayCount, by: -1) {
                    dx += pow(datas[i].close - maArray[i], 2)
                }
                var md = dx / Double(dayCount)
                md = pow(md, 0.5)
                mdArray.append(md)
            }else{
                var dx:Double = 0.0
                for i in stride(from: index, through: 0, by: -1) {
                    dx += pow(datas[i].close - maArray[i], 2)
                }
                var md = dx / Double(dayCount)
                md = pow(md, 0.5)
                mdArray.append(md)
            }
        }
        return mdArray
    }
    

    /// MACD(平滑異同移動平均線)
    ///
    /// - Parameters:
    ///   - p1: 天數(shù)1 (12)
    ///   - p2: 天數(shù)2 (26)
    ///   - p3: 天數(shù)3 (9)
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: ([DIF],[DEA],[BAR])
    fileprivate static func calculateMACD(p1:Int, p2:Int, p3:Int, datas:[KLineModel]) -> ([Double],[Double],[Double]){
        var difArray = [Double]()
        var deaArray = [Double]()
        var barArray = [Double]()
        
        //EMA(p1)=2/(p1+1)*(C-昨日EMA)+昨日EMA炕倘;
        let p1EmaArray = calculateEMA(dayCount: p1, datas: datas)
        //EMA(p2)=2/(p2+1)*(C-昨日EMA)+昨日EMA;
        let p2EmaArray = calculateEMA(dayCount: p2, datas: datas)
        //昨日dea
        var prevDea:Double = 0.0
        
        for i in 0 ..< datas.count {
            //DIF=今日EMA(p1)- 今日EMA(p2)
            let dif = p1EmaArray[i] - p2EmaArray[i]
            //dea(p3)=2/(p3+1)*(dif-昨日dea)+昨日dea翰撑;
            let dea = prevDea + (dif - prevDea) * 2 / (Double(p3) + 1)
            //BAR=2×(DIF-DEA)
            let bar = 2 * (dif - dea)
            prevDea = dea
            
            difArray.append(dif)
            deaArray.append(dea)
            barArray.append(bar)
        }
        return (difArray,deaArray,barArray)
    }
    
    
    /// MDJ()
    ///
    /// - Parameters:
    ///   - p1: k指標周期(9)
    ///   - p2: d指標周期(3)
    ///   - p3: j指標周期(3)
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: ([K],[D],[J])
    fileprivate static func calculateKDJ(p1:Int, p2:Int, p3:Int, datas:[KLineModel]) -> ([Double],[Double],[Double]) {
        var kArray = [Double]()
        var dArray = [Double]()
        var jArray = [Double]()
        
        var prevK:Double = 50.0
        var prevD:Double = 50.0
        
        let rsvArray = calculateRSV(dayCount: p1, datas: datas)
        
        for i in 0 ..< datas.count {
            let rsv = rsvArray[i]
            let k = (2 * prevK + rsv) / 3
            let d = (2 * prevD + k) / 3
            let j = 3 * k - 2 * d
            prevK = k
            prevD = d
            kArray.append(k)
            dArray.append(d)
            jArray.append(j)
        }
        return (kArray,dArray,jArray)
    }
    
    
    /// RSV(未成熟隨機值)
    ///
    /// - Parameters:
    ///   - dayCount: 計算天數(shù)范圍
    ///   - index: 當前的索引位
    ///   - datas: 數(shù)據(jù)集
    /// - Returns: RSV集
    fileprivate static func calculateRSV(dayCount:Int, datas:[KLineModel]) -> [Double] {
        var rsvArray = [Double]()
        for i in (0 ..< datas.count) {
            var rsv   = 0.0
            let close = datas[i].close
            var high  = datas[i].high
            var low   = datas[i].low
            
            let startIndex = i < dayCount ? 0 : i - dayCount + 1
            //計算dayCount天內最高價最低價
            for j in (startIndex ..< i){
                high = datas[j].high > high ? datas[j].high : high
                low  = datas[j].low  < low  ? datas[j].low  : low
            }
            if high != low {
                rsv = (close - low) / (high - low) * 100
            }
            rsvArray.append(rsv)
        }
        return rsvArray
    }
    
    
    /// RSI(相對強弱指標)
    /// 算法:N日RSI =N日內收盤漲幅的平均值/(N日內收盤漲幅均值+N日內收盤跌幅均值) ×100
    ///
    /// - Parameters:
    ///   - dayCount: 周期天數(shù)
    ///   - datas: 數(shù)據(jù)源
    /// - Returns: 結果集
    fileprivate static func calculateRSI(dayCount:Int, datas:[KLineModel]) -> [Double]{
        
        var rsiArray = [Double]()   //rsi值
        var ratioArray = [Double]() //漲跌幅
        for model in datas {
            if model.open == 0 {
                ratioArray.append(0)
            } else {
                let ratio = (model.close - model.open)/model.open
                ratioArray.append(ratio)
            }
        }
        for i in 0 ..< ratioArray.count {
            let startIndex = i <= dayCount ? 0 : i-dayCount
            var upSum:Double   = 0.0
            var downSum:Double = 0.0
            for j in startIndex ... i {
                if ratioArray[j] >= 0 {
                    upSum += ratioArray[j] //n日收盤漲幅總和
                }else{
                    downSum += ratioArray[j] //n日收盤跌幅總和
                }
            }
            let upAve = upSum / Double(dayCount)
            let downAve = downSum / Double(dayCount)
            let rsi = upAve / (upAve - downAve) * 100
            rsiArray.append(rsi)
        }
        return rsiArray
    }
}

不用拖Charts修改代碼的辦法

#Podfile使用我以改過的Charts代碼
pod 'Charts', git:"https://github.com/iCoobin/Charts.git", tag:"3.2.3-myself"

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末激才,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子额嘿,更是在濱河造成了極大的恐慌瘸恼,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,682評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件册养,死亡現(xiàn)場離奇詭異东帅,居然都是意外死亡,警方通過查閱死者的電腦和手機球拦,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,277評論 3 395
  • 文/潘曉璐 我一進店門靠闭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人坎炼,你說我怎么就攤上這事愧膀。” “怎么了谣光?”我有些...
    開封第一講書人閱讀 165,083評論 0 355
  • 文/不壞的土叔 我叫張陵檩淋,是天一觀的道長。 經常有香客問我萄金,道長蟀悦,這世上最難降的妖魔是什么媚朦? 我笑而不...
    開封第一講書人閱讀 58,763評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮日戈,結果婚禮上询张,老公的妹妹穿的比我還像新娘。我一直安慰自己浙炼,他們只是感情好份氧,可當我...
    茶點故事閱讀 67,785評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著弯屈,像睡著了一般半火。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上季俩,一...
    開封第一講書人閱讀 51,624評論 1 305
  • 那天,我揣著相機與錄音梅掠,去河邊找鬼酌住。 笑死,一個胖子當著我的面吹牛阎抒,可吹牛的內容都是我干的酪我。 我是一名探鬼主播,決...
    沈念sama閱讀 40,358評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼且叁,長吁一口氣:“原來是場噩夢啊……” “哼都哭!你這毒婦竟也來了?” 一聲冷哼從身側響起逞带,我...
    開封第一講書人閱讀 39,261評論 0 276
  • 序言:老撾萬榮一對情侶失蹤欺矫,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后展氓,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體穆趴,經...
    沈念sama閱讀 45,722評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年遇汞,在試婚紗的時候發(fā)現(xiàn)自己被綠了未妹。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,030評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡空入,死狀恐怖络它,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情歪赢,我是刑警寧澤化戳,帶...
    沈念sama閱讀 35,737評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站埋凯,受9級特大地震影響迂烁,放射性物質發(fā)生泄漏看尼。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,360評論 3 330
  • 文/蒙蒙 一盟步、第九天 我趴在偏房一處隱蔽的房頂上張望藏斩。 院中可真熱鬧,春花似錦却盘、人聲如沸狰域。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,941評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽兆览。三九已至,卻和暖如春塞关,著一層夾襖步出監(jiān)牢的瞬間抬探,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,057評論 1 270
  • 我被黑心中介騙來泰國打工帆赢, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留小压,地道東北人。 一個月前我還...
    沈念sama閱讀 48,237評論 3 371
  • 正文 我出身青樓椰于,卻偏偏與公主長得像怠益,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子瘾婿,可洞房花燭夜當晚...
    茶點故事閱讀 44,976評論 2 355

推薦閱讀更多精彩內容

  • 本篇文章十分的長蜻牢,大概有2萬7千字左右。 一偏陪、發(fā)展史 1抢呆、人臉識別的理解: 人臉識別(Face Recogniti...
    放飛人夜閱讀 19,988評論 8 123
  • 窺探我的一切, 察覺世界的有限笛谦, 永不停下的腳步镀娶, 隨時間退減。 左眼與右眼的摩擦揪罕, 承受日與月的魅力梯码, 滴答、滴...
    雅俗共賞Y閱讀 229評論 2 7
  • 三好啰、親密和信任的能力 當企業(yè)與員工間享有共同價值和道德觀并彼此尊重時轩娶,信任便產生了。員工被鼓勵著框往,作為邁向自身個人...
    程锘gr閱讀 128評論 0 0
  • 近來上火嚴重鳄抒,想起可以泡點枸杞菊花茶喝喝。 正好我的抽屜里有一包從超市買的“中寧枸杞”,小顆许溅,不甜瓤鼻,但曬得很干。我...
    徐小恬閱讀 866評論 0 1
  • 制造什么祭犯?銷往哪里? 這是我們要思考的一個非常重要的問題滚停。 今天沃粗,我們需要實現(xiàn)一個風格的切換,從漂浮到落地键畴。 真正...
    玥貉閱讀 149評論 1 0